qemush/bin/qemush

192 lines
4.2 KiB
Text
Raw Normal View History

2023-12-04 12:00:53 +01:00
#!/bin/bash
2024-01-25 14:44:56 +01:00
# version=0.3.0
2023-12-04 12:00:53 +01:00
# Function to re-exec the script as another user via sudo (only if needed)
2023-12-04 12:00:53 +01:00
exec_as() {
local user
user="$1"
shift
if [ "$(whoami)" != "$user" ]; then
exec sudo -E -H -u "$user" -- "$0" "$@"
2023-12-04 12:00:53 +01:00
fi
}
# Exec the script as qemu
2023-12-04 12:00:53 +01:00
exec_as qemu "$@"
# Environment
PATH="${HOME}/launchers:${HOME}/bin:${PATH}"
2023-12-04 12:00:53 +01:00
EDITOR="${EDITOR:-nvim}"
export QEMUSH_NAME
# Aliases
ls='ls --color=auto'
# Set a restrictive umask to make sure qemu user files are private
2023-12-04 12:00:53 +01:00
umask 027
# Function to print a colored error
2023-12-04 12:00:53 +01:00
perror() {
>&2 printf '\033[1;31mKO:\033[0m \033[1m%s\033[0m\n' "$*"
2023-12-04 12:00:53 +01:00
}
# Function to show the usage
2023-12-04 12:00:53 +01:00
public_help() {
local name
name=$(basename "$0")
2023-12-04 12:00:53 +01:00
exec cat << EOF
${name}: usage:
2024-01-25 14:41:20 +01:00
${name} running - (default behaviour) list running VMs
${name} start <VM name> - start a VM
2024-01-25 14:42:59 +01:00
${name} attach <VM name> - attach to the VM monitor socket
${name} ls - list available VMs
${name} edit <VM name> - edit VM launching script
${name} rm <VM name> - delete launch script
${name} diskls - list available disk images
${name} diskadd <disk name> <size> - create a disk image
${name} diskrm <disk name> - delete disk image
${name} shell - start a shell as user qemu
${name} help - show this help
${name} add <path to script> [<VM name>] - add a launching script
${name} do <command> - run shell input as user qemu
2023-12-04 12:00:53 +01:00
EOF
}
# Function to throw an invalid usage error (skill issue)
2023-12-04 12:00:53 +01:00
error_usage() {
perror "Invalid usage"
>&2 public_help
return 1
}
2024-01-21 01:51:07 +01:00
# Function to start a virtual machine
2023-12-04 12:00:53 +01:00
public_start() {
QEMUSH_NAME="$1"
2023-12-04 12:00:53 +01:00
set -- "$@" \
-monitor "unix:$(pathof socket),server,nowait" \
-daemonize
if ! "$@"; then
perror "error launching virtual machine \"${QEMUSH_NAME}\""
return 2
fi
2023-12-04 12:00:53 +01:00
}
# Attach to a running virtual machine output, the latest opened if no
# argument is provided
2024-01-25 14:42:59 +01:00
public_attach() {
2024-01-21 01:51:07 +01:00
QEMUSH_NAME="$1"
shift
2024-01-21 00:23:33 +01:00
exec socat -,echo=0,icanon=0 "UNIX-CONNECT:$(pathof socket)"
2023-12-04 12:00:53 +01:00
}
# List running virtual machines
2024-01-25 14:41:20 +01:00
public_running() {
cd || return
2024-01-21 01:51:07 +01:00
echo "Running machines:"
set -- $ls -t sockets "$@"
exec "$@"
2023-12-04 12:00:53 +01:00
}
# List available virtual machines entrypoints
2023-12-04 12:00:53 +01:00
public_ls() {
cd || return
2023-12-04 12:00:53 +01:00
echo "Available machines:"
set -- $ls launchers "$@"
exec "$@"
2023-12-04 12:00:53 +01:00
}
# Create a copy-on-write disk for a virtual machine
2023-12-04 12:00:53 +01:00
public_diskadd() {
QEMUSH_NAME="$1"
shift
exec qemu-img create -f qcow2 "$(pathof disk)" "$1"
2023-12-04 12:00:53 +01:00
}
# Delete a disk
2023-12-04 12:00:53 +01:00
public_diskrm() {
for disk in "$@"; do
QEMUSH_NAME="$disk"
rm -vi -- "$(pathof disk)"
done
2023-12-04 12:00:53 +01:00
}
# List available disks
2023-12-04 12:00:53 +01:00
public_diskls() {
cd || return
2023-12-04 12:00:53 +01:00
echo "Available disks:"
set -- $ls disks "$@"
exec "$@"
2023-12-04 12:00:53 +01:00
}
# Edit a virtual machine entrypoint with a text editor
2023-12-04 12:00:53 +01:00
public_edit() {
cd || return
local file="launchers/${1}"
2023-12-04 12:00:53 +01:00
"$EDITOR" "$file"
[ -f "$file" ] && exec chmod u+x "$file"
2023-12-04 12:00:53 +01:00
}
# Delete a virtual machine entrypoint
2023-12-04 12:00:53 +01:00
public_rm() {
cd ~/launchers || return
exec rm -vi -- "$@"
2023-12-04 12:00:53 +01:00
}
# Invoke bash as qemu user in its home directory
2023-12-04 12:00:53 +01:00
public_shell() {
cd || return
set -- bash -i "$@"
exec "$@"
2023-12-04 12:00:53 +01:00
}
# Output the content of an entrypoint, with coloration if on a virtual
# terminal
2023-12-04 12:00:53 +01:00
public_cat() {
cd ~/launchers || return
cat -- "$@"
2023-12-04 12:00:53 +01:00
}
# Copy a file in entrypoints folder
2023-12-04 23:19:24 +01:00
public_add() {
trap return EXIT
set -e
local name
if [ -n "$2" ]; then
name="$2"
else
name=$(basename "$1")
fi
name="${HOME}/launchers/${name}"
cp -vi -- "$1" "$name"
chmod 740 "$name"
2023-12-04 23:19:24 +01:00
set +e
trap - EXIT
}
# Run shell commands as qemu
2023-12-08 11:49:32 +01:00
public_do() {
2023-12-08 11:01:33 +01:00
exec sh -c "$*"
}
# Retrieve user requested function
2023-12-04 12:00:53 +01:00
function="$1"
shift
# Defauts to `active` if no function is supplied; else checks for a public
# function named after the argument; else fails
2023-12-04 12:00:53 +01:00
if [ -z "$function" ]; then
2024-01-25 14:41:20 +01:00
public_running
2023-12-04 12:00:53 +01:00
elif declare -F | cut -d \ -f 3- | grep '^public_' | sed 's/^public_//' | grep -q "^${function}$"; then
"public_${function}" "$@"
else
error_usage
fi