mirror of
https://git.FreeBSD.org/src.git
synced 2025-01-26 16:18:31 +00:00
b3dcd94961
don't clobber /etc/resolv.conf Add $FreeBSD. Submitted by: an j. peterson" <rbw@myplace.org> Verified fix in: dhcp-3.0.1rc6 PR: misc/34455
515 lines
11 KiB
Bash
515 lines
11 KiB
Bash
#!/bin/sh
|
|
|
|
#############################################################################
|
|
#
|
|
# Copyright (c) 1999, MindStep Corporation
|
|
# All rights reserved.
|
|
#
|
|
# Redistribution and use in source and binary forms, with or without
|
|
# modification, are permitted provided that the following conditions
|
|
# are met:
|
|
# 1. Redistributions of source code must retain the above copyright
|
|
# notice, this list of conditions and the following disclaimer.
|
|
# 2. Redistributions in binary form must reproduce the above copyright
|
|
# notice, this list of conditions and the following disclaimer in the
|
|
# documentation and/or other materials provided with the distribution.
|
|
#
|
|
# THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
|
|
# ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
# ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
|
|
# FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
# DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
# OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
# HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
# OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
# SUCH DAMAGE.
|
|
#
|
|
#
|
|
#############################################################################
|
|
#
|
|
# This script was written by Patrick Bihan-Faou, patrick@mindstep.com,
|
|
# Please contact us for bug reports, etc.
|
|
#
|
|
#############################################################################
|
|
# $MindStep_Id: dhclient-script.sh,v 1.8 1999/12/07 22:11:08 patrick Exp $
|
|
# $MindStep_Tag: CONTRIB_19991207 $
|
|
# $FreeBSD$
|
|
#############################################################################
|
|
|
|
|
|
#############################################################################
|
|
# hook functions prototypes
|
|
#
|
|
# The "pre_state_XXX_hook" functions are called before the main
|
|
# work is done for the state XXX
|
|
#
|
|
# The "post_state_XXX_hook" functions are called after the main
|
|
# work is done for the state XXX
|
|
#
|
|
# These functions are meant to be overridden by the user's
|
|
# dhclient-enter-hooks file
|
|
#############################################################################
|
|
|
|
pre_state_MEDIUM_hook () { }
|
|
pre_state_PREINIT_hook () { }
|
|
pre_state_ARPCHECK_hook () { }
|
|
pre_state_ARPSEND_hook () { }
|
|
pre_state_RENEW_hook () { }
|
|
pre_state_REBIND_hook () { }
|
|
pre_state_BOUND_hook () { }
|
|
pre_state_REBOOT_hook () { }
|
|
pre_state_EXPIRE_hook () { }
|
|
pre_state_FAIL_hook () { }
|
|
pre_state_TIMEOUT_hook () { }
|
|
post_state_MEDIUM_hook () { }
|
|
post_state_PREINIT_hook () { }
|
|
post_state_ARPCHECK_hook () { }
|
|
post_state_ARPSEND_hook () { }
|
|
post_state_RENEW_hook () { }
|
|
post_state_REBIND_hook () { }
|
|
post_state_BOUND_hook () { }
|
|
post_state_REBOOT_hook () { }
|
|
post_state_EXPIRE_hook () { }
|
|
post_state_FAIL_hook () { }
|
|
post_state_TIMEOUT_hook () { }
|
|
|
|
#############################################################################
|
|
# make_resolv_conf
|
|
#
|
|
# This function is called to update the information related to the
|
|
# DNS configuration (the resolver part)
|
|
#############################################################################
|
|
make_resolv_conf ()
|
|
{
|
|
if [ "x$new_domain_name" != x ] && [ "x$new_domain_name_servers" != x ]; then
|
|
echo search $new_domain_name >/etc/resolv.conf
|
|
for nameserver in $new_domain_name_servers; do
|
|
echo nameserver $nameserver >>/etc/resolv.conf
|
|
done
|
|
fi
|
|
}
|
|
|
|
# Must be used on exit. Invokes the local dhcp client exit hooks, if any.
|
|
exit_with_hooks () {
|
|
exit_status=$1
|
|
if [ -x /etc/dhclient-exit-hooks ]; then
|
|
. /etc/dhclient-exit-hooks
|
|
fi
|
|
# probably should do something with exit status of the local script
|
|
return $exit_status
|
|
}
|
|
|
|
#############################################################################
|
|
# set_XXX
|
|
# unset_XXX
|
|
#
|
|
# These function each deal with one particular setting.
|
|
# They are OS dependent and may be overridden in the
|
|
# dhclient-enter-hooks file if needed.
|
|
#
|
|
# These functions are called with either "new" or "old" to indicate which
|
|
# set of variables to use (new_ip_address or old_ip_address...)
|
|
#
|
|
#############################################################################
|
|
|
|
update_hostname ()
|
|
{
|
|
local current_hostname=`/bin/hostname`
|
|
if [ "$current_hostname" = "" ] || \
|
|
[ "$current_hostname" = "$old_host_name" ]
|
|
then
|
|
if [ "$new_host_name" != "$old_host_name" ]
|
|
then
|
|
$LOGGER "New Hostname: $new_host_name"
|
|
hostname $new_host_name
|
|
fi
|
|
fi
|
|
}
|
|
|
|
set_ip_address ()
|
|
{
|
|
local ip
|
|
local mask
|
|
local bcast
|
|
|
|
if [ $# -lt 1 ]
|
|
then
|
|
return 1
|
|
fi
|
|
|
|
eval ip="\$${1}_ip_address"
|
|
eval mask="\$${1}_subnet_mask"
|
|
eval bcast="\$${1}_broadcast_address"
|
|
|
|
if [ "$ip" != "" ]
|
|
then
|
|
ifconfig $interface inet $ip netmask $mask broadcast $bcast $medium
|
|
# route add $ip 127.0.0.1 > /dev/null 2>&1
|
|
fi
|
|
}
|
|
|
|
unset_ip_address ()
|
|
{
|
|
local ip
|
|
|
|
if [ $# -lt 1 ]
|
|
then
|
|
return 1
|
|
fi
|
|
|
|
eval ip="\$${1}_ip_address"
|
|
|
|
if [ "$ip" != "" ]
|
|
then
|
|
ifconfig $interface inet -alias $ip $medium
|
|
# route delete $ip 127.0.0.1 > /dev/null 2>&1
|
|
fi
|
|
}
|
|
|
|
set_ip_alias ()
|
|
{
|
|
if [ "$alias_ip_address" != "" ]
|
|
then
|
|
ifconfig $interface inet alias $alias_ip_address netmask $alias_subnet_mask
|
|
# route add $alias_ip_address 127.0.0.1
|
|
fi
|
|
}
|
|
|
|
unset_ip_alias ()
|
|
{
|
|
if [ "$alias_ip_address" != "" ]
|
|
then
|
|
ifconfig $interface inet -alias $alias_ip_address > /dev/null 2>&1
|
|
# route delete $alias_ip_address 127.0.0.1 > /dev/null 2>&1
|
|
fi
|
|
}
|
|
|
|
set_routers ()
|
|
{
|
|
local router_list
|
|
|
|
if [ $# -lt 1 ]
|
|
then
|
|
return 1
|
|
fi
|
|
|
|
eval router_list="\$${1}_routers"
|
|
|
|
for router in $router_list
|
|
do
|
|
route add default $router >/dev/null 2>&1
|
|
done
|
|
}
|
|
|
|
unset_routers ()
|
|
{
|
|
local router_list
|
|
|
|
if [ $# -lt 1 ]
|
|
then
|
|
return 1
|
|
fi
|
|
|
|
eval router_list="\$${1}_routers"
|
|
|
|
for router in $router_list
|
|
do
|
|
route delete default $router >/dev/null 2>&1
|
|
done
|
|
}
|
|
|
|
set_static_routes ()
|
|
{
|
|
local static_routes
|
|
|
|
if [ $# -lt 1 ]
|
|
then
|
|
return 1
|
|
fi
|
|
|
|
eval static_routes="\$${1}_static_routes"
|
|
|
|
set static_routes
|
|
|
|
while [ $# -ge 2 ]
|
|
do
|
|
$LOGGER "New Static Route: $1 -> $2"
|
|
route add $1 $2
|
|
shift; shift
|
|
done
|
|
}
|
|
|
|
unset_static_routes ()
|
|
{
|
|
local static_routes
|
|
|
|
if [ $# -lt 1 ]
|
|
then
|
|
return 1
|
|
fi
|
|
|
|
eval static_routes="\$${1}_static_routes"
|
|
|
|
set static_routes
|
|
|
|
while [ $# -ge 2 ]
|
|
do
|
|
route delete $1 $2
|
|
shift; shift
|
|
done
|
|
}
|
|
|
|
#############################################################################
|
|
#
|
|
# utility functions grouping what needs to be done in logical units.
|
|
#
|
|
#############################################################################
|
|
|
|
set_all ()
|
|
{
|
|
set_ip_address new
|
|
set_routers new
|
|
set_static_routes new
|
|
|
|
if [ "$new_ip_address" != "$alias_ip_address" ]
|
|
then
|
|
set_ip_alias
|
|
fi
|
|
}
|
|
|
|
set_others ()
|
|
{
|
|
update_hostname
|
|
make_resolv_conf
|
|
}
|
|
|
|
clear_arp_table ()
|
|
{
|
|
arp -d -a
|
|
}
|
|
|
|
unset_all ()
|
|
{
|
|
if [ "$alias_ip_address" != "$old_ip_address" ]
|
|
then
|
|
unset_ip_alias
|
|
fi
|
|
|
|
if [ "$old_ip_address" != "" ]
|
|
then
|
|
unset_ip_address old
|
|
unset_routers old
|
|
unset_static_routes old
|
|
clear_arp_table
|
|
fi
|
|
}
|
|
|
|
test_new_lease ()
|
|
{
|
|
local rc
|
|
|
|
set $new_routers
|
|
|
|
if [ $# -ge 1 ]
|
|
then
|
|
set_ip_address new
|
|
if ping -q -c 1 $1
|
|
then
|
|
rc=0
|
|
else
|
|
rc=1
|
|
fi
|
|
unset_ip_address new
|
|
else
|
|
rc=1
|
|
fi
|
|
return $rc
|
|
}
|
|
|
|
#############################################################################
|
|
# Main State functions.
|
|
#
|
|
# There is a state function for each state of the DHCP client
|
|
# These functions are OS specific and should be be tampered with.
|
|
#############################################################################
|
|
|
|
in_state_MEDIUM ()
|
|
{
|
|
ifconfig $interface $medium
|
|
ifconfig $interface inet -alias 0.0.0.0 $medium >/dev/null 2>&1
|
|
sleep 1
|
|
exit_status=0
|
|
}
|
|
|
|
in_state_PREINIT ()
|
|
{
|
|
unset_ip_alias
|
|
|
|
ifconfig $interface inet 0.0.0.0 netmask 0.0.0.0 \
|
|
broadcast 255.255.255.255 up
|
|
exit_status=0
|
|
}
|
|
|
|
in_state_ARPCHECK ()
|
|
{
|
|
exit_status=0
|
|
}
|
|
|
|
in_state_ARPSEND ()
|
|
{
|
|
exit_status=0
|
|
}
|
|
|
|
in_state_RENEW ()
|
|
{
|
|
if [ "$old_ip_address" != "$new_ip_address" ]
|
|
then
|
|
unset_all
|
|
set_all
|
|
fi
|
|
|
|
set_others
|
|
}
|
|
|
|
in_state_REBIND () {
|
|
in_state_RENEW
|
|
}
|
|
|
|
in_state_BOUND () {
|
|
unset_all
|
|
set_all
|
|
set_others
|
|
}
|
|
|
|
in_state_REBOOT () {
|
|
in_state_BOUND
|
|
}
|
|
|
|
in_state_EXPIRE ()
|
|
{
|
|
unset_all
|
|
set_ip_alias
|
|
exit_status=0
|
|
}
|
|
|
|
in_state_FAIL () {
|
|
in_state_EXPIRE
|
|
}
|
|
|
|
in_state_TIMEOUT ()
|
|
{
|
|
unset_all
|
|
|
|
if test_new_lease
|
|
then
|
|
set_all
|
|
set_others
|
|
else
|
|
$LOGGER "No good lease information in TIMEOUT state"
|
|
set_ip_alias
|
|
exit_status=1
|
|
fi
|
|
}
|
|
|
|
#############################################################################
|
|
# Main functions:
|
|
#
|
|
# dhclient_script_init() parses the optional "enter_hooks" script which can
|
|
# override any of the state functions
|
|
#
|
|
# This function also parses the variables and notifies the detected changes.
|
|
#############################################################################
|
|
dhclient_script_init ()
|
|
{
|
|
if [ -x /usr/bin/logger ]; then
|
|
LOGGER="/usr/bin/logger -s -p user.notice -t dhclient"
|
|
else
|
|
LOGGER=echo
|
|
fi
|
|
|
|
# Invoke the local dhcp client enter hooks, if they exist.
|
|
if [ -x /etc/dhclient-enter-hooks ]
|
|
then
|
|
exit_status=0
|
|
. /etc/dhclient-enter-hooks
|
|
# allow the local script to abort processing of this state
|
|
# local script must set exit_status variable to nonzero.
|
|
if [ $exit_status -ne 0 ]
|
|
then
|
|
exit $exit_status
|
|
fi
|
|
fi
|
|
|
|
if [ "$new_network_number" != "" ]
|
|
then
|
|
$LOGGER "New Network Number: $new_network_number"
|
|
fi
|
|
|
|
if [ "$new_ip_address" != "" ]
|
|
then
|
|
$LOGGER "New IP Address: $new_ip_address"
|
|
fi
|
|
|
|
if [ "$new_broadcast_address" != "" ]
|
|
then
|
|
$LOGGER "New Broadcast Address: $new_broadcast_address"
|
|
fi
|
|
|
|
if [ "$new_subnet_mask" != "" ]
|
|
then
|
|
$LOGGER "New Subnet Mask for $interface: $new_subnet_mask"
|
|
fi
|
|
|
|
if [ "$alias_subnet_mask" != "" ]
|
|
then
|
|
fi
|
|
}
|
|
|
|
#############################################################################
|
|
# dhclient_main() does the appropriate work depending on the state of
|
|
# the dhcp client
|
|
#############################################################################
|
|
dhclient_script_main ()
|
|
{
|
|
# set -x
|
|
exit_status=0
|
|
|
|
case $reason in
|
|
MEDIUM|\
|
|
PREINIT|\
|
|
ARPCHECK|\
|
|
ARPSEND|\
|
|
RENEW|\
|
|
REBIND|\
|
|
BOUND|\
|
|
REBOOT|\
|
|
EXPIRE|\
|
|
FAIL|\
|
|
TIMEOUT)
|
|
pre_state_${reason}_hook
|
|
in_state_${reason}
|
|
post_state_${reason}_hook
|
|
;;
|
|
*)
|
|
$LOGGER "dhclient-script called with invalid reason $reason"
|
|
exit_status=1
|
|
;;
|
|
esac
|
|
|
|
exit_with_hooks $exit_status
|
|
}
|
|
|
|
#############################################################################
|
|
# Let's do the work...
|
|
#############################################################################
|
|
|
|
dhclient_script_init
|
|
dhclient_script_main
|
|
exit $exit_status
|
|
|
|
#############################################################################
|
|
# That's all folks
|
|
#############################################################################
|