43df5c
#!/bin/bash
43df5c
#
43df5c
# iptables	Start iptables firewall
43df5c
#
43df5c
# chkconfig: 2345 08 92
43df5c
# description:	Starts, stops and saves iptables firewall
43df5c
#
43df5c
# config: /etc/sysconfig/iptables
43df5c
# config: /etc/sysconfig/iptables-config
43df5c
#
43df5c
### BEGIN INIT INFO
43df5c
# Provides: iptables
43df5c
# Required-Start:
43df5c
# Required-Stop:
43df5c
# Default-Start: 2 3 4 5
43df5c
# Default-Stop: 0 1 6
43df5c
# Short-Description: start and stop iptables firewall
43df5c
# Description: Start, stop and save iptables firewall
43df5c
### END INIT INFO
43df5c
43df5c
# Source function library.
43df5c
. /etc/init.d/functions
43df5c
43df5c
IPTABLES=iptables
43df5c
IPTABLES_DATA=/etc/sysconfig/$IPTABLES
43df5c
IPTABLES_FALLBACK_DATA=${IPTABLES_DATA}.fallback
43df5c
IPTABLES_CONFIG=/etc/sysconfig/${IPTABLES}-config
43df5c
IPV=${IPTABLES%tables} # ip for ipv4 | ip6 for ipv6
43df5c
[ "$IPV" = "ip" ] && _IPV="ipv4" || _IPV="ipv6"
43df5c
PROC_IPTABLES_NAMES=/proc/net/${IPV}_tables_names
43df5c
VAR_SUBSYS_IPTABLES=/var/lock/subsys/$IPTABLES
43df5c
RESTORECON=$(which restorecon 2>/dev/null)
43df5c
[ ! -x "$RESTORECON" ] && RESTORECON=/bin/true
43df5c
43df5c
# only usable for root
43df5c
if [ $EUID != 0 ]; then
43df5c
    echo -n $"${IPTABLES}: Only usable by root."; warning; echo
43df5c
    exit 4
43df5c
fi
43df5c
43df5c
if [ ! -x /sbin/$IPTABLES ]; then
43df5c
    echo -n $"${IPTABLES}: /sbin/$IPTABLES does not exist."; warning; echo
43df5c
    exit 5
43df5c
fi
43df5c
43df5c
# Default firewall configuration:
43df5c
IPTABLES_MODULES=""
43df5c
IPTABLES_SAVE_ON_STOP="no"
43df5c
IPTABLES_SAVE_ON_RESTART="no"
43df5c
IPTABLES_SAVE_COUNTER="no"
43df5c
IPTABLES_STATUS_NUMERIC="yes"
43df5c
IPTABLES_STATUS_VERBOSE="no"
43df5c
IPTABLES_STATUS_LINENUMBERS="yes"
43df5c
IPTABLES_SYSCTL_LOAD_LIST=""
43df5c
IPTABLES_RESTORE_WAIT=600
43df5c
IPTABLES_RESTORE_WAIT_INTERVAL=1000000
43df5c
43df5c
# Load firewall configuration.
43df5c
[ -f "$IPTABLES_CONFIG" ] && . "$IPTABLES_CONFIG"
43df5c
43df5c
# Get active tables
43df5c
NF_TABLES=$(cat "$PROC_IPTABLES_NAMES" 2>/dev/null)
43df5c
43df5c
# Prepare commands for wait options
43df5c
IPTABLES_CMD="$IPTABLES"
43df5c
IPTABLES_RESTORE_CMD="$IPTABLES-restore"
43df5c
if [ $IPTABLES_RESTORE_WAIT -ne 0 ]; then
43df5c
	OPT="--wait ${IPTABLES_RESTORE_WAIT}"
43df5c
	if [ $IPTABLES_RESTORE_WAIT_INTERVAL -lt 1000000 ]; then
43df5c
	    OPT+=" --wait-interval ${IPTABLES_RESTORE_WAIT_INTERVAL}"
43df5c
	fi
