| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484 | #!/usr/bin/env bash# -*- coding: utf-8; mode: sh indent-tabs-mode: nil -*-# SPDX-License-Identifier: AGPL-3.0-or-later# shellcheck source=utils/lib.shsource "$(dirname "${BASH_SOURCE[0]}")/lib.sh"source_dot_config# load environment of the LXC suiteLXC_ENV="${LXC_ENV:-${REPO_ROOT}/utils/lxc-searx.env}"source "$LXC_ENV"lxc_set_suite_env# ----------------------------------------------------------------------------# config# ----------------------------------------------------------------------------## read also:# - https://lxd.readthedocs.io/en/latest/LXC_HOST_PREFIX="${LXC_HOST_PREFIX:-test}"# where all folders from HOST are mountedLXC_SHARE_FOLDER="/share"LXC_REPO_ROOT="${LXC_SHARE_FOLDER}/$(basename "${REPO_ROOT}")"ubu1604_boilerplate="export DEBIAN_FRONTEND=noninteractiveapt-get update -yapt-get upgrade -yapt-get install -y git curl wget"ubu1804_boilerplate="$ubu1604_boilerplate"ubu1904_boilerplate="$ubu1804_boilerplate"ubu1910_boilerplate="$ubu1904_boilerplate"# shellcheck disable=SC2034ubu2004_boilerplate="$ubu1910_boilerplateecho 'Set disable_coredump false' >> /etc/sudo.conf"# shellcheck disable=SC2034archlinux_boilerplate="pacman -Syu --noconfirmpacman -S --noconfirm git curl wget sudoecho 'Set disable_coredump false' >> /etc/sudo.conf"# shellcheck disable=SC2034fedora31_boilerplate="dnf update -ydnf install -y git curl wget hostnameecho 'Set disable_coredump false' >> /etc/sudo.conf"REMOTE_IMAGES=()CONTAINERS=()LOCAL_IMAGES=()for ((i=0; i<${#LXC_SUITE[@]}; i+=2)); do    REMOTE_IMAGES=("${REMOTE_IMAGES[@]}" "${LXC_SUITE[i]}")    CONTAINERS=("${CONTAINERS[@]}" "${LXC_HOST_PREFIX}-${LXC_SUITE[i+1]}")    LOCAL_IMAGES=("${LOCAL_IMAGES[@]}" "${LXC_SUITE[i+1]}")doneHOST_USER="${SUDO_USER:-$USER}"HOST_USER_ID=$(id -u "${HOST_USER}")HOST_GROUP_ID=$(id -g "${HOST_USER}")# ----------------------------------------------------------------------------usage() {# ----------------------------------------------------------------------------    _cmd="$(basename "$0")"    cat <<EOFusage::  $_cmd build        [containers]  $_cmd copy         [images]  $_cmd remove       [containers|<name>|images]  $_cmd [start|stop] [containers|<name>]  $_cmd show         [info|config|suite|images]  $_cmd cmd          [--|<name>] '...'  $_cmd install      [suite|base|buildhost]build  :containers:   build, launch and 'install basic' packages on 'containers'copy:  :images:       copy remote images of the suite into local storageremove  :containers:   delete all 'containers' or only <container-name>  :images:       delete local images of the suitestart/stop  :containers:   start/stop all 'containers' from the suite  :<name>:       start/stop conatiner <name> from suiteshow  :info:         show info of all the containers from LXC suite  :config:       show config of all the containers from the LXC suite  :suite:        show services of all the containers from the LXC suite  :images:       show information of local imagescmd  use single qoutes to evaluate in container's bash, e.g. 'echo $(hostname)'  --             run command '...' in all containers of the LXC suite  :<name>:       run command '...' in container <name>install  :suite:        install LXC suite; ${LXC_SUITE_INSTALL_INFO}  :base:         prepare LXC; install basic packages  :buildhost:    prepare LXC; install buildhost packagesEOF    usage_images    echo    usage_containers    echo    [ -n "${1+x}" ] &&  err_msg "$1"}usage_containers() {    cat <<EOFLXC suite containers:$(echo "  ${CONTAINERS[*]}" | $FMT)EOF    [ -n "${1+x}" ] &&  err_msg "$1"}usage_images() {    cat <<EOFLXC suite images:$(echo "  ${LOCAL_IMAGES[*]}" | $FMT)EOF}lxd_info() {    cat <<EOFLXD is needed, to install run::  snap install lxd  lxd init --autoEOF}main() {    local exit_val    local _usage="unknown or missing $1 command $2"    # don't check prerequisite when in recursion    if [[ ! $1 == __* ]]; then        ! required_commands lxc && lxd_info && exit 42        [[ -z $LXC_SUITE ]] && err_msg "missing LXC_SUITE" && exit 42     fi    case $1 in        --source-only)  ;;        -h|--help) usage; exit 0;;        build)            sudo_or_exit            case $2 in                ''|containers) build_instances ;;                *) usage "$_usage"; exit 42;;            esac            ;;        copy)            case $2 in                ''|images) lxc_copy_images_localy;;                *) usage "$_usage"; exit 42;;            esac            ;;        remove)            sudo_or_exit            case $2 in                ''|containers) remove_instances ;;                images) lxc_delete_images_localy ;;                ${LXC_HOST_PREFIX}-*)                    ! lxc_exists "$2" && usage_containers "unknown container: $2" && exit 42                    if ask_yn "Do you really want to delete conatiner $2"; then                        lxc_delete_container "$2"                    fi                    ;;                *) usage "uknown or missing container <name> $2"; exit 42;;            esac            ;;        start|stop)            sudo_or_exit            case $2 in                ''|containers)  lxc_cmd "$1" ;;                ${LXC_HOST_PREFIX}-*)                    ! lxc_exists "$2" && usage_containers "unknown container: $2" && exit 42                    info_msg "lxc $1 $2"                    lxc "$1" "$2" | prefix_stdout "[${_BBlue}${i}${_creset}] "                    ;;                *) usage "uknown or missing container <name> $2"; exit 42;;            esac            ;;        show)            sudo_or_exit            case $2 in                suite)  show_suite ;;                images) show_images ;;                config)                    rst_title "container configurations"                    echo                    lxc list "$LXC_HOST_PREFIX-"                    echo                    lxc_cmd config show                    ;;                info)                    rst_title "container info"                    echo                    lxc_cmd info                    ;;                *) usage "$_usage"; exit 42;;            esac            ;;        __show)            # wrapped show commands, called once in each container            case $2 in                suite) lxc_suite_info ;;            esac            ;;        cmd)            sudo_or_exit            shift            case $1 in                --) shift; lxc_exec "$@" ;;                ${LXC_HOST_PREFIX}-*)                    ! lxc_exists "$1" && usage_containers "unknown container: $1" && exit 42                    local name=$1                    shift                    lxc_exec_cmd "${name}" "$@"                    ;;                *) usage "uknown or missing container <name> $1"; exit 42;;            esac            ;;        install)            sudo_or_exit            case $2 in                suite|base|buildhost)                    lxc_exec "${LXC_REPO_ROOT}/utils/lxc.sh" __install "$2"                    ;;                *) usage "$_usage"; exit 42 ;;            esac            ;;        __install)            # wrapped install commands, called once in each container            # shellcheck disable=SC2119            case $2 in                suite) lxc_suite_install ;;                base) FORCE_TIMEOUT=0 lxc_install_base_packages ;;                buildhost) lxc_suite_prepare_buildhost ;;            esac            ;;        doc)            echo            echo ".. generic utils/lxc.sh documentation"            ;;        -*) usage "unknown option $1"; exit 42;;        *)  usage "unknown or missing command $1"; exit 42;;    esac}build_instances() {    rst_title "Build LXC instances"    lxc_copy_images_localy    echo    rst_title "build containers" section    echo    lxc_init_containers    lxc_config_containers    lxc_boilerplate_containers    echo    lxc_exec "${LXC_REPO_ROOT}/utils/lxc.sh" __install base    echo    lxc list "$LXC_HOST_PREFIX"}remove_instances() {    rst_title "Remove LXC instances"    rst_para "existing containers matching ${_BGreen}$LXC_HOST_PREFIX-*${_creset}"    echo    lxc list "$LXC_HOST_PREFIX-"    echo -en "\\n${_BRed}LXC containers to delete::${_creset}\\n\\n  ${CONTAINERS[*]}\\n" | $FMT    if ask_yn "Do you really want to delete these conatiners"; then        for i in "${CONTAINERS[@]}"; do            lxc_delete_container "$i"        done    fi    echo    lxc list "$LXC_HOST_PREFIX-"}# images# ------lxc_copy_images_localy() {    rst_title "copy images" section    echo    for ((i=0; i<${#LXC_SUITE[@]}; i+=2)); do        if lxc_image_exists "local:${LXC_SUITE[i+1]}"; then            info_msg "image ${LXC_SUITE[i]} already copied --> ${LXC_SUITE[i+1]}"        else            info_msg "copy image locally ${LXC_SUITE[i]} --> ${LXC_SUITE[i+1]}"            lxc image copy "${LXC_SUITE[i]}" local: \                --alias  "${LXC_SUITE[i+1]}" | prefix_stdout        fi    done    # lxc image list local: && wait_key}lxc_delete_images_localy() {    rst_title "Delete LXC images"    rst_para "local existing images"    echo    lxc image list local:    echo -en "\\n${_BRed}LXC images to delete::${_creset}\\n\\n  ${LOCAL_IMAGES[*]}\\n"    if ask_yn "Do you really want to delete these images"; then        for i in "${LOCAL_IMAGES[@]}"; do            lxc_delete_local_image "$i"        done    fi    echo    lxc image list local:}show_images(){    rst_title "local images"    echo    lxc image list local:    echo -en "\\n${_Green}LXC suite images::${_creset}\\n\\n  ${LOCAL_IMAGES[*]}\\n"    wait_key    for i in "${LOCAL_IMAGES[@]}"; do        if lxc_image_exists "$i"; then            info_msg "lxc image info ${_BBlue}${i}${_creset}"            lxc image info "$i" | prefix_stdout "[${_BBlue}${i}${_creset}] "        else            warn_msg "image ${_BBlue}$i${_creset} does not yet exists"        fi    done}# container# ---------show_suite(){    rst_title "LXC suite ($LXC_HOST_PREFIX-*)"    echo    lxc list "$LXC_HOST_PREFIX-"    echo    for i in "${CONTAINERS[@]}"; do        if ! lxc_exists "$i"; then            warn_msg "container ${_BBlue}$i${_creset} does not yet exists"        else            lxc exec -t "${i}" -- "${LXC_REPO_ROOT}/utils/lxc.sh" __show suite \                | prefix_stdout "[${_BBlue}${i}${_creset}]  "            echo        fi    done}lxc_cmd() {    for i in "${CONTAINERS[@]}"; do        if ! lxc_exists "$i"; then            warn_msg "container ${_BBlue}$i${_creset} does not yet exists"        else            info_msg "lxc $* $i"            lxc "$@" "$i" | prefix_stdout "[${_BBlue}${i}${_creset}] "            echo        fi    done}lxc_exec_cmd() {    local name="$1"    shift    exit_val=    info_msg "[${_BBlue}${name}${_creset}] ${_BGreen}${*}${_creset}"    lxc exec -t --cwd "${LXC_REPO_ROOT}" "${name}" -- bash -c "$*"    exit_val=$?    if [[ $exit_val -ne 0 ]]; then        warn_msg "[${_BBlue}${name}${_creset}] exit code (${_BRed}${exit_val}${_creset}) from ${_BGreen}${*}${_creset}"    else        info_msg "[${_BBlue}${name}${_creset}] exit code (${exit_val}) from ${_BGreen}${*}${_creset}"    fi}lxc_exec() {    for i in "${CONTAINERS[@]}"; do        if ! lxc_exists "$i"; then            warn_msg "container ${_BBlue}$i${_creset} does not yet exists"        else            lxc_exec_cmd "${i}" "$@" | prefix_stdout "[${_BBlue}${i}${_creset}] "        fi    done}lxc_init_containers() {    local image_name    local container_name    for ((i=0; i<${#LXC_SUITE[@]}; i+=2)); do        image_name="${LXC_SUITE[i+1]}"        container_name="${LXC_HOST_PREFIX}-${image_name}"        if lxc info "${container_name}" &>/dev/null; then            info_msg "container '${container_name}' already exists"        else            info_msg "create conatiner instance: ${container_name}"            lxc init "local:${image_name}" "${container_name}"        fi    done}lxc_config_containers() {    for i in "${CONTAINERS[@]}"; do        info_msg "[${_BBlue}${i}${_creset}] configure container ..."        info_msg "[${_BBlue}${i}${_creset}] map uid/gid from host to container"        # https://lxd.readthedocs.io/en/latest/userns-idmap/#custom-idmaps        echo -e -n "uid $HOST_USER_ID 0\\ngid $HOST_GROUP_ID 0"\            | lxc config set "$i" raw.idmap -        info_msg "[${_BBlue}${i}${_creset}] share ${REPO_ROOT} (repo_share) from HOST into container"        # https://lxd.readthedocs.io/en/latest/instances/#type-disk        lxc config device add "$i" repo_share disk \            source="${REPO_ROOT}" \            path="${LXC_REPO_ROOT}" &>/dev/null        # lxc config show "$i" && wait_key    done}lxc_boilerplate_containers() {    local image_name    local container_name    local boilerplate_script    for ((i=0; i<${#LXC_SUITE[@]}; i+=2)); do        image_name="${LXC_SUITE[i+1]}"        container_name="${LXC_HOST_PREFIX}-${image_name}"        boilerplate_script="${image_name}_boilerplate"        boilerplate_script="${!boilerplate_script}"        info_msg "[${_BBlue}${container_name}${_creset}] init .."        if lxc start -q "${container_name}" &>/dev/null; then            sleep 5 # guest needs some time to come up and get an IP        fi        lxc_init_container "${container_name}"        info_msg "[${_BBlue}${container_name}${_creset}] install /.lxcenv.mk .."        cat <<EOF | lxc exec "${container_name}" -- bash | prefix_stdout "[${_BBlue}${container_name}${_creset}] "rm -f "/.lxcenv.mk"ln -s "${LXC_REPO_ROOT}/utils/makefile.lxc" "/.lxcenv.mk"ls -l "/.lxcenv.mk"EOF        info_msg "[${_BBlue}${container_name}${_creset}] install boilerplate .."        if lxc start -q "${container_name}" &>/dev/null; then            sleep 5 # guest needs some time to come up and get an IP        fi        if [[ -n "${boilerplate_script}" ]]; then            echo "${boilerplate_script}" \                | lxc exec "${container_name}" -- bash \                | prefix_stdout "[${_BBlue}${container_name}${_creset}] "        else            err_msg "[${_BBlue}${container_name}${_creset}] no boilerplate for image '${image_name}'"        fi    done}# ----------------------------------------------------------------------------main "$@"# ----------------------------------------------------------------------------
 |