Update jlmkr.sh
This commit is contained in:
parent
e8a5076809
commit
8e5140ec21
|
@ -6,18 +6,11 @@ shopt -s nullglob
|
||||||
ABSOLUTE_SCRIPT_PATH="$(realpath "${BASH_SOURCE[0]}")"
|
ABSOLUTE_SCRIPT_PATH="$(realpath "${BASH_SOURCE[0]}")"
|
||||||
SCRIPT_NAME=$(basename "${ABSOLUTE_SCRIPT_PATH}")
|
SCRIPT_NAME=$(basename "${ABSOLUTE_SCRIPT_PATH}")
|
||||||
SCRIPT_DIR_PATH="$(dirname "${ABSOLUTE_SCRIPT_PATH}")"
|
SCRIPT_DIR_PATH="$(dirname "${ABSOLUTE_SCRIPT_PATH}")"
|
||||||
LXC_DIR_PATH='.lxc'
|
|
||||||
LXC_CACHE_PATH="${LXC_DIR_PATH}/cache"
|
CREATE_DONE=
|
||||||
LXC_DOWNLOAD_SCRIPT_PATH="${LXC_DIR_PATH}/lxc-download.sh"
|
|
||||||
ARCH="$(dpkg --print-architecture)"
|
|
||||||
JAILS_DIR_PATH='jails'
|
JAILS_DIR_PATH='jails'
|
||||||
JAIL_ROOTFS_NAME='rootfs'
|
JAIL_ROOTFS_NAME='rootfs'
|
||||||
JAIL_NAME=
|
|
||||||
JAIL_PATH=
|
JAIL_PATH=
|
||||||
DISTRO=
|
|
||||||
RELEASE=
|
|
||||||
SYSTEMD_RUN_UNIT_NAME=
|
|
||||||
DONE=0
|
|
||||||
|
|
||||||
USAGE="WARNING: EXPERIMENTAL AND WORK IN PROGRESS, USE ONLY FOR TESTING!
|
USAGE="WARNING: EXPERIMENTAL AND WORK IN PROGRESS, USE ONLY FOR TESTING!
|
||||||
|
|
||||||
|
@ -27,44 +20,274 @@ TODO: complete writing usage
|
||||||
"
|
"
|
||||||
|
|
||||||
fail() {
|
fail() {
|
||||||
echo -e "$1" >&2 && exit 1
|
echo -e "$1" >&2 && exit 1
|
||||||
}
|
}
|
||||||
|
|
||||||
[[ $UID -ne 0 ]] && echo "${USAGE}" && fail "Run this script as root..."
|
[[ $UID -ne 0 ]] && echo "${USAGE}" && fail "Run this script as root..."
|
||||||
|
|
||||||
err() {
|
err() {
|
||||||
# https://unix.stackexchange.com/a/504829/477308
|
# https://unix.stackexchange.com/a/504829/477308
|
||||||
echo 'Error occurred:'
|
echo 'Error occurred:'
|
||||||
awk 'NR>L-4 && NR<L+4 { printf "%-5d%3s%s\n",NR,(NR==L?">>>":""),$0 }' L="${1}" "${0}"
|
awk 'NR>L-4 && NR<L+4 { printf "%-5d%3s%s\n",NR,(NR==L?">>>":""),$0 }' L="${1}" "${ABSOLUTE_SCRIPT_PATH}"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Trap errors
|
||||||
|
trap 'err $LINENO' ERR
|
||||||
|
|
||||||
|
#########################
|
||||||
|
# START RUN FUNCTIONALITY
|
||||||
|
#########################
|
||||||
|
|
||||||
|
run_jail() (
|
||||||
|
# Create a sub-shell to source the conf file
|
||||||
|
|
||||||
|
# Define as global vars: they are also global vars in the sourced start script
|
||||||
|
DOCKER_COMPATIBLE=
|
||||||
|
GPU_PASSTHROUGH=
|
||||||
|
SYSTEMD_RUN_ADDITIONAL_ARGS=()
|
||||||
|
SYSTEMD_NSPAWN_ADDITIONAL_ARGS=()
|
||||||
|
|
||||||
|
echo 'Load the config'
|
||||||
|
# shellcheck disable=SC1090
|
||||||
|
. "${1}"
|
||||||
|
echo 'Config loaded'
|
||||||
|
|
||||||
|
set -eEuo pipefail
|
||||||
|
if [[ "$(type -t start)" == 'function' ]]; then
|
||||||
|
if [[ "${DOCKER_COMPATIBLE}" -eq 1 ]]; then
|
||||||
|
# Enable ip forwarding on the host (docker needs it)
|
||||||
|
echo 1 >/proc/sys/net/ipv4/ip_forward
|
||||||
|
# To properly run docker inside the jail, we need to lift restrictions
|
||||||
|
# Without DevicePolicy=auto images with device nodes may not be pulled
|
||||||
|
# https://github.com/kinvolk/kube-spawn/pull/328
|
||||||
|
SYSTEMD_RUN_ADDITIONAL_ARGS+=(--setenv=SYSTEMD_SECCOMP=0 --property=DevicePolicy=auto)
|
||||||
|
# Add additional flags required for docker
|
||||||
|
SYSTEMD_NSPAWN_ADDITIONAL_ARGS+=(--capability=all --system-call-filter='add_key keyctl bpf')
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [[ "${GPU_PASSTHROUGH}" -eq 1 ]]; then
|
||||||
|
SYSTEMD_NSPAWN_ADDITIONAL_ARGS+=(--property=DeviceAllow='char-drm rw')
|
||||||
|
|
||||||
|
# Detect intel GPU device and if present add bind flag
|
||||||
|
[[ -d /dev/dri ]] && SYSTEMD_NSPAWN_ADDITIONAL_ARGS+=(--bind=/dev/dri)
|
||||||
|
|
||||||
|
# TODO: add bind mount flags in case of nvidia GPU passthrough
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "Starting jail..."
|
||||||
|
start
|
||||||
|
else
|
||||||
|
echo "Can't call the start function since start.sh didn't contain one..."
|
||||||
|
fi
|
||||||
|
)
|
||||||
|
|
||||||
|
############################
|
||||||
|
# START CREATE FUNCTIONALITY
|
||||||
|
############################
|
||||||
|
|
||||||
cleanup() {
|
cleanup() {
|
||||||
# If the script didn't complete (not DONE) then
|
# If the script didn't complete (not CREATE_DONE) then
|
||||||
# remove the JAIL_PATH (if set and a directory)
|
# remove the JAIL_PATH (if set and a directory)
|
||||||
if [[ "${DONE}" -ne 1 && -n "${JAIL_PATH}" && -d "${JAIL_PATH}" &&
|
if [[ "${CREATE_DONE}" -ne 1 && -n "${JAIL_PATH}" && -d "${JAIL_PATH}" &&
|
||||||
"${JAIL_PATH}" != "${JAILS_DIR_PATH}" && "${JAIL_PATH}" != "/" ]]; then
|
"${JAIL_PATH}" != "${JAILS_DIR_PATH}" && "${JAIL_PATH}" != "/" ]]; then
|
||||||
echo -e "\n\nCleaning up: ${JAIL_PATH}\n"
|
echo -e "\n\nCleaning up: ${JAIL_PATH}\n"
|
||||||
rm -rf "${JAIL_PATH}"
|
rm -rf "${JAIL_PATH}"
|
||||||
fi
|
fi
|
||||||
}
|
}
|
||||||
|
|
||||||
# Trap errors and cleanup on exit
|
|
||||||
trap 'err $LINENO' ERR && trap cleanup EXIT
|
|
||||||
|
|
||||||
stat_chmod() {
|
stat_chmod() {
|
||||||
# Only run chmod if mode is different from current mode
|
# Only run chmod if mode is different from current mode
|
||||||
if [[ "$(stat -c%a "${2}")" -ne "${1}" ]]; then chmod "${1}" "${2}"; fi
|
if [[ "$(stat -c%a "${2}")" -ne "${1}" ]]; then chmod "${1}" "${2}"; fi
|
||||||
}
|
}
|
||||||
|
|
||||||
# TODO: put body inside the create function (no need for a function)
|
validate_download_script() {
|
||||||
read_name() {
|
echo "6cca2eda73c7358c232fecb4e750b3bf0afa9636efb5de6a9517b7df78be12a4 ${1}" | sha256sum --check >/dev/null
|
||||||
local jail_name
|
}
|
||||||
local jail_path
|
|
||||||
|
|
||||||
while true; do
|
# Use a function as template instead of heredoc
|
||||||
read -e -r -p "Enter jail name: " jail_name && echo
|
# This allows syntax highlighting and linting to work
|
||||||
if ! [[ "${jail_name}" =~ ^[.a-zA-Z0-9-]{1,64}$ && "${jail_name}" != '.'* && "${jail_name}" != *'.' && "${jail_name}" != *'..'* ]]; then
|
# Template function must return the line number,
|
||||||
cat <<-'EOF'
|
# this marks the start of the template
|
||||||
|
template_start_script() {
|
||||||
|
echo "$((LINENO + 1))" && return
|
||||||
|
# TEMPLATE START
|
||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
# YOU CAN MANUALLY EDIT THE SETTINGS BELOW
|
||||||
|
|
||||||
|
DOCKER_COMPATIBLE=PLACEHOLDER_DOCKER_COMPATIBLE
|
||||||
|
GPU_PASSTHROUGH=PLACEHOLDER_GPU_PASSTHROUGH
|
||||||
|
|
||||||
|
# DO NOT EDIT THE SCRIPT FROM HERE ON, UNLESS YOU KNOW WHAT YOU ARE DOING
|
||||||
|
# THIS FILE WILL BE SOURCED IN A BASH SUB-SHELL BY PLACEHOLDER_SCRIPT_NAME
|
||||||
|
# THE START FUNCTION WILL BE CALLED TO START THE JAIL
|
||||||
|
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
# Move into the directory where this script is stored (commands are relative to this directory)
|
||||||
|
cd "$(dirname "${BASH_SOURCE[0]}")" || exit
|
||||||
|
JAIL_NAME="$(basename "$(pwd)")"
|
||||||
|
|
||||||
|
# You may add additional args to the two arrays below
|
||||||
|
# These args will be passed to systemd-run and systemd-nspawn in the start function
|
||||||
|
SYSTEMD_RUN_ADDITIONAL_ARGS=()
|
||||||
|
SYSTEMD_NSPAWN_ADDITIONAL_ARGS=(PLACEHOLDER_SYSTEMD_NSPAWN_ADDITIONAL_ARGS)
|
||||||
|
|
||||||
|
start() {
|
||||||
|
# Use mostly default settings for systemd-nspawn but with systemd-run instead of a service file
|
||||||
|
# https://github.com/systemd/systemd/blob/main/units/systemd-nspawn%40.service.in
|
||||||
|
# TODO: also compare settings for docker: https://github.com/docker/engine/blob/master/contrib/init/systemd/docker.service
|
||||||
|
systemd-run --property=KillMode=mixed --property=Type=notify --property=RestartForceExitStatus=133 \
|
||||||
|
--property=SuccessExitStatus=133 --property=Delegate=yes --property=TasksMax=16384 --same-dir \
|
||||||
|
--collect --setenv=SYSTEMD_NSPAWN_LOCK=0 \
|
||||||
|
--unit="jlmkr-${JAIL_NAME}" --description="jailmaker ${JAIL_NAME}" \
|
||||||
|
"${SYSTEMD_RUN_ADDITIONAL_ARGS[@]}" \
|
||||||
|
-- \
|
||||||
|
systemd-nspawn --keep-unit --quiet --boot \
|
||||||
|
--machine="${JAIL_NAME}" --directory='./PLACEHOLDER_ROOTFS_NAME' \
|
||||||
|
"${SYSTEMD_NSPAWN_ADDITIONAL_ARGS[@]}"
|
||||||
|
}
|
||||||
|
|
||||||
|
# Call the start function if this script is executed directly (not sourced)
|
||||||
|
# https://stackoverflow.com/a/28776166
|
||||||
|
(return 0 2>/dev/null) || {
|
||||||
|
echo 'This script was called directly, not sourced.'
|
||||||
|
echo 'The jail will now start...'
|
||||||
|
echo 'But the DOCKER_COMPATIBLE and GPU_PASSTHROUGH settings are not considered.'
|
||||||
|
echo "For this to work, start the jail from PLACEHOLDER_SCRIPT_NAME."
|
||||||
|
start
|
||||||
|
}
|
||||||
|
# TEMPLATE END
|
||||||
|
}
|
||||||
|
|
||||||
|
# Helper function to process the body of a function into a bash string
|
||||||
|
# which can be written as a new bash script
|
||||||
|
# Includes find-replace functionality to substitute placeholders in the template
|
||||||
|
process_template() {
|
||||||
|
local indent=""
|
||||||
|
# Read the current script file, starting from the passed line number
|
||||||
|
while IFS=$'\n' read -r line; do
|
||||||
|
# Get the indent level from the first line
|
||||||
|
[[ -z "${indent}" ]] && indent="${line%%#*}" && continue
|
||||||
|
|
||||||
|
# Break when we find the end of the template
|
||||||
|
[[ "$line" = "${indent}# TEMPLATE END" ]] && break
|
||||||
|
|
||||||
|
# Remove the indent from the start
|
||||||
|
line="${line#"$indent"}"
|
||||||
|
|
||||||
|
local find replace
|
||||||
|
# Loop over the additional argument pairs passed
|
||||||
|
# and find/replace template key with value
|
||||||
|
for ((n = 2; n <= $#; n++)); do
|
||||||
|
# https://stackoverflow.com/a/3575950
|
||||||
|
# Indirect expansion: look up the value of the variable whose name is in the variable
|
||||||
|
# So this looks up the nth argument passed to this function
|
||||||
|
find=${!n}
|
||||||
|
# Increment counter by one to get next argument
|
||||||
|
((n++))
|
||||||
|
replace=${!n}
|
||||||
|
line=${line//$find/$replace}
|
||||||
|
done
|
||||||
|
|
||||||
|
echo "${line}"
|
||||||
|
done < <(tail -n "+${1}" "${ABSOLUTE_SCRIPT_PATH}")
|
||||||
|
}
|
||||||
|
|
||||||
|
# Helper function which escapes and concatenates a string of arguments
|
||||||
|
escape() {
|
||||||
|
local result="" spacer=""
|
||||||
|
# Process each argument (newline separated thanks to: xargs -n 1)
|
||||||
|
while IFS=$'\n' read -r arg; do
|
||||||
|
arg="$(declare -p arg)"
|
||||||
|
# Get everything after the first = character
|
||||||
|
arg="${arg#*=}"
|
||||||
|
# Append to string with leading space
|
||||||
|
result+="${spacer}${arg}"
|
||||||
|
# From now on, use a space in between elements
|
||||||
|
spacer=" "
|
||||||
|
done < <(printf "%s" "${1}" | xargs -n 1)
|
||||||
|
|
||||||
|
printf "%s" "${result}"
|
||||||
|
}
|
||||||
|
|
||||||
|
create_jail() {
|
||||||
|
|
||||||
|
local reply arch distro release lxc_dir_path lxc_cache_path lxc_download_script_path
|
||||||
|
local jail_config_path jail_start_script_name jail_start_script_path docker_compatible gpu_passthrough
|
||||||
|
local jail_name systemd_nspawn_additional_args
|
||||||
|
|
||||||
|
arch="$(dpkg --print-architecture)"
|
||||||
|
lxc_dir_path='.lxc'
|
||||||
|
lxc_cache_path="${lxc_dir_path}/cache"
|
||||||
|
lxc_download_script_path="${lxc_dir_path}/lxc-download.sh"
|
||||||
|
|
||||||
|
[[ "$(basename "${SCRIPT_DIR_PATH}")" != 'jailmaker' ]] && {
|
||||||
|
echo "${SCRIPT_NAME} needs to create files."
|
||||||
|
echo "Currently it can't decide if it's safe to create files in:"
|
||||||
|
echo "${SCRIPT_DIR_PATH}"
|
||||||
|
fail "Please create a dedicated directory called 'jailmaker', store ${SCRIPT_NAME} there and try again."
|
||||||
|
}
|
||||||
|
|
||||||
|
if [[ $(findmnt --target . --output TARGET --noheadings --first-only) != /mnt/* ]]; then
|
||||||
|
echo "${SCRIPT_NAME} should be on a pool mounted under /mnt (it currently isn't)."
|
||||||
|
echo "Storing it on the boot-pool means losing all jails when updating TrueNAS."
|
||||||
|
echo "If you continue, jails will be stored under:"
|
||||||
|
echo "${SCRIPT_DIR_PATH}"
|
||||||
|
read -p "Do you wish to ignore this warning and continue? [y/N] " -n 1 -r reply && echo
|
||||||
|
# Enter accepts default (no)
|
||||||
|
! [[ "${reply}" =~ ^[Yy]$ ]] && exit
|
||||||
|
fi
|
||||||
|
|
||||||
|
cd "${SCRIPT_DIR_PATH}" || fail "Could not change working directory to ${SCRIPT_DIR_PATH}..."
|
||||||
|
|
||||||
|
# Set appropriate permissions (if not already set) for this file, since it's executed as root
|
||||||
|
stat_chmod 700 "${SCRIPT_NAME}"
|
||||||
|
|
||||||
|
# Create the lxc dirs if nonexistent
|
||||||
|
mkdir -p "${lxc_dir_path}"
|
||||||
|
stat_chmod 700 "${lxc_dir_path}"
|
||||||
|
mkdir -p "${lxc_cache_path}"
|
||||||
|
stat_chmod 700 "${lxc_cache_path}"
|
||||||
|
|
||||||
|
# Create the dir where to store the jails
|
||||||
|
mkdir -p "${JAILS_DIR_PATH}"
|
||||||
|
stat_chmod 700 "${JAILS_DIR_PATH}"
|
||||||
|
|
||||||
|
# Fetch the lxc download script if not present locally (or hash doesn't match)
|
||||||
|
if ! validate_download_script "${lxc_download_script_path}"; then
|
||||||
|
curl -fSL
|
||||||
|
https://raw.githubusercontent.com/Jip-Hop/lxc/58520263041b6864cadad96278848f9b8ce78ee9/templates/lxc-download.in -o "${lxc_download_script_path}"
|
||||||
|
# Validate after download to prevent executing arbitrary code as root
|
||||||
|
validate_download_script "${lxc_download_script_path}" || fail 'Abort! Downloaded script has unexpected contents.'
|
||||||
|
fi
|
||||||
|
|
||||||
|
stat_chmod 700 "${lxc_download_script_path}"
|
||||||
|
|
||||||
|
read -p "Install the recommended distro (Debian 11)? [Y/n] " -n 1 -r reply && echo
|
||||||
|
if [[ "${reply}" =~ ^([Yy]|)$ ]]; then
|
||||||
|
distro='debian'
|
||||||
|
release='bullseye'
|
||||||
|
else
|
||||||
|
echo
|
||||||
|
echo "ADVANCED USAGE"
|
||||||
|
echo "You may now choose from a list which distro to install."
|
||||||
|
echo "Not all of them will work with ${SCRIPT_NAME} (these images are made for LXC)."
|
||||||
|
echo "Distros based on systemd probably work (e.g. Ubuntu, Arch Linux and Rocky Linux)."
|
||||||
|
echo "Others (Alpine, Devuan, Void Linux) probably won't."
|
||||||
|
echo
|
||||||
|
read -p "Press any key to continue: " -n 1 -r reply && echo
|
||||||
|
lxc_cache_path=${lxc_cache_path} "${lxc_download_script_path}" --list --arch="${arch}" || :
|
||||||
|
echo "Choose from the DIST column."
|
||||||
|
read -e -r -p "Distribution: " distro && echo
|
||||||
|
echo "Choose from the RELEASE column (or ARCH if RELEASE is empty)."
|
||||||
|
read -e -r -p "Release: " release && echo
|
||||||
|
fi
|
||||||
|
|
||||||
|
while true; do
|
||||||
|
read -e -r -p "Enter jail name: " jail_name && echo
|
||||||
|
if ! [[ "${jail_name}" =~ ^[.a-zA-Z0-9-]{1,64}$ && "${jail_name}" != '.'* && "${jail_name}" != *'.' && "${jail_name}" != *'..'* ]]; then
|
||||||
|
cat <<-'EOF'
|
||||||
A valid name consists of:
|
A valid name consists of:
|
||||||
- allowed characters (alphanumeric, dash, dot)
|
- allowed characters (alphanumeric, dash, dot)
|
||||||
- no leading or trailing dots
|
- no leading or trailing dots
|
||||||
|
@ -72,306 +295,138 @@ read_name() {
|
||||||
- max 64 characters
|
- max 64 characters
|
||||||
|
|
||||||
EOF
|
EOF
|
||||||
else
|
else
|
||||||
jail_path="${JAILS_DIR_PATH}/${jail_name}"
|
JAIL_PATH="${JAILS_DIR_PATH}/${jail_name}"
|
||||||
|
|
||||||
if [[ -e "${jail_path}" ]]; then
|
if [[ -e "${JAIL_PATH}" ]]; then
|
||||||
echo "A jail with this name already exists."
|
echo "A jail with this name already exists."
|
||||||
echo
|
echo
|
||||||
else
|
else
|
||||||
# Only set global variables if we accept the name,
|
# Accept the name
|
||||||
# else the wrong directory may be cleaned up!
|
break
|
||||||
JAIL_NAME="${jail_name}"
|
fi
|
||||||
JAIL_PATH="${jail_path}"
|
fi
|
||||||
SYSTEMD_RUN_UNIT_NAME="jlmkr-${JAIL_NAME}"
|
done
|
||||||
return
|
|
||||||
fi
|
# Cleanup on exit, but only once the JAIL_PATH name is known
|
||||||
fi
|
trap cleanup EXIT
|
||||||
done
|
|
||||||
|
echo "${SCRIPT_NAME} will not install docker for you."
|
||||||
|
echo "But it can configure the jail with the capabilities required to run docker."
|
||||||
|
echo "You can turn docker_compatible mode on/off post-install."
|
||||||
|
echo
|
||||||
|
read -p "Make jail docker compatible right now? [y/N] " -n 1 -r reply && echo
|
||||||
|
# Enter accepts default (no)
|
||||||
|
if ! [[ "${reply}" =~ ^[Yy]$ ]]; then docker_compatible=0; else docker_compatible=1; fi
|
||||||
|
|
||||||
|
read -p "Give access to the GPU inside the jail? [y/N] " -n 1 -r reply && echo
|
||||||
|
# Enter accepts default (no)
|
||||||
|
if ! [[ "${reply}" =~ ^[Yy]$ ]]; then gpu_passthrough=0; else gpu_passthrough=1; fi
|
||||||
|
|
||||||
|
# TODO: ask for network setup (host, macvlan, bridge, physical nic)
|
||||||
|
echo
|
||||||
|
echo "You may pass additional systemd-nspawn flags."
|
||||||
|
echo "For example to mount directories inside the jail you may add:"
|
||||||
|
echo "--bind=/mnt/a/readwrite/directory --bind-ro=/mnt/a/readonly/directory"
|
||||||
|
echo
|
||||||
|
echo "Double check the syntax:"
|
||||||
|
echo "https://manpages.debian.org/bullseye/systemd-container/systemd-nspawn.1.en.html"
|
||||||
|
echo "With incorrect flags the jail may not start."
|
||||||
|
echo "It's possible to correct/add/remove flags post-install."
|
||||||
|
echo
|
||||||
|
read -e -r -p "Additional flags: " SYSTEMD_NSPAWN_USER_ARGS_STRING && echo
|
||||||
|
# Backslashes and colons need to be escaped for systemd-nspawn by the user:
|
||||||
|
# e.g. to bind mount a file called:
|
||||||
|
# weird chars :?\"
|
||||||
|
# the corresponding command would be:
|
||||||
|
# --bind-ro='/mnt/data/weird chars \:?\\"'
|
||||||
|
|
||||||
|
# Process the flags (read as a string) as actual command line arguments with xargs and
|
||||||
|
# pass these args to bash which then executes the escape function to convert the args into a properly escaped string
|
||||||
|
# which can be evaluated as an array declaration in the start.sh bash script
|
||||||
|
# https://superuser.com/a/1529316/1268213 and https://superuser.com/a/1627765
|
||||||
|
systemd_nspawn_additional_args="$(escape "${SYSTEMD_NSPAWN_USER_ARGS_STRING}")"
|
||||||
|
echo "systemd_nspawn_additional_args=${systemd_nspawn_additional_args}"
|
||||||
|
|
||||||
|
# Create directory for rootfs
|
||||||
|
JAIL_ROOTFS_PATH="${JAIL_PATH}/${JAIL_ROOTFS_NAME}"
|
||||||
|
mkdir -p "${JAIL_ROOTFS_PATH}"
|
||||||
|
|
||||||
|
jail_config_path="${JAIL_PATH}/config"
|
||||||
|
# LXC download script needs to write to this file during install
|
||||||
|
# but we don't need it so we will remove it later
|
||||||
|
touch "${jail_config_path}"
|
||||||
|
|
||||||
|
echo
|
||||||
|
LXC_CACHE_PATH=${lxc_cache_path} "${lxc_download_script_path}" \
|
||||||
|
--name="${jail_name}" --path="${JAIL_PATH}" --rootfs="${JAIL_ROOTFS_PATH}" \
|
||||||
|
--arch="${arch}" --dist="${distro}" --release="${release}" ||
|
||||||
|
fail "Aborted creating rootfs..."
|
||||||
|
echo
|
||||||
|
|
||||||
|
if [[ "$(basename "$(readlink -f "${JAIL_ROOTFS_PATH}/sbin/init")")" != systemd ]]; then
|
||||||
|
echo "Chosen distro appears not to use systemd..."
|
||||||
|
echo
|
||||||
|
echo "You probably won't get a shell with:"
|
||||||
|
echo "machinectl shell ${jail_name}"
|
||||||
|
echo
|
||||||
|
echo "You may get a shell with this command:"
|
||||||
|
# About nsenter:
|
||||||
|
# https://github.com/systemd/systemd/issues/12785#issuecomment-503019081
|
||||||
|
# https://github.com/systemd/systemd/issues/3144
|
||||||
|
# shellcheck disable=SC2016
|
||||||
|
echo 'nsenter -t $(machinectl show '"${jail_name}"' -p Leader --value) -a /bin/sh -l'
|
||||||
|
echo
|
||||||
|
echo "Using this distro with ${SCRIPT_NAME} is not recommended."
|
||||||
|
echo
|
||||||
|
read -p "Abort creating jail? [Y/n] " -n 1 -r reply && echo
|
||||||
|
# Enter accepts default (yes)
|
||||||
|
[[ "${reply}" =~ ^([Yy]|)$ ]] && exit
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Remove file we no longer need
|
||||||
|
rm -f "${jail_config_path}"
|
||||||
|
# Config which systemd handles for us
|
||||||
|
rm -f "${JAIL_ROOTFS_PATH}/etc/machine-id"
|
||||||
|
rm -f "${JAIL_ROOTFS_PATH}/etc/resolv.conf"
|
||||||
|
rm -f "${JAIL_ROOTFS_PATH}/etc/resolv.conf"
|
||||||
|
# https://github.com/systemd/systemd/issues/852
|
||||||
|
printf 'pts/%d\n' $(seq 0 10) >"${JAIL_ROOTFS_PATH}/etc/securetty"
|
||||||
|
|
||||||
|
jail_start_script_name='start.sh'
|
||||||
|
jail_start_script_path="${JAIL_PATH}/${jail_start_script_name}"
|
||||||
|
|
||||||
|
process_template "$(template_start_script)" PLACEHOLDER_DOCKER_COMPATIBLE "${docker_compatible}" PLACEHOLDER_GPU_PASSTHROUGH "${gpu_passthrough}" \
|
||||||
|
PLACEHOLDER_ROOTFS_NAME "${JAIL_ROOTFS_NAME}" PLACEHOLDER_SYSTEMD_NSPAWN_ADDITIONAL_ARGS "${systemd_nspawn_additional_args}" \
|
||||||
|
PLACEHOLDER_SCRIPT_NAME "${SCRIPT_NAME}" >"${jail_start_script_path}"
|
||||||
|
|
||||||
|
chmod 700 "${jail_start_script_path}"
|
||||||
|
|
||||||
|
echo "Done creating the jail."
|
||||||
|
CREATE_DONE=1
|
||||||
|
echo
|
||||||
|
read -p "Start the jail now? [Y/n] " -n 1 -r reply && echo
|
||||||
|
# Enter accepts default (yes)
|
||||||
|
if [[ "${reply}" =~ ^([Yy]|)$ ]]; then
|
||||||
|
run_jail "${jail_start_script_path}"
|
||||||
|
else
|
||||||
|
echo 'Skipped starting jail.'
|
||||||
|
fi
|
||||||
}
|
}
|
||||||
|
|
||||||
validate_download_script() {
|
##########################
|
||||||
echo "6cca2eda73c7358c232fecb4e750b3bf0afa9636efb5de6a9517b7df78be12a4 ${LXC_DOWNLOAD_SCRIPT_PATH}" | sha256sum --check >/dev/null
|
# END CREATE FUNCTIONALITY
|
||||||
}
|
##########################
|
||||||
|
|
||||||
run_jail() (
|
read -p "Create a new jail? [Y/n] " -n 1 -r reply && echo
|
||||||
# Create a sub-shell to source the conf file
|
# Enter accepts default (yes)
|
||||||
|
# https://stackoverflow.com/a/1885534
|
||||||
RUN_DOCKER=
|
if [[ "${reply}" =~ ^([Yy]|)$ ]]; then
|
||||||
GPU_PASSTHROUGH=
|
create_jail
|
||||||
SYSTEMD_RUN_ADDITIONAL_ARGS=()
|
else
|
||||||
SYSTEMD_NSPAWN_ADDITIONAL_ARGS=()
|
echo "${USAGE}"
|
||||||
|
exit
|
||||||
echo 'Load the config'
|
fi
|
||||||
# shellcheck disable=SC1090
|
|
||||||
. "${1}"
|
|
||||||
echo 'Config loaded'
|
|
||||||
|
|
||||||
set -eEuo pipefail
|
|
||||||
if [[ "$(type -t start)" == 'function' ]]; then
|
|
||||||
if [[ "${RUN_DOCKER}" -eq 1 ]]; then
|
|
||||||
# Enable ip forwarding on the host (docker needs it)
|
|
||||||
echo 1 >/proc/sys/net/ipv4/ip_forward
|
|
||||||
# To properly run docker inside the jail, we need to lift restrictions
|
|
||||||
# Without DevicePolicy=auto images with device nodes may not be pulled
|
|
||||||
# https://github.com/kinvolk/kube-spawn/pull/328
|
|
||||||
SYSTEMD_RUN_ADDITIONAL_ARGS+=(--setenv=SYSTEMD_SECCOMP=0 --property=DevicePolicy=auto)
|
|
||||||
# Add additional flags required for docker
|
|
||||||
SYSTEMD_NSPAWN_ADDITIONAL_ARGS+=(--capability=all --system-call-filter='add_key keyctl bpf')
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [[ "${GPU_PASSTHROUGH}" -eq 1 ]]; then
|
|
||||||
SYSTEMD_NSPAWN_ADDITIONAL_ARGS+=(--property=DeviceAllow='char-drm rw')
|
|
||||||
|
|
||||||
# Detect intel GPU device and if present add bind flag
|
|
||||||
[[ -d /dev/dri ]] && SYSTEMD_NSPAWN_ADDITIONAL_ARGS+=(--bind=/dev/dri)
|
|
||||||
|
|
||||||
# TODO: add bind mount flags in case of nvidia GPU passthrough
|
|
||||||
fi
|
|
||||||
|
|
||||||
echo "Starting jail..."
|
|
||||||
start
|
|
||||||
else
|
|
||||||
echo "Can't call the start function since the conf file didn't contain one..."
|
|
||||||
fi
|
|
||||||
)
|
|
||||||
|
|
||||||
escape() {
|
|
||||||
s=""
|
|
||||||
for t in "$@"; do
|
|
||||||
t="$(declare -p t)"
|
|
||||||
t="${t#*=}"
|
|
||||||
s+=" ${t}"
|
|
||||||
done
|
|
||||||
echo "${s}"
|
|
||||||
}
|
|
||||||
|
|
||||||
# TODO: remove indent
|
|
||||||
# https://stackoverflow.com/questions/6916856/can-bash-show-a-functions-definition
|
|
||||||
get_function_body() {
|
|
||||||
type "${1}" | sed '1,3d;$d'
|
|
||||||
}
|
|
||||||
|
|
||||||
create_jail() {
|
|
||||||
|
|
||||||
read -p "Create a new jail? [Y/n] " -n 1 -r REPLY && echo
|
|
||||||
# Enter accepts default (yes)
|
|
||||||
# https://stackoverflow.com/a/1885534
|
|
||||||
! [[ "${REPLY}" =~ ^([Yy]|)$ ]] && echo "${USAGE}" && exit
|
|
||||||
|
|
||||||
[[ "$(basename "${SCRIPT_DIR_PATH}")" != 'jailmaker' ]] && fail "${SCRIPT_NAME} needs to create files.
|
|
||||||
Currently it can't decide if it's safe to create files in:
|
|
||||||
${SCRIPT_DIR_PATH}
|
|
||||||
Please create a dedicated directory called 'jailmaker', store ${SCRIPT_NAME} there and try again."
|
|
||||||
|
|
||||||
if [[ $(findmnt --target . --output TARGET --noheadings --first-only) != /mnt/* ]]; then
|
|
||||||
echo "${SCRIPT_NAME} should be on a pool mounted under /mnt (it currently isn't)."
|
|
||||||
echo "Storing it on the boot-pool means losing all jails when updating TrueNAS."
|
|
||||||
echo "If you continue, jails will be stored under:"
|
|
||||||
echo "${SCRIPT_DIR_PATH}"
|
|
||||||
read -p "Do you wish to ignore this warning and continue? [y/N] " -n 1 -r REPLY && echo
|
|
||||||
# Enter accepts default (no)
|
|
||||||
! [[ "${REPLY}" =~ ^[Yy]$ ]] && exit
|
|
||||||
fi
|
|
||||||
|
|
||||||
cd "${SCRIPT_DIR_PATH}" || fail "Could not change working directory to ${SCRIPT_DIR_PATH}..."
|
|
||||||
|
|
||||||
# Set appropriate permissions (if not already set) for this file, since it's executed as root
|
|
||||||
stat_chmod 700 "${SCRIPT_NAME}"
|
|
||||||
|
|
||||||
# Create the lxc dirs if nonexistent
|
|
||||||
mkdir -p "${LXC_DIR_PATH}"
|
|
||||||
stat_chmod 700 "${LXC_DIR_PATH}"
|
|
||||||
mkdir -p "${LXC_CACHE_PATH}"
|
|
||||||
stat_chmod 700 "${LXC_CACHE_PATH}"
|
|
||||||
|
|
||||||
# Create the dir where to store the jails
|
|
||||||
mkdir -p "${JAILS_DIR_PATH}"
|
|
||||||
stat_chmod 700 "${JAILS_DIR_PATH}"
|
|
||||||
|
|
||||||
# Fetch the lxc download script if not present locally (or hash doesn't match)
|
|
||||||
if ! validate_download_script; then
|
|
||||||
curl -fSL
|
|
||||||
https://raw.githubusercontent.com/Jip-Hop/lxc/58520263041b6864cadad96278848f9b8ce78ee9/templates/lxc-download.in -o "${LXC_DOWNLOAD_SCRIPT_PATH}"
|
|
||||||
# Validate after download to prevent executing arbritrary code as root
|
|
||||||
validate_download_script || fail 'Abort! Downloaded script has unexpected contents.'
|
|
||||||
fi
|
|
||||||
|
|
||||||
stat_chmod 700 "${LXC_DOWNLOAD_SCRIPT_PATH}"
|
|
||||||
|
|
||||||
read -p "Install the recommended distro (Debian 11)? [Y/n] " -n 1 -r REPLY && echo
|
|
||||||
if [[ "${REPLY}" =~ ^([Yy]|)$ ]]; then
|
|
||||||
DISTRO='debian'
|
|
||||||
RELEASE='bullseye'
|
|
||||||
else
|
|
||||||
echo
|
|
||||||
echo "ADVANCED USAGE"
|
|
||||||
echo "You may now choose from a list which distro to install."
|
|
||||||
echo "Not all of them will work with ${SCRIPT_NAME} (these images are made for LXC)."
|
|
||||||
echo "Distros based on systemd probably work (e.g. Ubuntu, Arch Linux and Rocky Linux)."
|
|
||||||
echo "Others (Alpine, Devuan, Void Linux) probably won't."
|
|
||||||
echo
|
|
||||||
read -p "Press any key to continue: " -n 1 -r REPLY && echo
|
|
||||||
LXC_CACHE_PATH=${LXC_CACHE_PATH} "${LXC_DOWNLOAD_SCRIPT_PATH}" --list --arch="${ARCH}" || :
|
|
||||||
echo "Choose from the DIST column."
|
|
||||||
read -e -r -p "Distribution: " DISTRO && echo
|
|
||||||
echo "Choose from the RELEASE column (or ARCH if RELEASE is empty)."
|
|
||||||
read -e -r -p "Release: " RELEASE && echo
|
|
||||||
fi
|
|
||||||
|
|
||||||
read_name
|
|
||||||
|
|
||||||
local systemd_run_additional_args systemd_nspawn_additional_args
|
|
||||||
# TODO: don't use jail name inside these strings? Allow reading the basename of dir containing the script?
|
|
||||||
systemd_run_additional_args="--unit='${SYSTEMD_RUN_UNIT_NAME}' --description='jailmaker ${JAIL_NAME}'"
|
|
||||||
systemd_nspawn_additional_args="--machine='${JAIL_NAME}' --directory='./${JAIL_ROOTFS_NAME}'"
|
|
||||||
|
|
||||||
echo "${SCRIPT_NAME} will not install docker for you."
|
|
||||||
echo "But it can configure the jail with the capabilities required to run docker."
|
|
||||||
echo "You can turn DOCKER_COMPATIBLE mode on/off post-install."
|
|
||||||
echo
|
|
||||||
read -p "Make jail docker compatible right now? [y/N] " -n 1 -r REPLY && echo
|
|
||||||
# Enter accepts default (no)
|
|
||||||
if ! [[ "${REPLY}" =~ ^[Yy]$ ]]; then DOCKER_COMPATIBLE=0; else DOCKER_COMPATIBLE=1; fi
|
|
||||||
|
|
||||||
read -p "Give access to the GPU inside the jail? [y/N] " -n 1 -r REPLY && echo
|
|
||||||
# Enter accepts default (no)
|
|
||||||
if ! [[ "${REPLY}" =~ ^[Yy]$ ]]; then GPU_PASSTHROUGH=0; else GPU_PASSTHROUGH=1; fi
|
|
||||||
|
|
||||||
# TODO: ask for bind mounts (and warn if trying to mount a parent directory of the jailmaker dir?)
|
|
||||||
# TODO: ask for network setup (host, macvlan, bridge, physical nic)
|
|
||||||
# TODO: ask for additional flags (to bind mount etc.)
|
|
||||||
echo "You may pass additional systemd-nspawn flags."
|
|
||||||
echo "For example to mount directories inside the jail you may add:"
|
|
||||||
echo "--bind=/mnt/a/readwrite/directory --bind-ro=/mnt/a/readonly/directory"
|
|
||||||
echo
|
|
||||||
echo "WARNING: double check the syntax:"
|
|
||||||
echo "https://manpages.debian.org/bullseye/systemd-container/systemd-nspawn.1.en.html"
|
|
||||||
echo "With incorrect flags the jail may not start."
|
|
||||||
echo "It's possible to correct/add/remove flags post-install."
|
|
||||||
echo
|
|
||||||
read -e -r -p "Additional flags: " SYSTEMD_NSPAWN_USER_ARGS_STRING && echo
|
|
||||||
# Backslashes and colons need to be escaped for systemd-nspawn by the user:
|
|
||||||
# e.g. to bind mount a file called:
|
|
||||||
# weird chars :?\"
|
|
||||||
# the corresponding command would be:
|
|
||||||
# --bind-ro='/mnt/data/weird chars \:?\\"'
|
|
||||||
echo "$(get_function_body escape)"
|
|
||||||
systemd_nspawn_additional_args+="$(echo "${SYSTEMD_NSPAWN_USER_ARGS_STRING}" | xargs bash -c "$(get_function_body escape)" --)"
|
|
||||||
echo "$systemd_nspawn_additional_args"
|
|
||||||
cleanup
|
|
||||||
exit
|
|
||||||
# https://superuser.com/a/1529316/1268213
|
|
||||||
# https://superuser.com/a/1627765
|
|
||||||
|
|
||||||
# Create directory for rootfs
|
|
||||||
JAIL_ROOTFS_PATH="${JAIL_PATH}/${JAIL_ROOTFS_NAME}"
|
|
||||||
mkdir -p "${JAIL_ROOTFS_PATH}"
|
|
||||||
|
|
||||||
JAIL_CONFIG_NAME='config'
|
|
||||||
JAIL_CONFIG_PATH="${JAIL_PATH}/${JAIL_CONFIG_NAME}"
|
|
||||||
# LXC download script needs to write to this file during install
|
|
||||||
# but we don't need it so we will remove it later
|
|
||||||
touch "${JAIL_CONFIG_PATH}"
|
|
||||||
|
|
||||||
echo
|
|
||||||
LXC_CACHE_PATH=${LXC_CACHE_PATH} "${LXC_DOWNLOAD_SCRIPT_PATH}" \
|
|
||||||
--name="${JAIL_NAME}" --path="${JAIL_PATH}" --rootfs="${JAIL_ROOTFS_PATH}" \
|
|
||||||
--arch="${ARCH}" --dist="${DISTRO}" --release="${RELEASE}" ||
|
|
||||||
fail "Aborted creating rootfs..."
|
|
||||||
echo
|
|
||||||
|
|
||||||
if [[ "$(basename "$(readlink -f "${JAIL_ROOTFS_PATH}/sbin/init")")" != systemd ]]; then
|
|
||||||
echo "Chosen distro appears not to use systemd..."
|
|
||||||
echo
|
|
||||||
echo "You probably won't get a shell with:"
|
|
||||||
echo "machinectl shell ${JAIL_NAME}"
|
|
||||||
echo
|
|
||||||
echo "You may get a shell with this command:"
|
|
||||||
# About nsenter:
|
|
||||||
# https://github.com/systemd/systemd/issues/12785#issuecomment-503019081
|
|
||||||
# https://github.com/systemd/systemd/issues/3144
|
|
||||||
# shellcheck disable=SC2016
|
|
||||||
echo 'nsenter -t $(machinectl show '"${JAIL_NAME}"' -p Leader --value) -a /bin/sh -l'
|
|
||||||
echo
|
|
||||||
echo "Using this distro with ${SCRIPT_NAME} is not recommended."
|
|
||||||
echo
|
|
||||||
read -p "Abort creating jail? [Y/n] " -n 1 -r REPLY && echo
|
|
||||||
# Enter accepts default (yes)
|
|
||||||
[[ "${REPLY}" =~ ^([Yy]|)$ ]] && exit
|
|
||||||
fi
|
|
||||||
|
|
||||||
# Remove file we no longer need
|
|
||||||
rm -f "${JAIL_CONFIG_PATH}"
|
|
||||||
# Config which systemd handles for us
|
|
||||||
rm -f "${JAIL_ROOTFS_PATH}/etc/machine-id"
|
|
||||||
rm -f "${JAIL_ROOTFS_PATH}/etc/resolv.conf"
|
|
||||||
rm -f "${JAIL_ROOTFS_PATH}/etc/resolv.conf"
|
|
||||||
# https://github.com/systemd/systemd/issues/852
|
|
||||||
printf 'pts/%d\n' $(seq 0 10) >"${JAIL_ROOTFS_PATH}/etc/securetty"
|
|
||||||
|
|
||||||
JAIL_CONFIG_NAME='start.sh'
|
|
||||||
JAIL_CONFIG_PATH="${JAIL_PATH}/${JAIL_CONFIG_NAME}"
|
|
||||||
|
|
||||||
cat <<-EOF >"${JAIL_CONFIG_PATH}"
|
|
||||||
#!/bin/bash
|
|
||||||
# This file will be sourced in a a bash sub-shell by ${SCRIPT_NAME}
|
|
||||||
# The start function will be called to start the jail
|
|
||||||
# You can change the settings below and/or add custom code
|
|
||||||
set -eEuo pipefail
|
|
||||||
# Move into the directory where this script is stored (commands are relative to this directory)
|
|
||||||
cd "\$(dirname "\${BASH_SOURCE[0]}")" || exit
|
|
||||||
|
|
||||||
# Set RUN_DOCKER=1 to automatically add additional arguments required to properly run docker inside the jail
|
|
||||||
RUN_DOCKER=${DOCKER_COMPATIBLE}
|
|
||||||
# Set GPU_PASSTHROUGH=1 to automatically add additional arguments to access the GPU inside the jail
|
|
||||||
GPU_PASSTHROUGH=${GPU_PASSTHROUGH}
|
|
||||||
|
|
||||||
# You may add additional args to the two arrays below
|
|
||||||
# These args will be passed to systemd-run and systemd-nspawn in the start function
|
|
||||||
SYSTEMD_RUN_ADDITIONAL_ARGS=(${systemd_run_additional_args})
|
|
||||||
SYSTEMD_NSPAWN_ADDITIONAL_ARGS=(${systemd_nspawn_additional_args})
|
|
||||||
|
|
||||||
start(){
|
|
||||||
# Use mostly default settings for systemd-nspawn but with systemd-run instead of a service file
|
|
||||||
# https://github.com/systemd/systemd/blob/main/units/systemd-nspawn%40.service.in
|
|
||||||
systemd-run --property=KillMode=mixed --property=Type=notify --property=RestartForceExitStatus=133 \\
|
|
||||||
--property=SuccessExitStatus=133 --property=Delegate=yes --property=TasksMax=16384 --same-dir \\
|
|
||||||
--collect --setenv=SYSTEMD_NSPAWN_LOCK=0 \\
|
|
||||||
"\${SYSTEMD_RUN_ADDITIONAL_ARGS[@]}" \\
|
|
||||||
-- \\
|
|
||||||
systemd-nspawn --keep-unit --quiet --boot \\
|
|
||||||
"\${SYSTEMD_NSPAWN_ADDITIONAL_ARGS[@]}"
|
|
||||||
}
|
|
||||||
|
|
||||||
# Call the start function if this script is executed directly (not sourced)
|
|
||||||
# https://stackoverflow.com/a/28776166
|
|
||||||
(return 0 2>/dev/null) || {
|
|
||||||
echo 'This script was called directly, not sourced.'
|
|
||||||
echo 'The jail will now start...'
|
|
||||||
echo 'But the RUN_DOCKER and GPU_PASSTHROUGH settings are not considered.'
|
|
||||||
echo 'For this to work, start the jail from ${SCRIPT_NAME}.'
|
|
||||||
start
|
|
||||||
}
|
|
||||||
EOF
|
|
||||||
|
|
||||||
echo "FROM CONF"
|
|
||||||
cat "${JAIL_CONFIG_PATH}"
|
|
||||||
chmod 700 "${JAIL_CONFIG_PATH}"
|
|
||||||
|
|
||||||
echo "Done creating the jail."
|
|
||||||
DONE=1
|
|
||||||
echo
|
|
||||||
read -p "Start the jail now? [Y/n] " -n 1 -r REPLY && echo
|
|
||||||
# Enter accepts default (yes)
|
|
||||||
if [[ "${REPLY}" =~ ^([Yy]|)$ ]]; then
|
|
||||||
run_jail "${JAIL_CONFIG_PATH}"
|
|
||||||
else
|
|
||||||
echo 'Skipped starting jail.'
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
create_jail
|
|
||||||
|
|
||||||
# TODO document
|
# TODO document
|
||||||
# machinectl shell
|
# machinectl shell
|
||||||
|
|
Loading…
Reference in New Issue