43df5c
	IPTABLES_CMD+=" $OPT"
43df5c
	IPTABLES_RESTORE_CMD+=" $OPT"
43df5c
fi
43df5c
43df5c
flush_n_delete() {
43df5c
    local ret=0
43df5c
43df5c
    # Flush firewall rules and delete chains.
43df5c
    [ ! -e "$PROC_IPTABLES_NAMES" ] && return 0
43df5c
43df5c
    # Check if firewall is configured (has tables)
43df5c
    [ -z "$NF_TABLES" ] && return 1
43df5c
43df5c
    echo -n $"${IPTABLES}: Flushing firewall rules: "
43df5c
    # For all tables
43df5c
    for i in $NF_TABLES; do
43df5c
        # Flush firewall rules.
43df5c
	$IPTABLES_CMD -t $i -F;
43df5c
	let ret+=$?;
43df5c
43df5c
        # Delete firewall chains.
43df5c
	$IPTABLES_CMD -t $i -X;
43df5c
	let ret+=$?;
43df5c
43df5c
	# Set counter to zero.
43df5c
	$IPTABLES_CMD -t $i -Z;
43df5c
	let ret+=$?;
43df5c
    done
43df5c
43df5c
    [ $ret -eq 0 ] && success || failure
43df5c
    echo
43df5c
    return $ret
43df5c
}
43df5c
43df5c
set_policy() {
43df5c
    local ret=0
43df5c
43df5c
    # Set policy for configured tables.
43df5c
    policy=$1
43df5c
43df5c
    # Check if iptable module is loaded
43df5c
    [ ! -e "$PROC_IPTABLES_NAMES" ] && return 0
43df5c
43df5c
    # Check if firewall is configured (has tables)
43df5c
    tables=$(cat "$PROC_IPTABLES_NAMES" 2>/dev/null)
43df5c
    [ -z "$tables" ] && return 1
43df5c
43df5c
    echo -n $"${IPTABLES}: Setting chains to policy $policy: "
43df5c
    for i in $tables; do
43df5c
	echo -n "$i "
43df5c
	case "$i" in
43df5c
	    raw)
43df5c
		$IPTABLES_CMD -t raw -P PREROUTING $policy \
43df5c
		    && $IPTABLES_CMD -t raw -P OUTPUT $policy \
43df5c
		    || let ret+=1
43df5c
		;;
43df5c
	    filter)
43df5c
                $IPTABLES_CMD -t filter -P INPUT $policy \
43df5c
		    && $IPTABLES_CMD -t filter -P OUTPUT $policy \
43df5c
		    && $IPTABLES_CMD -t filter -P FORWARD $policy \
43df5c
		    || let ret+=1
43df5c
		;;
43df5c
	    nat)
43df5c
		$IPTABLES_CMD -t nat -P PREROUTING $policy \
43df5c
		    && $IPTABLES_CMD -t nat -P POSTROUTING $policy \
43df5c
		    && $IPTABLES_CMD -t nat -P OUTPUT $policy \
43df5c
		    || let ret+=1
43df5c
		;;
43df5c
	    mangle)
43df5c
	        $IPTABLES_CMD -t mangle -P PREROUTING $policy \
43df5c
		    && $IPTABLES_CMD -t mangle -P POSTROUTING $policy \
43df5c
		    && $IPTABLES_CMD -t mangle -P INPUT $policy \
43df5c
		    && $IPTABLES_CMD -t mangle -P OUTPUT $policy \
43df5c
		    && $IPTABLES_CMD -t mangle -P FORWARD $policy \
43df5c
		    || let ret+=1
43df5c
		;;
43df5c
	    security)
43df5c
	        # Ignore the security table
43df5c
	        ;;
43df5c
	    *)
43df5c
	        let ret+=1
43df5c
		;;
43df5c
        esac
43df5c
    done
43df5c
43df5c
    [ $ret -eq 0 ] && success || failure
43df5c
    echo
43df5c
    return $ret
