#!/bin/sh # # Copyright (C) 2002, Mark Lawrence . # Licence: GPL # # Copy/Sync a virtual host from one machine to another version="0.2.0" umask 022 me=${0##*/} ### Helper functions ### # Save stdin and stdout for later use exec 3>&1 exec 4>&2 noninteractive () { exec > /dev/null } interactive () { exec 1>&3 exec 2>&4 } info () { ! $quiet && echo "I: $1" >&3 } warn () { echo "W: $1" >&4 } error () { echo "E: $2" >&4 exit $1 } ### Usage/Info functions ### usage () { cat <&2 Usage: $me [-hVvqidrRs] vserver [host:]newname EOF } full_usage () { usage cat < 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. EOF } ### Default values and Command line options ### stopstart=(false) verbose=(false) quiet=(false) ssh="ssh" rsflag="-e" rsh=(false) colon=":" domain="" ip="" vsroot="/vservers" if [ $# -eq 0 ]; then # Script invoked with no command-line args? usage exit 1 fi temp=$(getopt -o hVvqd:i:rRs --long help,version,verbose,quiet,domain:,ip:,vsroot,rsh,stopstart, -n $me -- "$@") if [ $? != 0 ]; then echo " (See -h for help)" exit 1 fi # Note the quotes around `$temp': they are essential! eval set -- "$temp" while true; do case "$1" in -h|--help) full_usage exit 1 ;; -V|--version) full_version exit 1 ;; -v|--verbose) verbose=(true) shift ;; -q|--quiet) quiet=(true) shift ;; -d|--domain) domain="$2" shift 2 ;; -i|--ip) ip="$2" shift 2 ;; -r|--vsroot) vsroot="$2" shift 2 ;; -R|--rsh) rsh=(true) shift ;; -s|--stopstart) stopstart=(true) shift ;; --) shift break ;; *) echo "Internal error!" exit 1 ;; esac done if (($# != 2)); then usage exit 1 fi ### ### # By default we are reasonably quiet (ouput only via info, warn & error) if $verbose; then interactive else noninteractive fi vserver=$1 vconf=/etc/vservers/$vserver.conf vroot=$vsroot/$vserver if $rsh; then ssh="rsh" fi if (echo $2 | grep -q '^[a-z][a-z0-9]\+$'); then dhost="" newname=$2 ssh="" rsflag="" colon="" if $rsh; then warn "rsh is set but not used for a local copy" fi elif (echo $2 | grep -q '^[a-z][a-z0-9]\+:$'); then dhost=${2/:/} newname=$vserver elif (echo $2 | grep -q '^[a-z][a-z0-9]\+:[a-z][a-z0-9]\+$'); then dhost=${2/:*/} newname=${2/*:/} else error 1 "Second argument must be of the form \"[host:]name\" or \"host:\"" fi target=$vsroot/$newname targetconf=/etc/vservers/$newname.conf ### Perform some sanity checks ### if [ ! -d $vroot ]; then error 1 "Directory \"$vroot\" does not exist" fi if [ ! -e $vconf ]; then error 1 "Vserver file \"$vconf\" does not exist" fi if [ "$dhost" == "" ] && [ "$vserver" == "$newname" ]; then error 1 "Source and destination names cannot be the same on the localhost" fi if [ "$dhost" != "" ] && ! (host $dhost | grep -q 'has address'); then error 1 "$dhost does not resolve into an IP address" fi #if [ "$vserver" != "$newname" -a \( "$ip" == "" -o "$domain" == "" \) ]; then if [ \( "$ip" != "" -a "$domain" == "" \) -o \ \( "$ip" == "" -a "$domain" != "" \) ] then error 1 "Both IP address and domain must be specified together" fi if [ "$ip" != "" ] && \ ! (echo $ip | grep -q '^[0-9]\{1,3\}\(\.[0-9]\{1,3\}\)\{3\}$' ); then error 1 "\"$ip\" is not a valid IP address" fi # This works both locally and remote if ($ssh $dhost /usr/sbin/vserver $newname running | grep -q 'is running'); then warn "destination vserver \"$newname\" is running" error 1 "Cannot copy over a running vserver" fi ### Do the copy ### if $stopstart; then info "Stopping virtual server \"$vserver\" on localhost" /usr/sbin/vserver $vserver stop fi info "Syncing directories" # trailing slashes very important in the rsync! if ! rsync -avxz $rsflag $ssh $vroot/ $dhost$colon$target/; then error $? "vserver copy failed" fi if [ "$ip" != "" -a "$domain" != "" ]; then # Insert the new IPROOT/S_HOSTNAME values into the config file info "Modifying $targetconf" tmpf=$(tempfile) if (sed -e "s/^S_HOSTNAME=.*/S_HOSTNAME=\"$newname\"/" \ -e "s/^IPROOT=.*/IPROOT=\"$ip\"/" $vconf > $tmpf) then if ! rsync -v $rsflag $ssh $tmpf $dhost$colon$targetconf; then error $? "vserver config file copy/change failed" fi else warn "Unable to reconfigure virtual server config file" fi # create a new /etc/hostname info "Creating hostname file" echo $newname > $tmpf if ! rsync -v $rsflag $ssh $tmpf $dhost$colon$target/etc/hostname; then error $? "vserver /etc/hostname copy failed" fi info "Creating /etc/hosts" cat << EOF > $tmpf # /etc/hosts (automatically generated by $me) 127.0.0.1 localhost $ip $newname.$domain $newname # The following lines are desirable for IPv6 capable hosts ::1 ip6-localhost ip6-loopback fe00::0 ip6-localnet ff00::0 ip6-mcastprefix ff02::1 ip6-allnodes ff02::2 ip6-allrouters ff02::3 ip6-allhosts EOF # copy /etc/hosts if ! rsync -v $rsflag $ssh $tmpf $dhost$colon$target/etc/hosts; then error $? "vserver /etc/hosts copy failed" fi rm -f $tmpf else # copy newname.conf unchanged info "Copying $targetconf" if ! rsync -v $rsflag $ssh $vconf $dhost$colon$targetconf; then error $? "vserver config file copy/change failed" fi fi if $stopstart; then info "Starting virtual server \"$vserver\" on $dhost" $ssh $dhost /usr/sbin/vserver $vserver start if ($ssh $dhost /usr/sbin/vserver $vserver running | \ grep -q 'not running'); then error 1 "Virtual server \"$vserver\" failed to start on $dhost" fi fi exit 0