在验证工作正常时,网络掩码验证似乎无法在bash脚本中使用正则表达式

时间:2014-02-13 06:31:02

标签: regex linux bash validation

我编写了简单的脚本来验证IP地址和网络掩码,如下所示

#!/bin/bash

validFormatIP()
{
    echo $1 | grep -w -E -o '^(25[0-4]|2[0-4][0-9]|1[0-9][0-9]|[1]?[1-9][0-9]?)\.(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\.(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\.(25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)' > /dev/null
    if [ $? -eq 0 ]
    then
        echo "Valid ipaddress"
    else
        echo "Inalid ipaddress"
    fi
}

validNetMask()
{
    echo $1 | grep -w -E -o '^(254|252|248|240|224|192|128)\\.0\\.0\\.0|255\\.(254|252|248|240|224|192|128|0)\\.0\\.0|255\\.255\\.(254|252|248|240|224|192|128|0)\\.0|255\\.255\\.255\\.(254|252|248|240|224|192|128|0)' > /dev/null
    if [ $? -eq 0 ]
    then
        echo "Valid netmask"
    else
        echo "Invalid netmask"
    fi  
}

setIpAddress()
{
    ip_address=`echo $1 | awk -F= '{print $2}'` 
    validFormatIP $ip_address
}
setNetMask()
{
    ip_address=`echo $1 | awk -F= '{print $2}'` 
    validNetMask $ip_address
}

let "n_count=0"
netmask=""

let "i_count=0"
ipaddess=""

while getopts ":i:n:" OPTION
do
    case $OPTION in
        n)
            netmask="Netmask=$OPTARG"
            let "n_count+=1"
            ;;
        i)
            ipaddess="IpAddess=$OPTARG"
            let "i_count+=1"
            ;;

        ?)  
            echo "wrong command syntax"
            ;;
    esac
done

if [ $i_count -eq 1 ]
then
    setIpAddress $ipaddess
    exit 0
fi

if [ $n_count -eq 1 ]
then
    setNetMask $netmask
    exit 0
fi

使用上面的结果我已经成功过滤掉了无效的IP地址,但是无法过滤无效的网络掩码。我已经使用不同的参数运行上面的脚本,如下所示,并在脚本执行后查看下面的输出

$ ./script.bash -i 192.168.0.1
Valid ipaddress

$./script.bash -i 255.255.0.0
Inalid ipaddress

$./script.bash -n 255.255.255.0
Invalid netmask

如上所示输出IP地址验证的结果是预期的,但即使我输入有效的网络掩码`255.255.255.0,它拒绝网络掩码的原因是什么?

任何人都知道我在网络掩码验证中错过了什么或者我的脚本中出了什么问题?

2 个答案:

答案 0 :(得分:6)

grep没有双重转义点等,所以这将有效:

validNetMask() {
   echo $1 | grep -w -E -o '^(254|252|248|240|224|192|128)\.0\.0\.0|255\.(254|252|248|240|224|192|128|0)\.0\.0|255\.255\.(254|252|248|240|224|192|128|0)\.0|255\.255\.255\.(254|252|248|240|224|192|128|0)' > /dev/null
   if [ $? -eq 0 ]; then
      echo "Valid netmask"
   else
      echo "Invalid netmask"
   fi
}

最好使用这个简洁的版本:

validNetMask() {
   grep -E -q '^(254|252|248|240|224|192|128)\.0\.0\.0|255\.(254|252|248|240|224|192|128|0)\.0\.0|255\.255\.(254|252|248|240|224|192|128|0)\.0|255\.255\.255\.(254|252|248|240|224|192|128|0)' <<< "$1" && echo "Valid netmask" || echo "Invalid netmask"
}

答案 1 :(得分:0)

矿:

validate_netmask () {
    n_masks=(${1//./ })
    [ "${#n_masks[@]}" -ne 4 ] && return 1
    for i in ${1//./ }; do
        bits=$(echo "obase=2;ibase=10;$i" | bc)
        pre=$((8-${#bits}))
        if [ "$bits" = 0 ]; then
            zeros=00000000
        elif [ "$pre" -gt 0 ]; then
            zeros=$(for ((i=1;i<=$pre;i++)); do echo -n 0; done)
        fi
        b_mask=$b_mask$zeros$bits
            unset zeros
    done
    if [ $b_mask = ${b_mask%%0*}${b_mask##*1} ]; then
        return 0
    else
        return 1
    fi
}