X-Git-Url: https://iankelling.org/git/?p=automated-distro-installer;a=blobdiff_plain;f=pxe-server;h=6137386a1188d9ab9c50e4bee783e7ee57dfc053;hp=2e8b2a40c267dcb6cdd8e79390b625e423a03583;hb=HEAD;hpb=a43d3eec943c5f0491b300214a3c744d31a6e542 diff --git a/pxe-server b/pxe-server index 2e8b2a4..1e99b03 100755 --- a/pxe-server +++ b/pxe-server @@ -1,36 +1,286 @@ -#!/bin/bash -x +#!/bin/bash +# Copyright (C) 2016 Ian Kelling -# usage: $0 [TYPE] -# default distro is the base debian/fedora type. others are fai & arch +# This program is free software; you can redistribute it and/or +# modify it under the terms of the GNU General Public License +# as published by the Free Software Foundation; either version 2 +# of the License, or (at your option) any later version. -set -eE -o pipefail -trap 'echo "$0:$LINENO:error: \"$BASH_COMMAND\" returned $?"' ERR +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. -action=${1:-default} +# You should have received a copy of the GNU General Public License +# along with this program; if not, write to the Free Software +# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. -arch() { - default - cat <<'EOF' -dhcp-option-force=209,boot/syslinux/archiso.cfg -dhcp-option-force=210,/arch/ -dhcp-boot=/arch/boot/syslinux/lpxelinux.0 + +[[ $EUID == 0 ]] || exec sudo -E "${BASH_SOURCE[0]}" "$@" + +readonly this_file="$(readlink -f -- "${BASH_SOURCE[0]}")" +script_dir="${this_file%/*}" +# shellcheck source=./bash-trace +source "${script_dir}/bash-trace" +cd $script_dir +PATH="$PATH:$PWD" + + +usage() { + cat < tftpboot + + +Notes on debugging pxe dhcp tftp: + +For debugging dhcp, add to /etc/dnsmasq.conf: log-dhcp + +Newer openwrt runs dnsmasq with a whitelist of readable files and dirs: + +ps ww : +/sbin/ujail -t 5 -n dnsmasq -u -l -r /bin/ubus -r /etc/TZ -r /etc/dnsmasq.conf -r /etc/ethers -r /etc/group -r /etc/hosts -r /etc/passwd -w /tmp/dhcp.leases -r /tmp/dnsmasq.d -r /tmp/hosts -r /tmp/resolv.conf.d -r /usr/bin/jshn -r /usr/lib/dnsmasq/dhcp-script.sh -r /usr/share/dnsmasq/dhcpbogushostname.conf -r /usr/share/dnsmasq/rfc6761.conf -r /usr/share/dnsmasq/trust-anchors.conf -r /usr/share/libubox/jshn.sh -r /var/etc/dnsmasq.conf.cfg01411c -w /var/run/dnsmasq/ -- /usr/sbin/dnsmasq -C /var/etc/dnsmasq.conf.cfg01411c -k -x /var/run/dnsmasq/dnsmasq.cfg01411c.pid + +logging tftp requests: +/etc/default/tftpd-hpa: +add -vv: +TFTP_OPTIONS="--secure -vv" +jr -u tftpd-hpa -f + +Note: Uses GNU getopt options parsing style +EOF + exit $1 +} + +pre="${0##*/}:" +m() { printf "$pre %s\n" "$*"; "$@"; } +e() { printf "$pre %s\n" "$*"; } +err() { echo "[$(date +'%Y-%m-%d %H:%M:%S%z')]: $pre: $*" >&2; } + +PATH="/a/exe:$PATH" + +##### begin command line parsing ######## + +dhcp=true +redep=true +acks=2 +wait=false +fsf_office=false + +case $HOSTNAME in + x3|kw) fsf_office=true ;; +esac + +chboot_args=() +temp=$(getopt -l no-r,help adkrSwh "$@") || usage 1 +eval set -- "$temp" +while true; do + case $1 in + -a) wait=true; set=false; shift ;; + -d) dhcp=false; shift ;; + -k) chboot_args+=(-k); shift ;; + --no-r) chboot_args+=(--no-r); shift ;; + -r) redep=false; shift ;; + -S) chboot_args+=(-S); shift ;; + -w) wait=true; set=true; shift ;; + -h|--help) usage ;; + --) shift; break ;; + *) echo "$0: Internal error!" ; exit 1 ;; + esac +done + +read -r host type <<<"$@" + +case $# in + [01]);; + 2) + case $type in + arch|parabola) cmd=archlike ;; + fai) cmd=fai ;; + *) + echo "$0: error expected type of arch|parabola|fai" + echo + usage 1 + ;; + esac + ;; + *) + echo "$0: error: expected 0-2 arguments" + echo + usage 1 + ;; +esac + + +if $wait && ! $dhcp; then + echo "$0: error -w conflicts with -d, choose one or other" >&2 + exit 1 +fi + +if $fsf_office && [[ ! $host ]]; then + echo "$0: at fsf_office, provide HOST arg" >&2 + exit 1 + fi + +if [[ $host && $host != default ]]; then + host_tag="tag:$host," +fi + +##### end command line parsing ######## + +archlike() { + cat </dev/null || sudo apt-get -y install dnsutils +faiserverip=$(host faiserver | sed -rn 's/^\S+ has address //p;T;q' ||:) +if [[ ! $faiserverip || $faiserverip =~ [[:space:]] ]]; then + echo "$0: error: failed to get \$faiserverip, got: $faiserverip" + exit 1 +fi + + +if $set; then + set-pxe + if [[ $type == fai ]]; then + if $redep; then + m fai-redep + fi + m myfai-chboot ${chboot_args[@]} $host + else + # This will fail if faiserver is not setup, so ignore any + # failure and don't bother us about it. + m myfai-chboot &>/dev/null ||: + fi +fi + +if $wait; then + # fai's debian jessie 8.5ish does 2 dhcp requests when booting, + # roughly 4 seconds apart. Earlier + # versions did just 1. Now testing on a vm, it does 1. + # bleh. + echo "waiting for $acks dhcp acks then disabling pxe" + ack-wait $acks + type= + unset cmd + set-pxe + + # previously tried waiting for one more ack then disabling faiserver, + # since it can contain sensitive info, so turn it off when not in use, + # but disabling that for now as it's inconvenient to clean this + # up and run it in the background etc. -$action | ssh wrt "cedit pxe-server /etc/dnsmasq.conf || /etc/init.d/dnsmasq restart -if [[ $action == arch ]]; then arch-pxe-mount; fi" + # if [[ $type == fai ]]; then + # echo "waiting for 1 dhcp ack then disabling fai server" + # ack-wait 1 + # faiserver-disable + # fi +fi