43df5c
}
43df5c
43df5c
load_sysctl() {
43df5c
    local ret=0
43df5c
43df5c
    # load matched sysctl values
43df5c
    if [ -n "$IPTABLES_SYSCTL_LOAD_LIST" ]; then
43df5c
        echo -n $"Loading sysctl settings: "
43df5c
        for item in $IPTABLES_SYSCTL_LOAD_LIST; do
43df5c
            fgrep -hs $item /etc/sysctl.d/* | sysctl -p - >/dev/null
43df5c
            let ret+=$?;
43df5c
        done
43df5c
        [ $ret -eq 0 ] && success || failure
43df5c
        echo
43df5c
    fi
43df5c
    return $ret
43df5c
}
43df5c
43df5c
start() {
43df5c
    local ret=0
43df5c
43df5c
    # Do not start if there is no config file.
43df5c
    if [ ! -f "$IPTABLES_DATA" ]; then
43df5c
	echo -n $"${IPTABLES}: No config file."; warning; echo
43df5c
	return 6
43df5c
    fi
43df5c
43df5c
    # check if ipv6 module load is deactivated
43df5c
    if [ "${_IPV}" = "ipv6" ] \
43df5c
	&& grep -qIsE "^install[[:space:]]+${_IPV}[[:space:]]+/bin/(true|false)" /etc/modprobe.conf /etc/modprobe.d/* ; then
43df5c
	echo $"${IPTABLES}: ${_IPV} is disabled."
43df5c
	return 150
43df5c
    fi
43df5c
43df5c
    echo -n $"${IPTABLES}: Applying firewall rules: "
43df5c
43df5c
    OPT=
43df5c
    [ "x$IPTABLES_SAVE_COUNTER" = "xyes" ] && OPT="-c"
43df5c
43df5c
    $IPTABLES_RESTORE_CMD $OPT $IPTABLES_DATA
43df5c
    if [ $? -eq 0 ]; then
43df5c
	success; echo
43df5c
    else
43df5c
	failure; echo;
43df5c
	if [ -f "$IPTABLES_FALLBACK_DATA" ]; then
43df5c
	    echo -n $"${IPTABLES}: Applying firewall fallback rules: "
43df5c
	    $IPTABLES_RESTORE_CMD $OPT $IPTABLES_FALLBACK_DATA
43df5c
	    if [ $? -eq 0 ]; then
43df5c
		success; echo
43df5c
	    else
43df5c
		failure; echo; return 1
43df5c
	    fi
43df5c
	else
43df5c
	    return 1
43df5c
	fi
43df5c
    fi
43df5c
    
43df5c
    # Load additional modules (helpers)
43df5c
    if [ -n "$IPTABLES_MODULES" ]; then
43df5c
	echo -n $"${IPTABLES}: Loading additional modules: "
43df5c
	for mod in $IPTABLES_MODULES; do
43df5c
	    echo -n "$mod "
43df5c
	    modprobe $mod > /dev/null 2>&1
43df5c
	    let ret+=$?;
43df5c
	done
43df5c
	[ $ret -eq 0 ] && success || failure
43df5c
	echo
43df5c
    fi
43df5c
    
43df5c
    # Load sysctl settings
43df5c
    load_sysctl
43df5c
43df5c
    touch $VAR_SUBSYS_IPTABLES
43df5c
    return $ret
43df5c
}
43df5c
43df5c
stop() {
43df5c
    local ret=0
43df5c
43df5c
    # Do not stop if iptables module is not loaded.
43df5c
    [ ! -e "$PROC_IPTABLES_NAMES" ] && return 0
43df5c
43df5c
    # Set default chain policy to ACCEPT, in order to not break shutdown
43df5c
    # on systems where the default policy is DROP and root device is
43df5c
    # network-based (i.e.: iSCSI, NFS)
43df5c
    set_policy ACCEPT
43df5c
    let ret+=$?
43df5c
    # And then, flush the rules and delete chains
43df5c
    flush_n_delete
43df5c
    let ret+=$?
43df5c
    
43df5c
    rm -f $VAR_SUBSYS_IPTABLES
43df5c
    return $ret
43df5c
}
43df5c
43df5c
save() {
43df5c
    local ret=0
43df5c
43df5c
    # Check if iptable module is loaded
43df5c
    if [ ! -e "$PROC_IPTABLES_NAMES" ]; then
43df5c
	echo -n $"${IPTABLES}: Nothing to save."; warning; echo
43df5c
	return 0
43df5c
    fi
43df5c
43df5c
    # Check if firewall is configured (has tables)
43df5c
    if [ -z "$NF_TABLES" ]; then
43df5c
	echo -n $"${IPTABLES}: Nothing to save."; warning; echo
43df5c
	return 6
43df5c
    fi
43df5c
43df5c
    echo -n $"${IPTABLES}: Saving firewall rules to $IPTABLES_DATA: "
43df5c
43df5c
    OPT=
43df5c
    [ "x$IPTABLES_SAVE_COUNTER" = "xyes" ] && OPT="-c"
43df5c
43df5c
    TMP_FILE=$(/bin/mktemp -q $IPTABLES_DATA.XXXXXX) \
43df5c
	&& chmod 600 "$TMP_FILE" \
43df5c
	&& $IPTABLES-save $OPT > $TMP_FILE 2>/dev/null \
43df5c
	&& size=$(stat -c '%s' $TMP_FILE) && [ $size -gt 0 ] \
43df5c
	|| ret=1
43df5c
    if [ $ret -eq 0 ]; then
43df5c
	if [ -e $IPTABLES_DATA ]; then
43df5c
	    cp -f $IPTABLES_DATA $IPTABLES_DATA.save \
43df5c
		&& chmod 600 $IPTABLES_DATA.save \
43df5c
		&& $RESTORECON $IPTABLES_DATA.save \
43df5c
		|| ret=1
43df5c
	fi
43df5c
	if [ $ret -eq 0 ]; then
43df5c
	    mv -f $TMP_FILE $IPTABLES_DATA \
43df5c
		&& chmod 600 $IPTABLES_DATA \
43df5c
		&& $RESTORECON $IPTABLES_DATA \
43df5c
	        || ret=1
43df5c
	fi
43df5c
    fi
43df5c
    rm -f $TMP_FILE
43df5c
    [ $ret -eq 0 ] && success || failure
43df5c
    echo
43df5c
    return $ret
43df5c
}
43df5c
43df5c
status() {
43df5c
    if [ ! -f "$VAR_SUBSYS_IPTABLES" ] && [ -z "$NF_TABLES" ]; then
43df5c
	echo $"${IPTABLES}: Firewall is not running."
43df5c
	return 3
43df5c
    fi
43df5c
43df5c
    # Do not print status if lockfile is missing and iptables modules are not 
43df5c
    # loaded.
43df5c
    # Check if iptable modules are loaded
43df5c
    if [ ! -e "$PROC_IPTABLES_NAMES" ]; then
43df5c
	echo $"${IPTABLES}: Firewall modules are not loaded."
43df5c
	return 3
43df5c
    fi
43df5c
43df5c
    # Check if firewall is configured (has tables)
43df5c
    if [ -z "$NF_TABLES" ]; then
43df5c
	echo $"${IPTABLES}: Firewall is not configured. "
43df5c
	return 3
43df5c
    fi
43df5c
43df5c
    NUM=
43df5c
    [ "x$IPTABLES_STATUS_NUMERIC" = "xyes" ] && NUM="-n"
43df5c
    VERBOSE=
43df5c
    [ "x$IPTABLES_STATUS_VERBOSE" = "xyes" ] && VERBOSE="--verbose"
43df5c
    COUNT=
43df5c
    [ "x$IPTABLES_STATUS_LINENUMBERS" = "xyes" ] && COUNT="--line-numbers"
43df5c
43df5c
    for table in $NF_TABLES; do
43df5c
	echo $"Table: $table"
43df5c
	$IPTABLES -t $table --list $NUM $VERBOSE $COUNT && echo
43df5c
    done
43df5c
43df5c
    return 0
43df5c
}
43df5c
43df5c
reload() {
43df5c
    local ret=0
43df5c
43df5c
    # Do not reload if there is no config file.
43df5c
    if [ ! -f "$IPTABLES_DATA" ]; then
43df5c
	echo -n $"${IPTABLES}: No config file."; warning; echo
43df5c
	return 6
43df5c
    fi
43df5c
43df5c
    # check if ipv6 module load is deactivated
43df5c
    if [ "${_IPV}" = "ipv6" ] \
43df5c
	&& grep -qIsE "^install[[:space:]]+${_IPV}[[:space:]]+/bin/(true|false)" /etc/modprobe.conf /etc/modprobe.d/* ; then
43df5c
	echo $"${IPTABLES}: ${_IPV} is disabled."
43df5c
	return 150
43df5c
    fi
43df5c
43df5c
    echo -n $"${IPTABLES}: Trying to reload firewall rules: "
43df5c
43df5c
    OPT=
43df5c
    [ "x$IPTABLES_SAVE_COUNTER" = "xyes" ] && OPT="-c"
43df5c
43df5c
    $IPTABLES_RESTORE_CMD $OPT $IPTABLES_DATA
43df5c
    if [ $? -eq 0 ]; then
43df5c
	success; echo
43df5c
    else
43df5c
	failure; echo; echo "Firewall rules are not changed."; return 1
43df5c
    fi
43df5c
43df5c
    # Load additional modules (helpers)
43df5c
    if [ -n "$IPTABLES_MODULES" ]; then
43df5c
	echo -n $"${IPTABLES}: Loading additional modules: "
43df5c
	for mod in $IPTABLES_MODULES; do
43df5c
	    echo -n "$mod "
43df5c
	    modprobe $mod > /dev/null 2>&1
43df5c
	    let ret+=$?;
43df5c
	done
43df5c
	[ $ret -eq 0 ] && success || failure
43df5c
	echo
43df5c
    fi
43df5c
43df5c
    # Load sysctl settings
43df5c
    load_sysctl
43df5c
43df5c
    return $ret
43df5c
}
43df5c
43df5c
restart() {
43df5c
    [ "x$IPTABLES_SAVE_ON_RESTART" = "xyes" ] && save
43df5c
    stop
43df5c
    start
43df5c
}
43df5c
43df5c
43df5c
case "$1" in
43df5c
    start)
43df5c
	[ -f "$VAR_SUBSYS_IPTABLES" ] && exit 0
43df5c
	start
43df5c
	RETVAL=$?
43df5c
	;;
43df5c
    stop)
43df5c
	[ "x$IPTABLES_SAVE_ON_STOP" = "xyes" ] && save
43df5c
	stop
43df5c
	RETVAL=$?
43df5c
	;;
43df5c
    restart|force-reload)
43df5c
	restart
43df5c
	RETVAL=$?
43df5c
	;;
43df5c
    reload)
43df5c
	[ -e "$VAR_SUBSYS_IPTABLES" ] && reload
43df5c
	RETVAL=$?
43df5c
	;;      
43df5c
    condrestart|try-restart)
43df5c
	[ ! -e "$VAR_SUBSYS_IPTABLES" ] && exit 0
43df5c
	restart
43df5c
	RETVAL=$?
43df5c
	;;
43df5c
    status)
43df5c
	status
43df5c
	RETVAL=$?
43df5c
	;;
43df5c
    panic)
43df5c
	set_policy DROP
43df5c
	RETVAL=$?
43df5c
        ;;
43df5c
    save)
43df5c
	save
43df5c
	RETVAL=$?
43df5c
	;;
43df5c
    *)
43df5c
	echo $"Usage: ${IPTABLES} {start|stop|reload|restart|condrestart|status|panic|save}"
43df5c
	RETVAL=2
43df5c
	;;
43df5c
esac
43df5c
43df5c
exit $RETVAL