#!/bin/bash # Copyright (C) 2016 Ian Kelling # 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. # 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. # 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. [[ $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. # if [[ $type == fai ]]; then # echo "waiting for 1 dhcp ack then disabling fai server" # ack-wait 1 # faiserver-disable # fi fi