1829 lines
58 KiB
Plaintext
1829 lines
58 KiB
Plaintext
|
#!/bin/sh
|
||
|
#
|
||
|
# "SystemImager"
|
||
|
#
|
||
|
# Copyright (C) 1999-2011 Brian Elliott Finley <brian@thefinleys.com>
|
||
|
#
|
||
|
# $Id: functions 4559 2011-07-05 21:04:36Z finley $
|
||
|
# vi: set filetype=sh et ts=4:
|
||
|
#
|
||
|
# Others who have contributed to this code:
|
||
|
# Charles C. Bennett, Jr. <ccb@acm.org>
|
||
|
# Sean Dague <japh@us.ibm.com>
|
||
|
# Dann Frazier <dannf@dannf.org>
|
||
|
# Curtis Zinzilieta <czinzilieta@valinux.com>
|
||
|
#
|
||
|
|
||
|
|
||
|
################################################################################
|
||
|
#
|
||
|
# Variables
|
||
|
#
|
||
|
PATH=/sbin:/bin:/usr/bin:/usr/sbin:/tmp
|
||
|
LD_LIBRARY_PATH=/lib
|
||
|
SCRIPTS=scripts
|
||
|
SCRIPTS_DIR=/scripts
|
||
|
TORRENTS=torrents
|
||
|
TORRENTS_DIR=/torrents
|
||
|
FLAMETHROWER_DIRECTORY_DIR=/var/lib/systemimager/flamethrower
|
||
|
BOEL_BINARIES_DIR=/tmp/boel_binaries
|
||
|
VERSION="4.3.0"
|
||
|
FLAVOR="standard"
|
||
|
IMAGESERVER=$XCATMASTER
|
||
|
#
|
||
|
################################################################################
|
||
|
|
||
|
|
||
|
################################################################################
|
||
|
#
|
||
|
# Subroutines
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
# logmsg
|
||
|
#
|
||
|
# Usage: log a message, redirects to console / syslog depending on usage
|
||
|
logmsg() {
|
||
|
# log to console
|
||
|
echo $@
|
||
|
# log to temporary file (which will go away when we reboot)
|
||
|
# this is good for envs that have bad consoles
|
||
|
local FILE=/tmp/si.log
|
||
|
echo $@ >> $FILE || shellout
|
||
|
|
||
|
# if syslog is running, log to it. In order to avoid hangs we have to
|
||
|
# add the "logger: " part in case $@ is ""
|
||
|
if [ ! -z $USELOGGER ] ;
|
||
|
then logger "logger: $@"
|
||
|
fi
|
||
|
}
|
||
|
|
||
|
|
||
|
################################################################################
|
||
|
#
|
||
|
# check_version
|
||
|
#
|
||
|
# Usage: check_version
|
||
|
check_version() {
|
||
|
logmsg
|
||
|
logmsg check_version
|
||
|
INITRD_VERSION=$VERSION
|
||
|
KERNEL_VERSION=`uname -r | sed -e s/.*boel_v//`
|
||
|
if [ "$INITRD_VERSION" != "$KERNEL_VERSION" ] ; then
|
||
|
logmsg "FATAL: Kernel version ($KERNEL_VERSION) doesn't match initrd version ($INITRD_VERSION)!"
|
||
|
shellout
|
||
|
fi
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
# get_arch
|
||
|
#
|
||
|
# Usage: get_arch; echo $ARCH
|
||
|
get_arch() {
|
||
|
logmsg
|
||
|
logmsg get_arch
|
||
|
ARCH=`uname -m | sed -e s/i.86/i386/ -e s/sun4u/sparc64/ -e s/arm.*/arm/ -e s/sa110/arm/`
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
# adjust_arch
|
||
|
#
|
||
|
# based on info in /proc adjust the ARCH variable. This needs to run
|
||
|
# after proc is mounted.
|
||
|
#
|
||
|
adjust_arch() {
|
||
|
logmsg
|
||
|
logmsg adjust_arch
|
||
|
if [ "ppc64" = "$ARCH" ] ; then
|
||
|
# This takes a little bit of futzing with due to all the PPC platforms that exist.
|
||
|
if [ -d /proc/iSeries ] ; then
|
||
|
ARCH=ppc64-iSeries
|
||
|
logmsg "Detected ppc64 is really an iSeries partition..."
|
||
|
fi
|
||
|
if grep -qs PS3 /proc/cpuinfo; then
|
||
|
ARCH=ppc64-ps3
|
||
|
fi
|
||
|
fi
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
# write_variables
|
||
|
#
|
||
|
# Usage: write_variables
|
||
|
write_variables() {
|
||
|
|
||
|
logmsg
|
||
|
logmsg write_variables
|
||
|
|
||
|
# pass all variables set here on to the hostname.sh script
|
||
|
rm -f /tmp/variables.txt
|
||
|
|
||
|
echo "HOSTNAME=$HOSTNAME" >> /tmp/variables.txt || shellout
|
||
|
echo "DOMAINNAME=$DOMAINNAME" >> /tmp/variables.txt
|
||
|
|
||
|
echo "DEVICE=$DEVICE" >> /tmp/variables.txt
|
||
|
echo "IPADDR=$IPADDR" >> /tmp/variables.txt
|
||
|
echo "NETMASK=$NETMASK" >> /tmp/variables.txt
|
||
|
echo "NETWORK=$NETWORK" >> /tmp/variables.txt
|
||
|
echo "BROADCAST=$BROADCAST" >> /tmp/variables.txt
|
||
|
|
||
|
echo "GATEWAY=$GATEWAY" >> /tmp/variables.txt
|
||
|
echo "GATEWAYDEV=$GATEWAYDEV" >> /tmp/variables.txt
|
||
|
|
||
|
echo "IMAGESERVER=$IMAGESERVER" >> /tmp/variables.txt
|
||
|
echo "IMAGENAME=$IMAGENAME" >> /tmp/variables.txt
|
||
|
|
||
|
echo "LOG_SERVER=$LOG_SERVER" >> /tmp/variables.txt
|
||
|
echo "LOG_SERVER_PORT=$LOG_SERVER_PORT" >> /tmp/variables.txt
|
||
|
echo "USELOGGER=$USELOGGER" >> /tmp/variables.txt
|
||
|
|
||
|
echo "TMPFS_STAGING=$TMPFS_STAGING" >> /tmp/variables.txt
|
||
|
|
||
|
echo "SSH=$SSH" >> /tmp/variables.txt
|
||
|
echo "SSHD=$SSHD" >> /tmp/variables.txt
|
||
|
echo "SSH_USER=$SSH_USER" >> /tmp/variables.txt
|
||
|
echo "SSH_DOWNLOAD_URL=$SSH_DOWNLOAD_URL" >> /tmp/variables.txt
|
||
|
|
||
|
echo "FLAMETHROWER_DIRECTORY_PORTBASE=$FLAMETHROWER_DIRECTORY_PORTBASE" >> /tmp/variables.txt
|
||
|
|
||
|
echo "MONITOR_SERVER=$MONITOR_SERVER" >> /tmp/variables.txt
|
||
|
echo "MONITOR_PORT=$MONITOR_PORT" >> /tmp/variables.txt
|
||
|
echo "MONITOR_CONSOLE=$MONITOR_CONSOLE" >> /tmp/variables.txt
|
||
|
|
||
|
echo "BITTORRENT=$BITTORRENT" >> /tmp/variables.txt
|
||
|
echo "BITTORRENT_STAGING=$BITTORRENT_STAGING" >> /tmp/variables.txt
|
||
|
echo "BITTORRENT_POLLING_TIME=$BITTORRENT_POLLING_TIME" >> /tmp/variables.txt
|
||
|
echo "BITTORRENT_SEED_WAIT=$BITTORRENT_SEED_WAIT" >> /tmp/variables.txt
|
||
|
echo "BITTORRENT_UPLOAD_MIN=$BITTORRENT_UPLOAD_MIN" >> /tmp/variables.txt
|
||
|
|
||
|
echo "GROUPNAMES=\"$GROUPNAMES\"" >> /tmp/variables.txt
|
||
|
echo "GROUP_OVERRIDES=\"$GROUP_OVERRIDES\"" >> /tmp/variables.txt
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
# Description:
|
||
|
# watches the tmpfs filesystem (/) and gives warnings and/or does a shellout
|
||
|
#
|
||
|
# Usage: tmpfs_watcher
|
||
|
#
|
||
|
tmpfs_watcher() {
|
||
|
|
||
|
logmsg
|
||
|
logmsg tmpfs_watcher
|
||
|
|
||
|
# Note: Transfer to staging area can fail if tmpfs runs out of inodes.
|
||
|
{
|
||
|
while :; do
|
||
|
DF=`df -k / | egrep ' /$' | sed -e 's/ */ /g' -e 's/.*[0-9] //' -e 's/%.*//'`
|
||
|
[ $DF -ge 95 ] && logmsg "WARNING: Your tmpfs filesystem is ${DF}% full!"
|
||
|
[ $DF -ge 99 ] && logmsg " Search the FAQ for tmpfs to learn about sizing options."
|
||
|
[ $DF -ge 99 ] && shellout
|
||
|
sleep 1
|
||
|
done
|
||
|
unset DF
|
||
|
}&
|
||
|
TMPFS_WATCHER_PID=$!
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
# Description:
|
||
|
# Exit with the message stored in /etc/issue.
|
||
|
#
|
||
|
# Usage: $COMMAND || shellout
|
||
|
#
|
||
|
shellout() {
|
||
|
|
||
|
logmsg "Last command exited with $?"
|
||
|
COUNT="$RETRY"
|
||
|
logmsg "Killing off running processes."
|
||
|
kill -9 $TMPFS_WATCHER_PID >/dev/null 2>/dev/null
|
||
|
killall -9 udp-receiver rsync >/dev/null 2>/dev/null
|
||
|
write_variables
|
||
|
cat /etc/issue
|
||
|
if [ ! -z $USELOGGER ] ;
|
||
|
then cat /etc/issue | logger
|
||
|
fi
|
||
|
if [ ! -z $MONITOR_SERVER ]; then
|
||
|
logmsg "Installation failed!! Stopping report task."
|
||
|
stop_report_task -1
|
||
|
fi
|
||
|
exec sh > /dev/console 2>&1
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
# Description:
|
||
|
# Count the specified number, printing each number, and exit only the count
|
||
|
# loop when <ctrl>+<c> is hit (SIGINT, or Signal 2). Thanks to
|
||
|
# CCB <ccb@acm.org> for this chunk of code. -BEF-
|
||
|
#
|
||
|
# Usage:
|
||
|
# count_loop 35
|
||
|
# count_loop $ETHER_SLEEP
|
||
|
#
|
||
|
count_loop() {
|
||
|
|
||
|
COUNT=$1
|
||
|
|
||
|
trap 'echo ; echo "Skipping ETHER_SLEEP -> Caught <ctrl>+<c>" ; I=$COUNT' INT
|
||
|
|
||
|
I=0
|
||
|
while [ $I -lt $COUNT ]; do
|
||
|
I=$(( $I + 1 ))
|
||
|
logmsg -n "$I "
|
||
|
sleep 1
|
||
|
done
|
||
|
|
||
|
trap INT
|
||
|
|
||
|
logmsg
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
# Usage: get_torrents_directory
|
||
|
#
|
||
|
get_torrents_directory() {
|
||
|
if [ ! "x$BITTORRENT" = "xy" ]; then
|
||
|
return
|
||
|
fi
|
||
|
|
||
|
logmsg
|
||
|
logmsg get_torrents_directory
|
||
|
|
||
|
if [ ! -z $FLAMETHROWER_DIRECTORY_PORTBASE ]; then
|
||
|
#
|
||
|
# We're using Multicast, so we should already have a directory
|
||
|
# full of scripts. Break out here, so that we don't try to pull
|
||
|
# the scripts dir again (that would be redundant).
|
||
|
#
|
||
|
MODULE_NAME="autoinstall_torrents"
|
||
|
DIR="${SCRIPTS_DIR}"
|
||
|
RETRY=7
|
||
|
flamethrower_client
|
||
|
|
||
|
else
|
||
|
mkdir -p ${TORRENTS_DIR}
|
||
|
CMD="rsync -a ${IMAGESERVER}::${TORRENTS}/ ${TORRENTS_DIR}/"
|
||
|
logmsg "$CMD"
|
||
|
$CMD
|
||
|
fi
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
# Usage: get_scripts_directory
|
||
|
#
|
||
|
get_scripts_directory() {
|
||
|
logmsg
|
||
|
logmsg get_scripts_directory
|
||
|
|
||
|
if [ ! -z $FLAMETHROWER_DIRECTORY_PORTBASE ]; then
|
||
|
#
|
||
|
# We're using Multicast, so we should already have a directory
|
||
|
# full of scripts. Break out here, so that we don't try to pull
|
||
|
# the scripts dir again (that would be redundant).
|
||
|
#
|
||
|
MODULE_NAME="autoinstall_scripts"
|
||
|
DIR="${SCRIPTS_DIR}"
|
||
|
RETRY=7
|
||
|
flamethrower_client
|
||
|
|
||
|
else
|
||
|
mkdir -p ${SCRIPTS_DIR}
|
||
|
CMD="rsync -a ${IMAGESERVER}::${SCRIPTS}/ ${SCRIPTS_DIR}/"
|
||
|
logmsg "$CMD"
|
||
|
$CMD || shellout
|
||
|
fi
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
# Usage: get_flamethrower_directory
|
||
|
#
|
||
|
get_flamethrower_directory() {
|
||
|
logmsg
|
||
|
logmsg Using multicast...
|
||
|
logmsg get_flamethrower_directory
|
||
|
|
||
|
MODULE_NAME=flamethrower_directory
|
||
|
DIR=${FLAMETHROWER_DIRECTORY_DIR}
|
||
|
RETRY=7
|
||
|
PORTBASE=9000
|
||
|
flamethrower_client
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
# Usage:
|
||
|
#
|
||
|
# MODULE_NAME=my_module # Required
|
||
|
# DIR=/my/destination/dir # Required
|
||
|
# [RETRY=7] # Optional
|
||
|
# [PORTBASE=9000] # Required if a sourceable file called $MODULE_NAME
|
||
|
# # doesn't exist
|
||
|
# [FLAMETHROWER_TARPIPE] # If not empty, untar received data directly,
|
||
|
# # without storing it to a temporary file
|
||
|
#
|
||
|
# flamethrower_client
|
||
|
#
|
||
|
flamethrower_client() {
|
||
|
|
||
|
if [ ! -z $FLAMETHROWER_TARPIPE ]; then
|
||
|
FLAMETHROWER_TARPIPE=tarpipe
|
||
|
fi
|
||
|
logmsg
|
||
|
logmsg "flamethrower_client(${MODULE_NAME}) $FLAMETHROWER_TARPIPE "
|
||
|
logmsg ---------------------------------------------------------------------
|
||
|
|
||
|
# validate
|
||
|
if [ -z $PORTBASE ]; then
|
||
|
if [ -f ${FLAMETHROWER_DIRECTORY_DIR}/${MODULE_NAME} ]; then
|
||
|
. ${FLAMETHROWER_DIRECTORY_DIR}/${MODULE_NAME}
|
||
|
else
|
||
|
logmsg WARNING WARNING WARNING WARNING WARNING
|
||
|
logmsg You must either set PORTBASE, or have a sourceable file called
|
||
|
logmsg ${FLAMETHROWER_DIRECTORY_DIR}/MODULE_NAME
|
||
|
# allow for now to continue until overrides get their modules
|
||
|
return
|
||
|
#shellout
|
||
|
fi
|
||
|
fi
|
||
|
if [ -z $DIR ]; then
|
||
|
logmsg "Must set DIR !!!"
|
||
|
shellout
|
||
|
else
|
||
|
mkdir -p $DIR
|
||
|
fi
|
||
|
|
||
|
# build command
|
||
|
UDP_RECEIVER_OPTIONS="--interface ${DEVICE} --portbase $PORTBASE --nokbd"
|
||
|
if [ ! -z $TTL ]; then
|
||
|
UDP_RECEIVER_OPTIONS="$UDP_RECEIVER_OPTIONS --ttl $TTL"
|
||
|
fi
|
||
|
if [ "$NOSYNC" = "on" ]; then
|
||
|
UDP_RECEIVER_OPTIONS="$UDP_RECEIVER_OPTIONS --nosync"
|
||
|
fi
|
||
|
if [ "$ASYNC" = "on" ]; then
|
||
|
UDP_RECEIVER_OPTIONS="$UDP_RECEIVER_OPTIONS --async"
|
||
|
fi
|
||
|
if [ ! -z $MCAST_ALL_ADDR ]; then
|
||
|
UDP_RECEIVER_OPTIONS="$UDP_RECEIVER_OPTIONS --mcast-all-addr $MCAST_ALL_ADDR"
|
||
|
fi
|
||
|
|
||
|
# Which tar opts should we use? If our tar has --overwrite capability, use it.
|
||
|
# Summary: busybox tar doesn't (boel_binaries and prior).
|
||
|
# Debian patched gnu tar does (image and on).
|
||
|
# We want this option enabled for the image to ensure proper directory
|
||
|
# permissions. -BEF-
|
||
|
tar --help 2>&1 | grep -q overwrite && TAR_OPTS='--overwrite -xp' || TAR_OPTS='-x'
|
||
|
|
||
|
# set vars
|
||
|
[ -z $RETRY ] && RETRY=0
|
||
|
COUNT=0
|
||
|
FLAMETHROWER_CLIENT_SLEEP=3
|
||
|
|
||
|
# it's the new new style (loop)
|
||
|
SUCCESS="Not Yet"
|
||
|
until [ "$SUCCESS" = "yes" ]
|
||
|
do
|
||
|
# receive cast
|
||
|
# example udp-receiver command:
|
||
|
# udp-receiver --interface lo --portbase 9002 --nokbd --nosync --file /tmp/multicast.tar
|
||
|
|
||
|
if [ ! -z $FLAMETHROWER_TARPIPE ]; then
|
||
|
TAR_OPTS="$TAR_OPTS -f -"
|
||
|
logmsg "udp-receiver $UDP_RECEIVER_OPTIONS | tar $TAR_OPTS -C $DIR"
|
||
|
udp-receiver $UDP_RECEIVER_OPTIONS | tar $TAR_OPTS -C $DIR
|
||
|
TAR_EXIT_STATUS=$?
|
||
|
UDP_RECEIVER_EXIT_STATUS=0
|
||
|
else
|
||
|
logmsg udp-receiver $UDP_RECEIVER_OPTIONS --file /tmp/multicast.tar
|
||
|
udp-receiver $UDP_RECEIVER_OPTIONS --file /tmp/multicast.tar
|
||
|
UDP_RECEIVER_EXIT_STATUS=$?
|
||
|
|
||
|
# untar it
|
||
|
if [ "$UDP_RECEIVER_EXIT_STATUS" = "0" ]; then
|
||
|
logmsg tar ${TAR_OPTS} -f /tmp/multicast.tar -C ${DIR}
|
||
|
tar ${TAR_OPTS} -f /tmp/multicast.tar -C ${DIR}
|
||
|
TAR_EXIT_STATUS=$?
|
||
|
fi
|
||
|
# discard used tarball like an old sock (recommended by: Ramon Bastiaans <bastiaans@sara.nl>)
|
||
|
rm -f /tmp/multicast.tar
|
||
|
fi
|
||
|
|
||
|
# did everything work properly
|
||
|
if [ $UDP_RECEIVER_EXIT_STATUS -eq 0 ] && [ $TAR_EXIT_STATUS -eq 0 ]; then
|
||
|
SUCCESS=yes
|
||
|
else
|
||
|
if [ $COUNT -lt $RETRY ]; then
|
||
|
COUNT=$(( $COUNT + 1 ))
|
||
|
logmsg "flamethrower_client: Proceeding with retry $COUNT of $RETRY"
|
||
|
else
|
||
|
logmsg
|
||
|
logmsg "flamethrower_client: FATAL: Initial attempt and $RETRY retries failed!"
|
||
|
shellout
|
||
|
fi
|
||
|
fi
|
||
|
|
||
|
# sleep apnea
|
||
|
sleep_loop $FLAMETHROWER_CLIENT_SLEEP
|
||
|
done
|
||
|
|
||
|
# done
|
||
|
logmsg 'finished!'
|
||
|
logmsg
|
||
|
|
||
|
# Unset vars, so next module (which may not have them set) won't use then unintentially
|
||
|
unset TTL
|
||
|
unset NOSYNC
|
||
|
unset ASYNC
|
||
|
unset MCAST_ALL_ADDR
|
||
|
unset RETRY
|
||
|
unset COUNT
|
||
|
unset DIR
|
||
|
unset PORTBASE
|
||
|
unset UDP_RECEIVER_EXIT_STATUS
|
||
|
unset UDP_RECEIVER_OPTIONS
|
||
|
unset TAR_EXIT_STATUS
|
||
|
unset TAR_OPTS
|
||
|
unset SUCCESS
|
||
|
unset FLAMETHROWER_TARPIPE
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
# Autodetect a staging directory for the bittorrent tarball
|
||
|
#
|
||
|
# Usage: bittorrent_autodetect_staging_dir torrent
|
||
|
#
|
||
|
bittorrent_autodetect_staging_dir() {
|
||
|
torrent=$1
|
||
|
if [ ! -f $torrent ]; then
|
||
|
logmsg "warning: torrent file $torrent does not exist!"
|
||
|
return
|
||
|
fi
|
||
|
|
||
|
# List of preferred staging directory (/tmp = ramdisk staging)
|
||
|
preferred_dirs="/tmp /a/tmp `df 2>/dev/null | sed '1d' | sed 's/[[:space:]]\+/ /g' | cut -d' ' -f6`"
|
||
|
|
||
|
# Use a breathing room of 100MB (this should be enough for a lot of cases)
|
||
|
breathing_room=102400
|
||
|
|
||
|
# Evaluate torrent size
|
||
|
torrent_size=$((`/bin/torrentinfo-console $torrent | sed -ne 's/^file size\.*: \([0-9]\+\).*$/\1/p'` / 1024 + $breathing_room))
|
||
|
|
||
|
# Find a directory to host the image tarball
|
||
|
for dir in $preferred_dirs; do
|
||
|
[ ! -d $dir ] && continue;
|
||
|
dir_space=`df $dir 2>/dev/null | sed '1d' | sed 's/[[:space:]]\+/ /g' | cut -d' ' -f4 | sed -ne '$p'`
|
||
|
[ -z $dir_space ] && continue
|
||
|
[ $torrent_size -lt $dir_space ] && echo $dir && return
|
||
|
done
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
# Download a file using bittorrent.
|
||
|
#
|
||
|
# Usage: bittorrent_get_file torrent destination
|
||
|
#
|
||
|
bittorrent_get_file() {
|
||
|
torrent=$1
|
||
|
destination=$2
|
||
|
|
||
|
# Bittorrent log file
|
||
|
bittorrent_log=/tmp/bittorrent-`basename ${torrent}`.log
|
||
|
# Time to poll bittorrent events
|
||
|
bittorrent_polling_time=${BITTORRENT_POLLING_TIME:-5}
|
||
|
# Wait after the download is finished to seed the other peers
|
||
|
bittorrent_seed_wait=${BITTORRENT_SEED_WAIT:-n}
|
||
|
# Minimum upload rate threshold (in KB/s), if lesser stop seeding
|
||
|
bittorrent_upload_min=${BITTORRENT_UPLOAD_MIN:-50}
|
||
|
|
||
|
# Start downloading.
|
||
|
/bin/bittorrent-console --no_upnp --no_start_trackerless_client --max_upload_rate 0 --display_interval 1 --rerequest_interval 1 --bind ${IPADDR} --save_in ${destination} ${torrent} > $bittorrent_log &
|
||
|
pid=$!
|
||
|
if [ ! -d /proc/$pid ]; then
|
||
|
logmsg "error: couldn't run bittorrent-console!"
|
||
|
shellout
|
||
|
fi
|
||
|
unset pid
|
||
|
|
||
|
# Wait for BitTorrent log to appear.
|
||
|
while [ ! -e $bittorrent_log ]; do
|
||
|
sleep 1
|
||
|
done
|
||
|
|
||
|
# Checking download...
|
||
|
while :; do
|
||
|
while :; do
|
||
|
status=`grep 'percent done:' $bittorrent_log | sed -ne '$p' | sed 's/percent done: *//' | sed -ne '/^[0-9]*\.[0-9]*$/p'`
|
||
|
[ ! -z "$status" ] && break
|
||
|
done
|
||
|
logmsg "percent done: $status %"
|
||
|
if [ "$status" = "100.0" ]; then
|
||
|
# Sleep until upload rate reaches the minimum threshold
|
||
|
while [ "$bittorrent_seed_wait" = "y" ]; do
|
||
|
sleep $bittorrent_polling_time
|
||
|
while :; do
|
||
|
upload_rate=`grep 'upload rate:' $bittorrent_log | sed -ne '$p' | sed 's/upload rate: *\([0-9]*\)\.[0-9]* .*$/\1/' | sed -ne '/^\([0-9]*\)$/p'`
|
||
|
[ ! -z $upload_rate ] && break
|
||
|
done
|
||
|
logmsg "upload rate: $upload_rate KB/s"
|
||
|
[ $upload_rate -lt $bittorrent_upload_min ] && break
|
||
|
done
|
||
|
logmsg "Download completed"
|
||
|
unset bittorrent_log upload_rate counter
|
||
|
break
|
||
|
fi
|
||
|
sleep $bittorrent_polling_time
|
||
|
done
|
||
|
|
||
|
unset bittorrent_polling_time
|
||
|
unset bittorrent_seed_wait
|
||
|
unset bittorrent_upload_min
|
||
|
unset torrent
|
||
|
unset destination
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
# Stop bittorrent client.
|
||
|
#
|
||
|
# Usage: bittorrent_stop
|
||
|
#
|
||
|
bittorrent_stop() {
|
||
|
# Try to kill all the BitTorrent processes
|
||
|
btclient="bittorrent-console"
|
||
|
|
||
|
logmsg "killing BitTorrent client..."
|
||
|
killall -15 $btclient >/dev/null 2>&1
|
||
|
|
||
|
# Forced kill after 5 secs.
|
||
|
sleep 5
|
||
|
killall -9 $btclient >/dev/null 2>&1
|
||
|
|
||
|
# Remove bittorrent logs.
|
||
|
rm -rf /tmp/bittorrent*.log
|
||
|
unset btclient
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
# Get other binaries, kernel module tree, and miscellaneous other stuff that
|
||
|
# got put in the binaries tarball. -BEF-
|
||
|
#
|
||
|
get_boel_binaries_tarball() {
|
||
|
|
||
|
logmsg
|
||
|
logmsg get_boel_binaries_tarball
|
||
|
# mkdir -p ${BOEL_BINARIES_DIR}
|
||
|
#
|
||
|
# if [ ! -z $SSH_DOWNLOAD_URL ]; then
|
||
|
# # If we're using SSH, get the boel_binaries from a web server.
|
||
|
# logmsg "SSH_DOWNLOAD_URL variable is set, so we will install over SSH!"
|
||
|
#
|
||
|
# if [ ! -z $FLAMETHROWER_DIRECTORY_PORTBASE ]; then
|
||
|
# logmsg "FLAMETHROWER_DIRECTORY_PORTBASE is also set, but I will be conservative and proceed with SSH."
|
||
|
# fi
|
||
|
#
|
||
|
# # Remove possible trailing / from URL
|
||
|
# SSH_DOWNLOAD_URL=`echo $SSH_DOWNLOAD_URL | sed 's/\/$//'`
|
||
|
#
|
||
|
# cd ${BOEL_BINARIES_DIR}
|
||
|
# CMD="wget ${SSH_DOWNLOAD_URL}/${ARCH}/${FLAVOR}/boel_binaries.tar.gz"
|
||
|
# logmsg "$CMD"
|
||
|
# $CMD || shellout
|
||
|
#
|
||
|
# elif [ "x$BITTORRENT" = "xy" ]; then
|
||
|
#
|
||
|
# # Download BOEL binaries from peers
|
||
|
# bittorrent_tarball="boel_binaries.tar.gz"
|
||
|
# logmsg "Start downloading ${bittorrent_tarball} (${ARCH}) using bittorrent"
|
||
|
# logmsg ""
|
||
|
# logmsg "--> INFO: remember to start /etc/init.d/systemimager-server-bittorrent on the image server!"
|
||
|
# logmsg ""
|
||
|
# bittorrent_get_file ${TORRENTS_DIR}/${ARCH}-${bittorrent_tarball}.torrent ${BOEL_BINARIES_DIR}
|
||
|
# cd ${BOEL_BINARIES_DIR} && mv ${ARCH}-${bittorrent_tarball} ${bittorrent_tarball}
|
||
|
# unset bittorrent_tarball
|
||
|
#
|
||
|
# elif [ ! -z $FLAMETHROWER_DIRECTORY_PORTBASE ]; then
|
||
|
#
|
||
|
# MODULE_NAME="boot-${ARCH}-${FLAVOR}"
|
||
|
# DIR="${BOEL_BINARIES_DIR}"
|
||
|
# RETRY=7
|
||
|
# flamethrower_client
|
||
|
#
|
||
|
# else
|
||
|
# # Use rsync
|
||
|
# CMD="rsync -av ${IMAGESERVER}::boot/${ARCH}/${FLAVOR}/boel_binaries.tar.gz ${BOEL_BINARIES_DIR}"
|
||
|
# logmsg "$CMD"
|
||
|
# $CMD || shellout
|
||
|
# fi
|
||
|
#
|
||
|
# # Untar the tarball
|
||
|
# tar -C / -xzf ${BOEL_BINARIES_DIR}/boel_binaries.tar.gz || shellout
|
||
|
# chown -R 0.0 /lib/modules || shellout
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
# Parse tmpfs options from /proc/cpuinfo
|
||
|
#
|
||
|
parse_tmpfs_opts() {
|
||
|
logmsg
|
||
|
logmsg parse_tmpfs_opts
|
||
|
tmpfs_size=$(tr ' ' '\n' < /proc/cmdline | grep tmpfs_size\= | sed 's/.*=//')
|
||
|
tmpfs_nr_blocks=$(tr ' ' '\n' < /proc/cmdline | grep tmpfs_nr_blocks\= | sed 's/.*=//')
|
||
|
tmpfs_nr_inodes=$(tr ' ' '\n' < /proc/cmdline | grep tmpfs_nr_inodes\= | sed 's/.*=//')
|
||
|
tmpfs_mode=$(tr ' ' '\n' < /proc/cmdline | grep tmpfs_mode\= | sed 's/.*=//')
|
||
|
|
||
|
if [ "$tmpfs_size" != "" ]; then
|
||
|
tmpfs_opts="size=$tmpfs_size"
|
||
|
fi
|
||
|
|
||
|
if [ "$tmpfs_nr_blocks" != "" ]; then
|
||
|
if [ "$tmpfs_opts" != "" ]; then
|
||
|
tmpfs_opts="${tmpfs_opts},nr_blocks=$tmpfs_nr_blocks"
|
||
|
else
|
||
|
tmpfs_opts="nr_blocks=$tmpfs_nr_blocks"
|
||
|
fi
|
||
|
fi
|
||
|
|
||
|
if [ "$tmpfs_nr_inodes" != "" ]; then
|
||
|
if [ "$tmpfs_opts" != "" ]; then
|
||
|
tmpfs_opts="${tmpfs_opts},nr_inodes=$tmpfs_nr_inodes"
|
||
|
else
|
||
|
tmpfs_opts="nr_inodes=$tmpfs_nr_inodes"
|
||
|
fi
|
||
|
fi
|
||
|
|
||
|
if [ "$tmpfs_mode" != "" ]; then
|
||
|
if [ "$tmpfs_opts" != "" ]; then
|
||
|
tmpfs_opts="${tmpfs_opts},mode=$tmpfs_mode"
|
||
|
else
|
||
|
tmpfs_opts="mode=$tmpfs_mode"
|
||
|
fi
|
||
|
fi
|
||
|
|
||
|
if [ "$tmpfs_opts" != "" ]; then
|
||
|
tmpfs_opts="-o $tmpfs_opts"
|
||
|
fi
|
||
|
|
||
|
unset tmpfs_size
|
||
|
unset tmpfs_nr_blocks
|
||
|
unset tmpfs_nr_inodes
|
||
|
unset tmpfs_mode
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
# Switch root to tmpfs
|
||
|
#
|
||
|
#switch_root_to_tmpfs() {
|
||
|
# local MODULE=tmpfs
|
||
|
# logmsg
|
||
|
# logmsg switch_root_to_tmpfs
|
||
|
# logmsg "Loading $MODULE... "
|
||
|
# modprobe $MODULE 2>/dev/null && logmsg "done!" || logmsg "Didn't load -- assuming it's built into the kernel."
|
||
|
# parse_tmpfs_opts
|
||
|
#
|
||
|
# # Switch root over to tmpfs so we don't have to worry about the size of
|
||
|
# # the tarball and binaries that users may decide to copy over. -BEF-
|
||
|
# if [ -d /old_root ]; then
|
||
|
# logmsg
|
||
|
# logmsg "already switched to tmpfs..."
|
||
|
# else
|
||
|
# logmsg
|
||
|
# logmsg "switching root to tmpfs..."
|
||
|
#
|
||
|
# mkdir -p /new_root || shellout
|
||
|
# mount tmpfs /new_root -t tmpfs $tmpfs_opts || shellout
|
||
|
#
|
||
|
# cd / || shellout
|
||
|
# cp -a `/bin/ls | grep -v -E '^(new_root|dev)$'` /new_root/ || shellout
|
||
|
#
|
||
|
# mkdir -p /new_root/dev || shellout
|
||
|
# mount -t devtmpfs -o mode=0755 none /new_root/dev || shellout
|
||
|
#
|
||
|
# cd /new_root || shellout
|
||
|
# mkdir -p old_root || shellout
|
||
|
# pivot_root . old_root || switch_root
|
||
|
# fi
|
||
|
#
|
||
|
# unset tmpfs_opts
|
||
|
#}
|
||
|
#
|
||
|
################################################################################
|
||
|
################################################################################
|
||
|
#
|
||
|
mount_initial_filesystems() {
|
||
|
|
||
|
# Much of this taken from "init" from an Ubuntu Lucid initrd.img
|
||
|
logmsg
|
||
|
logmsg mount_initial_filesystems
|
||
|
|
||
|
[ -d /dev ] || mkdir -m 0755 /dev
|
||
|
[ -d /root ] || mkdir -m 0700 /root
|
||
|
[ -d /sys ] || mkdir /sys
|
||
|
[ -d /proc ] || mkdir /proc
|
||
|
[ -d /tmp ] || mkdir /tmp
|
||
|
|
||
|
mkdir -p /var/lock
|
||
|
|
||
|
mount -t sysfs -o nodev,noexec,nosuid none /sys
|
||
|
mount -t proc -o nodev,noexec,nosuid none /proc
|
||
|
|
||
|
# Note that this only becomes /dev on the real filesystem if udev's scripts
|
||
|
# are used; which they will be, but it's worth pointing out
|
||
|
if ! mount -t devtmpfs -o mode=0755 none /dev; then
|
||
|
mount -t tmpfs -o mode=0755 none /dev
|
||
|
mknod -m 0600 /dev/console c 5 1
|
||
|
mknod -m 0666 /dev/null c 1 3
|
||
|
mknod -m 0660 /dev/kmsg c 1 11
|
||
|
fi
|
||
|
|
||
|
mkdir /dev/pts
|
||
|
mount -t devpts -o noexec,nosuid,gid=5,mode=0620 none /dev/pts || true
|
||
|
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
################################################################################
|
||
|
#
|
||
|
monitor_save_dmesg() {
|
||
|
if [ -z $MONITOR_SERVER ]; then
|
||
|
return
|
||
|
fi
|
||
|
logmsg
|
||
|
logmsg monitor_save_dmesg
|
||
|
dmesg -s 16392 > /tmp/si_monitor.log
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
start_udevd() {
|
||
|
logmsg
|
||
|
logmsg start_udevd
|
||
|
echo -n " "
|
||
|
/etc/init.d/udev start
|
||
|
|
||
|
# If udev failed fall back to a static /dev
|
||
|
if [ $? -ne 0 ]; then
|
||
|
logmsg "failed!"
|
||
|
#XXX -delete me -BEF- - logmsg "Creating a static /dev..."
|
||
|
# cd / && tar -xzf dev.tar.gz || shellout
|
||
|
logmsg done
|
||
|
fi
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
# Configure loopback interface (may as well)
|
||
|
ifconfig_loopback() {
|
||
|
logmsg
|
||
|
logmsg ifconfig_loopback
|
||
|
ifconfig lo 127.0.0.1
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
# Load any modules that were placed in the my_modules directory prior to
|
||
|
# running "make initrd.gz". -BEF-
|
||
|
load_my_modules() {
|
||
|
logmsg
|
||
|
logmsg load_my_modules
|
||
|
cd /my_modules || shellout
|
||
|
sh ./INSMOD_COMMANDS
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
# read in varibles obtained from kernel appends
|
||
|
#
|
||
|
read_kernel_append_parameters() {
|
||
|
logmsg
|
||
|
logmsg read_kernel_append_parameters
|
||
|
|
||
|
. /tmp/kernel_append_parameter_variables.txt
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
# Variable-ize /proc/cmdline arguments
|
||
|
#
|
||
|
variableize_kernel_append_parameters() {
|
||
|
logmsg
|
||
|
logmsg variableize_kernel_append_parameters
|
||
|
cat /proc/cmdline | tr ' ' '\n' | grep '=' > /tmp/kernel_append_parameter_variables.txt
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
# Look for local.cfg file
|
||
|
# This code inspired by Ian McLeod <ian@valinux.com>
|
||
|
#
|
||
|
read_local_cfg() {
|
||
|
logmsg
|
||
|
logmsg read_local_cfg
|
||
|
|
||
|
if [ "x$SKIP_LOCAL_CFG" = "xy" ]; then
|
||
|
logmsg "Skipping local.cfg: option SKIP_LOCAL_CFG=y has been specified"
|
||
|
return
|
||
|
fi
|
||
|
|
||
|
# Try with local.cfg directly from initrd.
|
||
|
if [ -f /local.cfg ]; then
|
||
|
cp -f /local.cfg /tmp/local.cfg
|
||
|
fi
|
||
|
|
||
|
#
|
||
|
# BEGIN try hard drive
|
||
|
#
|
||
|
if [ ! -z "$LAST_ROOT" ]; then
|
||
|
|
||
|
logmsg
|
||
|
logmsg "Checking for /local.cfg file on hard drive..."
|
||
|
mkdir /last_root
|
||
|
logmsg "Mounting hard drive..."
|
||
|
mount $LAST_ROOT /last_root -o ro > /dev/null 2>&1
|
||
|
if [ $? != 0 ]; then
|
||
|
logmsg "FATAL: Couldn't mount hard drive!"
|
||
|
logmsg "Your kernel must have all necessary block and filesystem drivers compiled in"
|
||
|
logmsg "statically (not modules) in order to use a local.cfg on your hard drive. The"
|
||
|
logmsg "standard SystemImager kernel is modular, so you will need to compile your own"
|
||
|
logmsg "kernel. See the SystemImager documentation for details. To proceed at this"
|
||
|
logmsg "point, you will need to unset the LAST_ROOT append parameter by typing"
|
||
|
logmsg ""systemimager LAST_ROOT=", or similar, at your boot prompt. This will not use"
|
||
|
logmsg "the local.cfg file on your hard drive, but will still use one on a floppy."
|
||
|
shellout
|
||
|
fi
|
||
|
|
||
|
if [ -f /last_root/local.cfg ]; then
|
||
|
logmsg "Found /local.cfg on hard drive."
|
||
|
logmsg "Copying /local.cfg settings to /tmp/local.cfg."
|
||
|
cat /last_root/local.cfg >> /tmp/local.cfg || shellout
|
||
|
else
|
||
|
logmsg "No /local.cfg on hard drive."
|
||
|
fi
|
||
|
logmsg "Unmounting hard drive..."
|
||
|
umount /last_root || shellout
|
||
|
logmsg
|
||
|
fi
|
||
|
# END try hard drive
|
||
|
|
||
|
### BEGIN try floppy ###
|
||
|
logmsg "Checking for floppy diskette."
|
||
|
logmsg 'YOU MAY SEE SOME "wrong magic" ERRORS HERE, AND THAT IS NORMAL.'
|
||
|
mkdir -p /floppy
|
||
|
mount /dev/fd0 /floppy -o ro > /dev/null 2>&1
|
||
|
if [ $? = 0 ]; then
|
||
|
logmsg "Found floppy diskette."
|
||
|
if [ -f /floppy/local.cfg ]; then
|
||
|
logmsg "Found /local.cfg on floppy."
|
||
|
logmsg "Copying /local.cfg settings to /tmp/local.cfg."
|
||
|
logmsg "NOTE: local.cfg settings from a floppy will override settings from"
|
||
|
logmsg " a local.cfg file on your hard drive and DHCP."
|
||
|
# We use cat instead of copy, so that floppy settings can
|
||
|
# override hard disk settings. -BEF-
|
||
|
cat /floppy/local.cfg >> /tmp/local.cfg || shellout
|
||
|
else
|
||
|
logmsg "No /local.cfg on floppy diskette."
|
||
|
fi
|
||
|
else
|
||
|
logmsg "No floppy diskette in drive."
|
||
|
fi
|
||
|
### END try floppy ###
|
||
|
|
||
|
# /tmp/local.cfg may be created from a local.cfg file on the hard drive, or a
|
||
|
# floppy. If both are used, settings on the floppy take precedence. -BEF-
|
||
|
if [ -f /tmp/local.cfg ]; then
|
||
|
logmsg "Reading configuration from /tmp/local.cfg"
|
||
|
. /tmp/local.cfg || shellout
|
||
|
fi
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
# Configure network interface using local.cfg settings if possible, else
|
||
|
# use DHCP. -BEF-
|
||
|
#
|
||
|
start_network() {
|
||
|
logmsg
|
||
|
logmsg start_network
|
||
|
if [ ! -z $IPADDR ]; then
|
||
|
|
||
|
# configure interface and add default gateway
|
||
|
ifconfig $DEVICE $IPADDR netmask $NETMASK broadcast $BROADCAST
|
||
|
if [ $? != 0 ]; then
|
||
|
logmsg
|
||
|
logmsg "I couldn't configure the network interface using your pre-boot settings:"
|
||
|
logmsg " DEVICE: $DEVICE"
|
||
|
logmsg " IPADDR: $IPADDR"
|
||
|
logmsg " NETMASK: $NETMASK"
|
||
|
logmsg " BROADCAST: $BROADCAST"
|
||
|
logmsg
|
||
|
shellout
|
||
|
fi
|
||
|
|
||
|
if [ ! -z $GATEWAY ]; then
|
||
|
route add default gw $GATEWAY
|
||
|
if [ $? != 0 ]; then
|
||
|
logmsg
|
||
|
logmsg "The command \"route add default gw $GATEWAY\" failed."
|
||
|
logmsg "Check your pre-boot network settings."
|
||
|
logmsg
|
||
|
shellout
|
||
|
fi
|
||
|
fi
|
||
|
|
||
|
else
|
||
|
|
||
|
### try dhcp ###
|
||
|
logmsg "IP Address not set with pre-boot settings."
|
||
|
|
||
|
### BEGIN ether sleep ###
|
||
|
# Give the switch time to start passing packets. Some switches won't
|
||
|
# forward packets until 30 seconds or so after an interface comes up.
|
||
|
# This means the dhcp server won't even get the request for 30 seconds.
|
||
|
# Many ethernet cards aren't considered "up" by the switch until the
|
||
|
# driver is loaded. Because the driver is compiled directly into the
|
||
|
# kernel here, the driver is definitely loaded at this point.
|
||
|
#
|
||
|
# Default is 0. The recommended setting of ETHER_SLEEP=35 can be set
|
||
|
# with a local.cfg file. -BEF-
|
||
|
#
|
||
|
[ -z $ETHER_SLEEP ] && ETHER_SLEEP=0
|
||
|
logmsg
|
||
|
logmsg "sleep $ETHER_SLEEP: This is to give your switch (if you're using one) time to"
|
||
|
logmsg " recognize your ethernet card before we try the network."
|
||
|
logmsg " Tip: You can use <ctrl>+<c> to pass the time (pun intended)."
|
||
|
logmsg
|
||
|
count_loop $ETHER_SLEEP
|
||
|
logmsg
|
||
|
### END ether sleep ###
|
||
|
|
||
|
# create directory to catch dhcp information
|
||
|
DHCLIENT_DIR="/var/state/dhcp"
|
||
|
mkdir -p $DHCLIENT_DIR
|
||
|
|
||
|
# combine systemimager code to the stock debian dhclient-script
|
||
|
# and make executable
|
||
|
cat /etc/dhclient-script.si-prefix \
|
||
|
/etc/dhclient-script.debian-dist \
|
||
|
> /etc/dhclient-script
|
||
|
chmod +x /etc/dhclient-script
|
||
|
|
||
|
# be sure AF_PACKET is supported in the kernel
|
||
|
[ -f /lib/modules/`uname -r`/modules.dep ] && modprobe af_packet &> /dev/null
|
||
|
|
||
|
# get info via dhcp
|
||
|
logmsg
|
||
|
logmsg "dhclient"
|
||
|
dhclient $DEVICE
|
||
|
if [ ! -s ${DHCLIENT_DIR}/dhclient.leases ]; then
|
||
|
logmsg
|
||
|
logmsg "I couldn't configure the network interface using DHCP."
|
||
|
logmsg
|
||
|
shellout
|
||
|
fi
|
||
|
|
||
|
if [ -z ${DEVICE} ]; then
|
||
|
# Figure out which interface actually got configured.
|
||
|
# Suggested by James Oakley.
|
||
|
#
|
||
|
DEVICE=`grep interface ${DHCLIENT_DIR}/dhclient.leases | \
|
||
|
sed -e 's/^.*interface "//' -e 's/";//'`
|
||
|
fi
|
||
|
|
||
|
# read dhcp info in as variables -- this file will be created by
|
||
|
# the /etc/dhclient-start script that is run automatically by
|
||
|
# dhclient.
|
||
|
. /tmp/dhcp_info.${DEVICE} || shellout
|
||
|
### END dhcp ###
|
||
|
|
||
|
# Re-read configuration information from local.cfg to over-ride
|
||
|
# DHCP settings, if necessary. -BEF-
|
||
|
if [ -f /tmp/local.cfg ]; then
|
||
|
logmsg
|
||
|
logmsg "Overriding any DHCP settings with pre-boot local.cfg settings."
|
||
|
. /tmp/local.cfg || shellout
|
||
|
fi
|
||
|
|
||
|
logmsg
|
||
|
logmsg "Overriding any DHCP settings with pre-boot settings from kernel append"
|
||
|
logmsg "parameters."
|
||
|
read_kernel_append_parameters
|
||
|
fi
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
# Ping test
|
||
|
ping_test() {
|
||
|
logmsg
|
||
|
logmsg ping_test
|
||
|
|
||
|
# The reason we don't ping the IMAGESERVER if FLAMETHROWER_DIRECTORY_PORTBASE
|
||
|
# is set, is that the client may never be given, know, or need to know, the
|
||
|
# IP address of the imageserver because the client is receiving _all_ of it's
|
||
|
# data via multicast, which is more like listening to a channel, as compared
|
||
|
# with connecting directly to a server. -BEF-
|
||
|
#
|
||
|
if [ ! -z "$FLAMETHROWER_DIRECTORY_PORTBASE" ]; then
|
||
|
PING_DESTINATION=$GATEWAY
|
||
|
HOST_TYPE="default gateway"
|
||
|
else
|
||
|
PING_DESTINATION=$IMAGESERVER
|
||
|
HOST_TYPE="SystemImager server"
|
||
|
fi
|
||
|
logmsg
|
||
|
logmsg "Pinging your $HOST_TYPE to ensure we have network connectivity."
|
||
|
logmsg
|
||
|
|
||
|
|
||
|
# Ping test code submitted by Grant Noruschat <grant@eigen.ee.ualberta.ca>
|
||
|
# modified slightly by Brian Finley.
|
||
|
PING_COUNT=1
|
||
|
PING_EXIT_STATUS=1
|
||
|
while [ "$PING_EXIT_STATUS" != "0" ]
|
||
|
do
|
||
|
logmsg "PING ATTEMPT $PING_COUNT: "
|
||
|
ping -c 1 $PING_DESTINATION
|
||
|
PING_EXIT_STATUS=$?
|
||
|
|
||
|
if [ "$PING_EXIT_STATUS" = "0" ]; then
|
||
|
logmsg
|
||
|
logmsg " We have connectivity to your $HOST_TYPE!"
|
||
|
fi
|
||
|
|
||
|
PING_COUNT=$(( $PING_COUNT + 1 ))
|
||
|
if [ "$PING_COUNT" = "4" ]; then
|
||
|
logmsg
|
||
|
logmsg " WARNING: Failed ping test."
|
||
|
logmsg " Despite this seemingly depressing result, I will attempt"
|
||
|
logmsg " to proceed with the install. Your $HOST_TYPE may be"
|
||
|
logmsg " configured to not respond to pings, but it wouldn't hurt"
|
||
|
logmsg " to double check that your networking equipment is"
|
||
|
logmsg " working properly!"
|
||
|
logmsg
|
||
|
sleep 5
|
||
|
PING_EXIT_STATUS=0
|
||
|
fi
|
||
|
done
|
||
|
|
||
|
unset PING_DESTINATION
|
||
|
unset HOST_TYPE
|
||
|
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
start_syslogd() {
|
||
|
logmsg
|
||
|
logmsg start_syslogd
|
||
|
if [ ! -z $LOG_SERVER ]; then
|
||
|
logmsg "Starting syslogd..."
|
||
|
[ -z $LOG_SERVER_PORT ] && LOG_SERVER_PORT="514"
|
||
|
syslogd -R ${LOG_SERVER}:${LOG_SERVER_PORT}
|
||
|
# as long as we are starting syslogd, start klogd as well, in case
|
||
|
# there is a kernel issue that happens
|
||
|
klogd
|
||
|
# set USELOGGER=1 so logmsg knows to do the right thing
|
||
|
USELOGGER=1
|
||
|
logmsg "Successfully started syslogd!"
|
||
|
fi
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
show_loaded_modules() {
|
||
|
# Show loaded modules
|
||
|
logmsg ">> Loaded kernel modules:"
|
||
|
for m in `cut -d' ' -f1 /proc/modules`; do
|
||
|
logmsg "$m"
|
||
|
done
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
get_hostname_by_hosts_file() {
|
||
|
|
||
|
logmsg
|
||
|
logmsg get_hostname_by_hosts_file
|
||
|
|
||
|
#
|
||
|
# Look in $FILE for that magic joy.
|
||
|
#
|
||
|
FILE=${SCRIPTS_DIR}/hosts
|
||
|
if [ -e $FILE ]; then
|
||
|
|
||
|
logmsg "Hosts file exists..."
|
||
|
|
||
|
# add escape characters to IPADDR so that it can be used to find HOSTNAME below
|
||
|
IPADDR_ESCAPED=`echo "$IPADDR" | sed -e 's/\./\\\./g'`
|
||
|
|
||
|
# get HOSTNAME by parsing hosts file
|
||
|
logmsg "Searching for this machine's hostname in $FILE by IP: $IPADDR"
|
||
|
|
||
|
# Command summary by line:
|
||
|
# 1: convert tabs to spaces -- contains a literal tab: <ctrl>+<v> then <tab>
|
||
|
# 2: remove comments
|
||
|
# 3: add a space at the beginning of every line
|
||
|
# 4: get line with IP address (no more no less)
|
||
|
# 5: strip out ip address
|
||
|
# 6: strip out space(s) before first hostname on line
|
||
|
# 7: remove any aliases on line
|
||
|
# 8: remove domain name, leaving naught but the hostname, naked as the day it were born
|
||
|
|
||
|
HOSTNAME=`
|
||
|
sed 's/[[:space:]]/ /g' $FILE | \
|
||
|
grep -v '^ *#' | \
|
||
|
sed 's/^/ /' | \
|
||
|
grep " $IPADDR_ESCAPED " | \
|
||
|
sed 's/ [0-9]*\.[0-9]*\.[0-9]*\.[0-9]*//' | \
|
||
|
sed 's/ *//' | \
|
||
|
sed 's/ .*//' | \
|
||
|
sed 's/\..*$//g'
|
||
|
`
|
||
|
else
|
||
|
logmsg "No hosts file."
|
||
|
fi
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
get_hostname_by_dns() {
|
||
|
|
||
|
logmsg
|
||
|
logmsg get_hostname_by_dns
|
||
|
|
||
|
# Get base hostname. For example, www7.domain.com will become www7. -BEF-
|
||
|
HOSTNAME=`nslookup $IPADDR | sed -ne "s/^Address 1:[[:space:]]\+$IPADDR[[:space:]]\+\([^\.]\+\).*$/\1/p"`
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
get_base_hostname() {
|
||
|
BASE_HOSTNAME=`echo $HOSTNAME | sed "s/[.0-9].*$//"`
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
get_group_name() {
|
||
|
if [ -f ${SCRIPTS_DIR}/cluster.txt ]; then
|
||
|
[ -z "$GROUPNAMES" ] && \
|
||
|
GROUPNAMES=`unique $(grep "^${HOSTNAME}:" ${SCRIPTS_DIR}/cluster.txt | cut -d: -f2 | tr "\n" ' ')`
|
||
|
[ -z "$IMAGENAME" ] && \
|
||
|
IMAGENAME=`grep "^${HOSTNAME}:" ${SCRIPTS_DIR}/cluster.txt | cut -d: -f3 | grep -v '^[[:space:]]*$' | sed -ne '1p'`
|
||
|
if [ -z "$GROUP_OVERRIDES" ]; then
|
||
|
GROUP_OVERRIDES=`reverse $(unique $(grep "^${HOSTNAME}:" ${SCRIPTS_DIR}/cluster.txt | cut -d: -f4 | tr "\n" ' '))`
|
||
|
# Add the global override on top (least important).
|
||
|
GROUP_OVERRIDES="`sed -ne 's/^# global_override=:\([^:]*\):$/\1/p' ${SCRIPTS_DIR}/cluster.txt` $GROUP_OVERRIDES"
|
||
|
fi
|
||
|
fi
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
choose_autoinstall_script() {
|
||
|
|
||
|
logmsg
|
||
|
logmsg choose_autoinstall_script
|
||
|
|
||
|
#
|
||
|
# Get the base hostname for the last attempt at choosing an autoinstall
|
||
|
# script. For example, if the hostname is compute99, then try to get
|
||
|
# compute.master. -BEF-
|
||
|
#
|
||
|
get_base_hostname
|
||
|
|
||
|
# Get group name (defined in /etc/systemimager/cluster.xml on the image
|
||
|
# server). -AR-
|
||
|
get_group_name
|
||
|
|
||
|
#
|
||
|
# If SCRIPTNAME is specified as a kernel append, or via local.cfg, then use that script.
|
||
|
#
|
||
|
if [ ! -z $SCRIPTNAME ]; then
|
||
|
#
|
||
|
# SCRIPTNAME was specified, but let's be flexible. Try explicit first, then .master, .sh. -BEF-
|
||
|
#
|
||
|
SCRIPTNAMES="${SCRIPTS_DIR}/${SCRIPTNAME} ${SCRIPTS_DIR}/${SCRIPTNAME}.sh ${SCRIPTS_DIR}/${SCRIPTNAME}.master"
|
||
|
|
||
|
else
|
||
|
#
|
||
|
# If SCRIPTNAME was not specified, choose one, in order of preference. First hit wins.
|
||
|
# Order of preference is:
|
||
|
# HOSTNAME (i.e. node001.sh)
|
||
|
# GROUPNAMES (i.e. Login.sh) - see /etc/systemimager/cluster.xml on the image server
|
||
|
# BASE_HOSTNAME (i.e. node.sh)
|
||
|
# IMAGENAME (i.e. ubuntu7_04.sh)
|
||
|
#
|
||
|
[ ! -z $HOSTNAME ] && \
|
||
|
SCRIPTNAMES="${SCRIPTNAMES} ${SCRIPTS_DIR}/${HOSTNAME}.sh ${SCRIPTS_DIR}/${HOSTNAME}.master"
|
||
|
for GROUPNAME in $GROUPNAMES; do
|
||
|
SCRIPTNAMES="${SCRIPTNAMES} ${SCRIPTS_DIR}/${GROUPNAME}.sh ${SCRIPTS_DIR}/${GROUPNAME}.master"
|
||
|
done
|
||
|
unset GROUPNAME
|
||
|
[ ! -z $BASE_HOSTNAME ] && \
|
||
|
SCRIPTNAMES="${SCRIPTNAMES} ${SCRIPTS_DIR}/${BASE_HOSTNAME}.sh ${SCRIPTS_DIR}/${BASE_HOSTNAME}.master"
|
||
|
[ ! -z $IMAGENAME ] && \
|
||
|
SCRIPTNAMES="${SCRIPTNAMES} ${SCRIPTS_DIR}/${IMAGENAME}.sh ${SCRIPTS_DIR}/${IMAGENAME}.master"
|
||
|
fi
|
||
|
|
||
|
#
|
||
|
# Choose a winner!
|
||
|
#
|
||
|
for SCRIPTNAME in $SCRIPTNAMES
|
||
|
do
|
||
|
[ -e $SCRIPTNAME ] && break
|
||
|
done
|
||
|
|
||
|
# Did we really find one, or just exit the loop without a 'break'
|
||
|
if [ ! -e $SCRIPTNAME ]; then
|
||
|
logmsg "FATAL: couldn't find any of the following autoinstall scripts!"
|
||
|
logmsg "---"
|
||
|
logmsg ${SCRIPTNAMES}
|
||
|
logmsg "---"
|
||
|
logmsg "Be sure that at least one of the scripts above exists in"
|
||
|
logmsg "the autoinstall scripts directory on your image server."
|
||
|
logmsg
|
||
|
logmsg "See also: si_mkautoinstallscript(8)."
|
||
|
shellout
|
||
|
fi
|
||
|
logmsg "Using autoinstall script: ${SCRIPTNAME}"
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
run_autoinstall_script() {
|
||
|
|
||
|
logmsg
|
||
|
logmsg run_autoinstall_script
|
||
|
|
||
|
# Run the autoinstall script.
|
||
|
chmod 755 $SCRIPTNAME || shellout
|
||
|
logmsg ">>> $SCRIPTNAME"
|
||
|
$SCRIPTNAME || shellout
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
# Description: remove duplicated elements from a list, preserving the order.
|
||
|
#
|
||
|
unique() {
|
||
|
ret=
|
||
|
for i in $*; do
|
||
|
flag=0
|
||
|
for j in $ret; do
|
||
|
[ "$i" = "$j" ] && flag=1 && break
|
||
|
done
|
||
|
[ $flag -eq 0 ] && ret="$ret $i"
|
||
|
done
|
||
|
echo $ret
|
||
|
unset i j flag ret
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
# Description: reverse a list
|
||
|
#
|
||
|
reverse() {
|
||
|
ret=
|
||
|
for i in $*; do
|
||
|
ret="$i $ret"
|
||
|
done
|
||
|
echo $ret
|
||
|
unset i
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
run_pre_install_scripts() {
|
||
|
|
||
|
logmsg
|
||
|
logmsg run_pre_install_scripts
|
||
|
|
||
|
get_base_hostname
|
||
|
|
||
|
# Get group name (defined in /etc/systemimager/cluster.xml on the image
|
||
|
# server). -AR-
|
||
|
get_group_name
|
||
|
|
||
|
if [ -e "${SCRIPTS_DIR}/pre-install/" ]; then
|
||
|
|
||
|
cd ${SCRIPTS_DIR}/pre-install/
|
||
|
|
||
|
PRE_INSTALL_SCRIPTS="$PRE_INSTALL_SCRIPTS `ls | grep "^[0-9][0-9]all\..*"`"
|
||
|
PRE_INSTALL_SCRIPTS="$PRE_INSTALL_SCRIPTS `ls | grep "^[0-9][0-9]${IMAGENAME}\..*"`"
|
||
|
PRE_INSTALL_SCRIPTS="$PRE_INSTALL_SCRIPTS `ls | grep "^[0-9][0-9]${BASE_HOSTNAME}\..*"`"
|
||
|
for GROUPNAME in ${GROUPNAMES}; do
|
||
|
PRE_INSTALL_SCRIPTS="$PRE_INSTALL_SCRIPTS `ls | grep "^[0-9][0-9]${GROUPNAME}\..*"`"
|
||
|
done
|
||
|
unset GROUPNAME
|
||
|
PRE_INSTALL_SCRIPTS="$PRE_INSTALL_SCRIPTS `ls | grep "^[0-9][0-9]${HOSTNAME}\..*"`"
|
||
|
|
||
|
# Now, to get rid of those pesky newlines. -BEF-
|
||
|
PRE_INSTALL_SCRIPTS=`echo $PRE_INSTALL_SCRIPTS | tr '\n' ' '`
|
||
|
|
||
|
if [ ! -z "`echo ${PRE_INSTALL_SCRIPTS}|sed 's/ //'`" ]; then
|
||
|
|
||
|
for PRE_INSTALL_SCRIPT in `unique $PRE_INSTALL_SCRIPTS`
|
||
|
do
|
||
|
logmsg ">>> $PRE_INSTALL_SCRIPT"
|
||
|
chmod +x $PRE_INSTALL_SCRIPT || shellout
|
||
|
./$PRE_INSTALL_SCRIPT || shellout
|
||
|
done
|
||
|
else
|
||
|
logmsg "No pre-install scripts found."
|
||
|
fi
|
||
|
|
||
|
if [ -e "/tmp/pre-install_variables.txt" ]; then
|
||
|
. /tmp/pre-install_variables.txt
|
||
|
fi
|
||
|
|
||
|
fi
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
run_post_install_scripts() {
|
||
|
|
||
|
logmsg
|
||
|
logmsg run_post_install_scripts
|
||
|
|
||
|
get_base_hostname
|
||
|
|
||
|
# Get group name (defined in /etc/systemimager/cluster.xml on the image
|
||
|
# server). -AR-
|
||
|
get_group_name
|
||
|
|
||
|
if [ -e "${SCRIPTS_DIR}/post-install/" ]; then
|
||
|
|
||
|
# make a copy of variables.txt available to post-install scripts -BEF-
|
||
|
cp -f /tmp/variables.txt ${SCRIPTS_DIR}/post-install/
|
||
|
|
||
|
cd ${SCRIPTS_DIR}/post-install/
|
||
|
|
||
|
POST_INSTALL_SCRIPTS="$POST_INSTALL_SCRIPTS `ls | grep "^[0-9][0-9]all\..*" | grep -v "~$" `"
|
||
|
POST_INSTALL_SCRIPTS="$POST_INSTALL_SCRIPTS `ls | grep "^[0-9][0-9]${IMAGENAME}\..*" | grep -v "~$" `"
|
||
|
POST_INSTALL_SCRIPTS="$POST_INSTALL_SCRIPTS `ls | grep "^[0-9][0-9]${BASE_HOSTNAME}\..*" | grep -v "~$" `"
|
||
|
for GROUPNAME in ${GROUPNAMES}; do
|
||
|
POST_INSTALL_SCRIPTS="$POST_INSTALL_SCRIPTS `ls | grep "^[0-9][0-9]${GROUPNAME}\..*" | grep -v "~$" `"
|
||
|
done
|
||
|
POST_INSTALL_SCRIPTS="$POST_INSTALL_SCRIPTS `ls | grep "^[0-9][0-9]${HOSTNAME}\..*" | grep -v "~$" `"
|
||
|
|
||
|
# Now, to get rid of those pesky newlines. -BEF-
|
||
|
POST_INSTALL_SCRIPTS=`echo $POST_INSTALL_SCRIPTS | tr '\n' ' '`
|
||
|
|
||
|
if [ ! -z "`echo ${POST_INSTALL_SCRIPTS}|sed 's/ //'`" ]; then
|
||
|
|
||
|
mkdir -p /a/tmp/post-install/ || shellout
|
||
|
|
||
|
rsync -a ${SCRIPTS_DIR}/post-install/ /a/tmp/post-install/ || shellout
|
||
|
|
||
|
for POST_INSTALL_SCRIPT in `unique $POST_INSTALL_SCRIPTS`
|
||
|
do
|
||
|
if [ -e "$POST_INSTALL_SCRIPT" ]; then
|
||
|
logmsg ">>> $POST_INSTALL_SCRIPT"
|
||
|
chmod +x /a/tmp/post-install/$POST_INSTALL_SCRIPT || shellout
|
||
|
chroot /a/ /tmp/post-install/$POST_INSTALL_SCRIPT || shellout
|
||
|
fi
|
||
|
done
|
||
|
else
|
||
|
logmsg "No post-install scripts found."
|
||
|
fi
|
||
|
|
||
|
# Clean up post-install script directory.
|
||
|
rm -rf /a/tmp/post-install/ || shellout
|
||
|
fi
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
# Stuff for SSH installs
|
||
|
#
|
||
|
|
||
|
|
||
|
start_sshd() {
|
||
|
mkdir -p /root/.ssh/ || shellout
|
||
|
|
||
|
# download ssh authorized_keys if it's not present into the initrd.
|
||
|
if [ ! -f /root/.ssh/authorized_keys ]; then
|
||
|
if [ -z $SSH_DOWNLOAD_URL ]; then
|
||
|
logmsg
|
||
|
logmsg "error: authorized_keys not found and SSH_DOWNLOAD_URL not defined in the installation parameters!"
|
||
|
logmsg "sshd can't be started!"
|
||
|
shellout
|
||
|
fi
|
||
|
CMD="wget ${SSH_DOWNLOAD_URL}/${ARCH}/ssh/authorized_keys"
|
||
|
logmsg
|
||
|
logmsg $CMD
|
||
|
$CMD || shellout
|
||
|
fi
|
||
|
|
||
|
# set permissions to 600 -- otherwise, sshd will refuse to use it
|
||
|
chmod 600 /root/.ssh/authorized_keys || shellout
|
||
|
|
||
|
# must be owned by root
|
||
|
chown -R 0.0 /root/
|
||
|
|
||
|
# create a private host key for this autoinstall client
|
||
|
logmsg
|
||
|
logmsg "Using ssh-keygen to create this hosts private key"
|
||
|
logmsg
|
||
|
mkdir -p /var/empty || shellout
|
||
|
if [ ! -f /etc/ssh/ssh_host_dsa_key ]; then
|
||
|
ssh-keygen -t dsa -N "" -f /etc/ssh/ssh_host_dsa_key || shellout
|
||
|
fi
|
||
|
if [ ! -f /etc/ssh/ssh_host_rsa_key ]; then
|
||
|
ssh-keygen -t rsa -N "" -f /etc/ssh/ssh_host_rsa_key || shellout
|
||
|
fi
|
||
|
|
||
|
# try to mount devpts (sometimes it's not really necessary)
|
||
|
mkdir -p /dev/pts
|
||
|
mount -t devpts none /dev/pts >/dev/null 2>&1
|
||
|
|
||
|
# fire up sshd
|
||
|
mkdir -p /var/run/sshd || shellout
|
||
|
chmod 0755 /var/run/sshd || shellout
|
||
|
/usr/sbin/sshd || shellout
|
||
|
logmsg "sshd started"
|
||
|
touch /tmp/sshd_started
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
start_ssh() {
|
||
|
|
||
|
# create root's ssh dir
|
||
|
mkdir -p /root/.ssh
|
||
|
|
||
|
############################################################################
|
||
|
#
|
||
|
# If a private key exists, put it in the right place so this autoinstall
|
||
|
# client can use it to authenticate itself to the imageserver.
|
||
|
#
|
||
|
if [ -e /root/.ssh/id_dsa ]; then
|
||
|
# (ssh2 dsa style user private key)
|
||
|
PRIVATE_KEY=/root/.ssh/id_dsa
|
||
|
chmod 600 $PRIVATE_KEY || shellout
|
||
|
elif [ -e /root/.ssh/id_rsa ]; then
|
||
|
# (ssh2 rsa style user private key)
|
||
|
PRIVATE_KEY=/root/.ssh/id_rsa
|
||
|
chmod 600 $PRIVATE_KEY || shellout
|
||
|
elif [ -e /floppy/id_dsa ]; then
|
||
|
# (ssh2 dsa style user private key) from floppy
|
||
|
PRIVATE_KEY=/root/.ssh/id_dsa
|
||
|
cp /floppy/id_dsa $PRIVATE_KEY || shellout
|
||
|
chmod 600 $PRIVATE_KEY || shellout
|
||
|
elif [ -e /floppy/id_rsa ]; then
|
||
|
#
|
||
|
# (ssh2 rsa style user private key) from floppy
|
||
|
PRIVATE_KEY=/root/.ssh/id_rsa
|
||
|
cp /floppy/id_rsa $PRIVATE_KEY || shellout
|
||
|
chmod 600 $PRIVATE_KEY || shellout
|
||
|
fi
|
||
|
#
|
||
|
############################################################################
|
||
|
|
||
|
# If we have a private key from the media above, go ahead and open secure tunnel
|
||
|
# to the imageserver and continue with the autoinstall like normal.
|
||
|
if [ ! -z $PRIVATE_KEY ]; then
|
||
|
|
||
|
# With the prep ready, start the ssh tunnel connection.
|
||
|
#
|
||
|
# Determine if we should run interactive and set redirection options appropriately.
|
||
|
# So if the key is blank, go interactive. (Suggested by Don Stocks <don_stocks@leaseloan.com>)
|
||
|
if [ -s $PRIVATE_KEY ]; then
|
||
|
# key is *not* blank
|
||
|
REDIRECTION_OPTIONS="> /dev/null 2>&1"
|
||
|
else
|
||
|
# key is blank - go interactive
|
||
|
REDIRECTION_OPTIONS=""
|
||
|
fi
|
||
|
|
||
|
# Default ssh user is root.
|
||
|
[ -z $SSH_USER ] && SSH_USER=root
|
||
|
|
||
|
CMD="ssh -N -l $SSH_USER -n -f -L873:127.0.0.1:873 $IMAGESERVER $REDIRECTION_OPTIONS"
|
||
|
logmsg $CMD
|
||
|
$CMD || shellout
|
||
|
|
||
|
# Since we're using SSH, change the $IMAGESERVER variable to reflect
|
||
|
# the forwarded connection.
|
||
|
IMAGESERVER=127.0.0.1
|
||
|
|
||
|
else
|
||
|
########################################################################
|
||
|
#
|
||
|
# Looks like we didn't get a private key so let's just fire up
|
||
|
# sshd and wait for someone to connect to us to initiate the
|
||
|
# next step of the autoinstall.
|
||
|
#
|
||
|
if [ -z $HOSTNAME ]; then
|
||
|
logmsg
|
||
|
logmsg "Trying to get hostname via DNS..."
|
||
|
logmsg
|
||
|
get_hostname_by_dns
|
||
|
fi
|
||
|
|
||
|
if [ -z $HOSTNAME ]; then
|
||
|
HOST_OR_IP=$IPADDR
|
||
|
else
|
||
|
HOST_OR_IP=$HOSTNAME
|
||
|
fi
|
||
|
|
||
|
if [ ! -f /tmp/sshd_started ]; then
|
||
|
start_sshd
|
||
|
# Give sshd time to initialize before we yank the parent process
|
||
|
# rug out from underneath it.
|
||
|
sleep 15
|
||
|
fi
|
||
|
|
||
|
logmsg
|
||
|
logmsg
|
||
|
logmsg "Started sshd. You must now go to your imageserver and issue"
|
||
|
logmsg "the following command:"
|
||
|
logmsg
|
||
|
logmsg " \"si_pushinstall --hosts ${HOST_OR_IP}\"."
|
||
|
logmsg
|
||
|
logmsg
|
||
|
|
||
|
# Since we're using SSH, change the $IMAGESERVER variable to reflect
|
||
|
# the forwarded connection.
|
||
|
IMAGESERVER=127.0.0.1
|
||
|
|
||
|
while [ ! -f /tmp/si_pushupdate.completed ]; do
|
||
|
sleep 5
|
||
|
done
|
||
|
fi
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
# send_monitor_msg
|
||
|
#
|
||
|
# Description:
|
||
|
# Redirect a message to the monitor server.
|
||
|
#
|
||
|
# Usage: send_monitor_msg "var=$msg"
|
||
|
#
|
||
|
|
||
|
send_monitor_msg() {
|
||
|
if [ -z $MONITOR_SERVER ]; then
|
||
|
return
|
||
|
fi
|
||
|
if [ -z $MONITOR_PORT ]; then
|
||
|
MONITOR_PORT=8181
|
||
|
fi
|
||
|
|
||
|
# Message to send.
|
||
|
msg=`echo "$@"`
|
||
|
|
||
|
# Get the client mac address.
|
||
|
if [ -z "$mac" ]; then
|
||
|
mac=`ifconfig $DEVICE 2>/dev/null | sed -ne "s/.*HWaddr //p" | sed "s/ //g" | sed s/:/./g`
|
||
|
fi
|
||
|
|
||
|
# Collect some special info only after proc file system is mounted.
|
||
|
if [ `mount 2>/dev/null | grep proc > /dev/null; echo $?` -eq 0 ]; then
|
||
|
# Collect the CPU info.
|
||
|
if [ -z "$cpu" ]; then
|
||
|
cpu=$(echo `cat /proc/cpuinfo | grep "cpu\|clock\|model name\|cpu MHz" | grep -v "cpu family" | sed -ne '1,2p' | sed "s/.*: //" | sed "s/^\([0-9\.]*\)MHz$/(\1 MHz)/" | sed "s/^\([0-9\.]*\)$/(\1 MHz)/"` | sed "s/\(MHz)\)/\1 |/g" | sed "s/ |$//")
|
||
|
fi
|
||
|
|
||
|
# Collect the number of CPUs.
|
||
|
if [ -z "$ncpus" ]; then
|
||
|
ncpus=$((`cat /proc/cpuinfo | grep "^processor" | sed -n '$p' | sed "s/.*: \([0-9]\)*$/\1/"` + 1))
|
||
|
fi
|
||
|
|
||
|
# Collect the kernel information.
|
||
|
if [ -z "$kernel_name" ]; then
|
||
|
kernel_name=`uname -r`
|
||
|
fi
|
||
|
|
||
|
# Collect the amount of phyisical memory.
|
||
|
if [ -z "$mem" ]; then
|
||
|
mem=`cat /proc/meminfo | sed -ne "s/MemTotal: *//p" | sed "s/ kB//"`
|
||
|
fi
|
||
|
|
||
|
# Evaluate the amount of available RAM.
|
||
|
tmpfs=`df | grep tmpfs | grep "/$" | sed "s/.* \([0-9]*%\) .*/\1/"`
|
||
|
|
||
|
# Evaluate the uptime of the client.
|
||
|
time=`cat /proc/uptime | sed "s/\..*//"`
|
||
|
fi
|
||
|
|
||
|
# Report the message to the monitor server.
|
||
|
send_msg=`echo "mac=$mac:ip=$IPADDR:host=$HOSTNAME:cpu=$cpu:ncpus=$ncpus:kernel=$kernel_name:mem=$mem:os=$IMAGENAME:tmpfs=$tmpfs:time=$time:$msg"`
|
||
|
|
||
|
# Send data to monitor server.
|
||
|
echo "$send_msg" | nc $MONITOR_SERVER $MONITOR_PORT
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
# send_monitor_stdout
|
||
|
#
|
||
|
# Description:
|
||
|
# Redirect a stdout of a command to the monitor console.
|
||
|
#
|
||
|
# Usage: <cmd> | send_monitor_stdout
|
||
|
#
|
||
|
|
||
|
send_monitor_stdout() {
|
||
|
while read l; do
|
||
|
# echo to the console -AR-
|
||
|
echo "$l"
|
||
|
# Send the message to the monitor daemon.
|
||
|
if [ "x$MONITOR_CONSOLE" = "xy" ]; then
|
||
|
MONITOR_CONSOLE=yes
|
||
|
fi
|
||
|
if [ "x$MONITOR_CONSOLE" = "xyes" ]; then
|
||
|
# Log message into the global monitor log.
|
||
|
echo "$l" >> /tmp/si_monitor.log
|
||
|
fi
|
||
|
done
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
# Initialize the monitor server
|
||
|
#
|
||
|
|
||
|
init_monitor_server() {
|
||
|
# Send initialization status.
|
||
|
send_monitor_msg "status=0:first_timestamp=on:speed=0"
|
||
|
logmsg "Monitoring initialized."
|
||
|
# Start client log gathering server: for each connection
|
||
|
# to the local client on port 8181 the full log is sent
|
||
|
# to the requestor. -AR-
|
||
|
if [ "x$MONITOR_CONSOLE" = "xy" ]; then
|
||
|
MONITOR_CONSOLE=yes
|
||
|
fi
|
||
|
if [ "x$MONITOR_CONSOLE" = "xyes" ]; then
|
||
|
while :; do nc -p 8181 -l < /tmp/si_monitor.log; done &
|
||
|
fi
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
# Report installation status to the monitor server
|
||
|
#
|
||
|
|
||
|
start_report_task() {
|
||
|
# Reporting interval (in sec).
|
||
|
REPORT_INTERVAL=10
|
||
|
|
||
|
# Evaluate image size.
|
||
|
logmsg "Evaluating image size..."
|
||
|
if [ ! "x$BITTORRENT" = "xy" ]; then
|
||
|
IMAGESIZE=`rsync -av --numeric-ids $IMAGESERVER::$IMAGENAME | grep "total size" | sed -e "s/total size is \([0-9]*\).*/\1/"`
|
||
|
else
|
||
|
if [ -f "${TORRENTS_DIR}/image-${IMAGENAME}.tar.torrent" ]; then
|
||
|
torrent_file="${TORRENTS_DIR}/image-${IMAGENAME}.tar.torrent"
|
||
|
elif [ -f "${TORRENTS_DIR}/image-${IMAGENAME}.tar.gz.torrent" ]; then
|
||
|
torrent_file="${TORRENTS_DIR}/image-${IMAGENAME}.tar.gz.torrent"
|
||
|
else
|
||
|
logmsg "error: cannot find a valid torrent file for image ${IMAGENAME}"
|
||
|
shellout
|
||
|
fi
|
||
|
IMAGESIZE=`/bin/torrentinfo-console $torrent_file | sed -ne "s/file size\.*: \([0-9]*\) .*$/\1/p"`
|
||
|
fi
|
||
|
IMAGESIZE=`expr $IMAGESIZE / 1024`
|
||
|
logmsg " --> Image size = `expr $IMAGESIZE / 1024`MiB"
|
||
|
|
||
|
# Evaluate disks size.
|
||
|
LIST=`df 2>/dev/null | grep "/" | sed "s/ */ /g" | cut -d' ' -f3 | sed -ne 's/^\([0-9]*\)$/\1+/p'`0
|
||
|
DISKSIZE=`echo $LIST | bc`
|
||
|
|
||
|
# Spawn the report task -AR-
|
||
|
{
|
||
|
TOT=0; CURR_SIZE=0
|
||
|
while :; do
|
||
|
LIST=`df 2>/dev/null | grep "/" | sed "s/ */ /g" | cut -d' ' -f3 | sed -ne 's/^\([0-9]*\)$/\1+/p'`0
|
||
|
TOT=`echo $LIST | bc`
|
||
|
|
||
|
# Evaluate bandwidth.
|
||
|
speed=`echo "scale=2; (($TOT - $DISKSIZE) - $CURR_SIZE) / $REPORT_INTERVAL" | bc`
|
||
|
speed=`echo "scale=2; if ($speed >= 0) { print $speed; } else { print 0; }" | bc`
|
||
|
|
||
|
# Evaluate status.
|
||
|
CURR_SIZE=$(($TOT - $DISKSIZE))
|
||
|
status=`echo "scale=2; $CURR_SIZE * 100 / $IMAGESIZE" | bc`
|
||
|
if [ `echo "scale=2; $status <= 0" | bc` -eq 1 ]; then
|
||
|
status=1
|
||
|
elif [ `echo "scale=2; $status >= 100" | bc` -eq 1 ]; then
|
||
|
status=99
|
||
|
fi
|
||
|
|
||
|
# Send status and bandwidth to the monitor server.
|
||
|
send_monitor_msg "status=$status:speed=$speed"
|
||
|
|
||
|
# Wait $REPORT_INTERVAL sec between each report -AR-
|
||
|
sleep $REPORT_INTERVAL
|
||
|
done
|
||
|
}&
|
||
|
|
||
|
logmsg "Report task started."
|
||
|
REPORT_PID=$!
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
# Stop to report installation status to the monitor server
|
||
|
#
|
||
|
|
||
|
stop_report_task() {
|
||
|
if [ ! -z $REPORT_PID ]; then
|
||
|
kill -9 $REPORT_PID
|
||
|
logmsg "Report task stopped."
|
||
|
fi
|
||
|
|
||
|
# Try to report the error to the monitor server.
|
||
|
send_monitor_msg "status=$1:speed=0"
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
# Beep incessantly
|
||
|
#
|
||
|
beep_incessantly() {
|
||
|
local SECONDS=1
|
||
|
local MINUTES
|
||
|
local MINUTES_X_SIXTY
|
||
|
{ while :;
|
||
|
do
|
||
|
echo -n -e "\\a"
|
||
|
if [ $SECONDS -lt 60 ]; then
|
||
|
logmsg "I have been done for $SECONDS seconds. Reboot me already!"
|
||
|
else
|
||
|
MINUTES=`echo "$SECONDS / 60"|bc`
|
||
|
MINUTES_X_SIXTY=`echo "$MINUTES * 60"|bc`
|
||
|
if [ "$MINUTES_X_SIXTY" = "$SECONDS" ]; then
|
||
|
logmsg "I have been done for $MINUTES minutes now. Reboot me already!"
|
||
|
fi
|
||
|
fi
|
||
|
sleep 1
|
||
|
SECONDS=`echo "$SECONDS + 1"|bc`
|
||
|
done
|
||
|
}
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
# Beep incessantly
|
||
|
#
|
||
|
# Usage: beep [$COUNT [$INTERVAL]]
|
||
|
# Usage: beep
|
||
|
beep() {
|
||
|
local COUNT=$1
|
||
|
local INTERVAL=$2
|
||
|
|
||
|
[ -z $COUNT ] && COUNT=1
|
||
|
[ -z $INTERVAL ] && INTERVAL=1
|
||
|
|
||
|
local COUNTED=0
|
||
|
until [ "$COUNTED" = "$COUNT" ]
|
||
|
do
|
||
|
echo -n -e "\\a"
|
||
|
sleep $INTERVAL
|
||
|
COUNTED=$(( $COUNTED + 1 ))
|
||
|
done
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|
||
|
#
|
||
|
# Print out dots while sleeping
|
||
|
#
|
||
|
# Usage: sleep_loop [[$COUNT [$INTERVAL]] $CHARACTER]
|
||
|
# Usage: sleep_loop
|
||
|
sleep_loop() {
|
||
|
local COUNT=$1
|
||
|
local INTERVAL=$2
|
||
|
local CHARACTER=$3
|
||
|
local COUNTED
|
||
|
|
||
|
[ -z $COUNT ] && COUNT=1
|
||
|
[ -z $INTERVAL ] && INTERVAL=1
|
||
|
[ -z $CHARACTER ] && CHARACTER=.
|
||
|
|
||
|
COUNTED=0
|
||
|
until [ "$COUNTED" = "$COUNT" ]
|
||
|
do
|
||
|
echo -n "$CHARACTER"
|
||
|
sleep $INTERVAL
|
||
|
COUNTED=$(( $COUNTED + 1 ))
|
||
|
done
|
||
|
}
|
||
|
#
|
||
|
################################################################################
|