如何向后读取IP地址?


19

如果我们有以下字符串(IP地址):192.168.1.1

如何从此字符串派生(DNS反向记录形式),因此它将像1.1.168.192.in-addr.arpa使用Shell脚本一样显示?


它必须与外壳?
Braiam

1
停止这样做
迈克尔·莫罗泽克

1
有了这么多的答案,最好将其放在codegolf中;)
tkausl 2014年

@tkausl,如果您愿意的话,在这里分享:)
Networker

Answers:


29

您可以使用AWK做到这一点。有更好的方法可以做到这一点,但是我认为这是最简单的方法。

echo '192.168.1.1' | awk 'BEGIN{FS="."}{print $4"."$3"."$2"."$1".in-addr.arpa"}'

这将颠倒IP地址的顺序。

正如Mikel所建议的那样,仅保存一些按键,我们可以进一步缩短上面的语句:

echo '192.168.1.1' | awk -F . '{print $4"."$3"."$2"."$1".in-addr.arpa"}'

要么

echo '192.168.1.1' | awk -F. '{print $4"."$3"."$2"."$1".in-addr.arpa"}'

要么

echo '192.168.1.1' | awk -F. -vOFS=. '{print $4,$3,$2,$1,"in-addr.arpa"}'

AWK非常灵活。:)


如果要保存一些击键,-F .应等效于BEGIN{FS="."}
Mikel

33

只是出于好奇值...使用tac来自GNU的coreutils:给定一个变量ip的形式192.168.1.1,然后

$(printf %s "$ip." | tac -s.)in-addr.arpa

$ ip=192.168.1.1
$ rr=$(printf %s "$ip." | tac -s.)in-addr.arpa
$ echo "$rr"
1.1.168.192.in-addr.arpa

9
+1这绝对是我最喜欢的答案。感谢您向我介绍tac
乔纳森·莱因哈特2014年

也要感谢@StéphaneChazelas进行的优雅printf编辑(我最初发布了一个丑陋的图片echo -n
steeldriver

1
显然更简单:printf 'arpa.in-addr.%s.' "$ip" | tac -s.
艾萨克(Isaac)

17

如果你想只用shell( ,zshksh93),bash这里是另一种方式:

IFS=. read w x y z <<<'192.168.1.1'
printf '%d.%d.%d.%d.in-addr.arpa.' "$z" "$y" "$x" "$w"

或在普通的旧外壳中:

echo '192.168.1.1' | { IFS=. read w x y z; echo "$z.$y.$w.$x.in-addr.arpa."; }

12

通过Perl轻松实现:

$ echo 192.168.1.1|perl -nle 'print join ".",reverse(split /\./,$_)'
1.1.168.192

6
可以做得更紧凑:perl -F'\.' -lane '$,=".";print reverse @F'
Joseph R.

6

为了解决这个问题,Ruby:

ruby -r ipaddr -e 'puts IPAddr.new(ARGV.first).reverse' 192.168.1.1

其中也支持IPv6

2607:F8B0:4000:080A:0000:0000:0000:2000
=> 0.0.0.2.0.0.0.0.0.0.0.0.0.0.0.0.a.0.8.0.0.0.0.4.0.b.8.f.7.0.6.2.ip6.arpa

您可以从脚本中删除“ require”:ruby -r ipaddr -e 'puts ...'
glenn jackman 2014年

5

通过GNU sed

sed -r 's/^([0-9]{1,3}).([0-9]{1,3}).([0-9]{1,3}).([0-9]{1,3})$/\4.\3.\2.\1.in-addr.arpa/g' file

它会反转任何IPv4地址格式。

例:

$ echo '192.168.1.1' | sed -r 's/^([0-9]{1,3}).([0-9]{1,3}).([0-9]{1,3}).([0-9]{1,3})$/\4.\3.\2.\1.in-addr.arpa/g'
1.1.168.192.in-addr.arpa

$ echo '192.1.1.1' | sed -r 's/^([0-9]{1,3}).([0-9]{1,3}).([0-9]{1,3}).([0-9]{1,3})$/\4.\3.\2.\1.in-addr.arpa/g'
1.1.1.192.in-addr.arpa

$ echo '1.1.1.1' | sed -r 's/^([0-9]{1,3}).([0-9]{1,3}).([0-9]{1,3}).([0-9]{1,3})$/\4.\3.\2.\1.in-addr.arpa/g'
1.1.1.1.in-addr.arpa

$ sed -r 's/^([0-9]{1,3}).([0-9]{1,3}).([0-9]{1,3}).([0-9]{1,3})$/\4.\3.\2.\1.in-addr.arpa/g' <<< '192.168.189.23'
23.189.168.192.in-addr.arpa

1
这是最快的解决方案!45k IP只需0.794s即可转换。
Petr Javorik

4

使用Python的标准库

>>> ipaddress.ip_address('192.168.1.1').reverse_pointer
'1.1.168.192.in-addr.arpa'

1
它位于3.5标准库中,计划于2015年9月发布。您可以通过脚本执行以下操作python3.5 -c "import ipaddress; ipaddress.ip_address('192.168.1.1').reverse_pointer":(全部在一行上)
Anthon

3

zsh

$ ip=192.168.1.1
$ echo ${(j:.:)${(s:.:Oa)ip}}.in-addr.arpa
1.1.168.192.in-addr.arpa

这些是变量扩展标志:

  • s:.:小号上PLIT.
  • Oa:反向Ó刻申的一个 rray
  • j:.:Ĵ OIN上.

3

另一种可能性是将“ dig”命令行工具与“ -x”开关一起使用。

它实际上是在PTR条目上进行请求,但是如果您过滤“ PTR”,它将显示一条注释行(该请求),并且可能还会有一些答复。

使用“ dig”可以方便地快速编写PTR名称,而不必编写小脚本。特别是如果您需要交互地使用它(剪切和粘贴结果)。它也适用于IPv6。


2

如果您希望它也能与IPv6一起使用,则可以使用dig -x

例如:

$ dig -x 194.68.208.240 | egrep '^;.*PTR$' | cut -c 2- | awk '{print $1}'
240.208.68.194.in-addr.arpa.

$ dig -x 2001:db8:dc61:2a61::1 | egrep '^;.*PTR$' | cut -c 2- | awk '{print $1}'
1.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.1.6.a.2.1.6.c.d.8.b.d.0.1.0.0.2.ip6.arpa.

2

在Python中

 a = "192.168.1.122"
 import re
 m = re.search(r'(\d+)\.(\d+)\.(\d+)\.(\d+)',a)
 ip = m.group(4),m.group(3),m.group(2),m.group(1)
 '.'.join(ip) + ".in-addr.arpa"
 '122.1.168.192.in-addr.arpa'

2
或更习惯上,"{}.in-addr.arpa".format(".".join(reversed(a.split("."))))在Python2.7中
iruvar,2014年

2
IFS=. ; set -- ${0+192.168.1.2}
printf %b. $4 $3 $2 $1 in-addr.arpa\\c

IFS=. ; printf %s\\n \
    in-addr.arpa ${0+192.168.1.2} |    
sed '1!G;$s/\n/./gp;h;d'

IFS=. ; printf '[%b.] ' \
    ${0+192.168.1.2.]PPPPP\\c} |dc
echo in-addr.arpa

1
$ while read ip
while> do
while> n=( $(echo $ip) ) && echo "${n[4]}"'.'"${n[3]}"'.'"${n[2]}"'.'"${n[1]}"'.'"in-addr.arpa"
while> done
192.168.1.2
2.1.168.192.in-addr.arpa

这样,您可以输入地址,然后按回车以得到结果。


1

使用host来自的命令dnsutils

$ host -t ptr 192.168.1.1 | cut -d' ' -f 2
1.1.168.192.in-addr.arpa.

这仅适用于您可以在DNS中查找的地址。如果您尝试使用未知地址,则会收到错误消息(但是包含反向地址,但需要稍作不同的后处理)。
亚历山大·雷梅施

1

假设一个变种包含IP: ip=192.168.2.1。如果需要将值提供给新变量,只需将任何解决方案封装在其中$()并将其分配给varrr=$(...)

一些解决方案是可能的:

最简单的 : printf 'arpa.in-addr.%s.' "$ip" | tac -s.
大多数外壳: IFS=. eval 'set -- $ip'; echo "$4.$3.$2.$1.in-addr.arpa"
有些外壳 : IFS=. read d c b a <<<"$ip"; printf %s "$a.$b.$c.$d.in-addr.arpa."
: echo "$ip" | awk -F. '{OFS=FS;print $4,$3,$2,$1,"in-addr.arpa"}'
: echo "$ip" | sed -E 's/([^.]+)\.([^.]+)\.([^.]+)\.([^.]+)$/\4.\3.\2.\1.in-addr.arpa./'
: echo "arpa.in-addr.$ip" | sed 'y/./\n/' | sed 'G;$s/\n/./gp;h;d'
: echo "$ip" | perl -F\\. -lane '$,=".";print( join(".",(reverse @F),"in-addr.arpa"))'
: dig -x "$ip" | awk -F '[; \t]+' '/^;.*PTR$/{print($2)}'
: host -t ptr 192.168.2.1 | cut -d' ' -f2

挖掘和主机解决方案都可以与IPv6一起使用。


1
#!/bin/bash
# script file name reverseip.sh
if [ -z $1 ] || [ "help" == $1 ]
then
echo 'Convert a full ipv4 or ipv6 address to arpa notation'
echo "usage:"
echo ./reverseip.sh "help"
echo ./reverseip.sh "ipv4 address format: xxxx.xxxx.xxxx.xxxx"
echo ./reverseip.sh "ipv6 address format: xxxx:xxxx:xxxx:xxxx::xxxx"
echo "examples:"
echo ./reverseip.sh 216.58.207.35
echo ./reverseip.sh 2a00:1450:4001:824::2003
exit
fi

# if ip address passed containing ':'
if [[ $1 = *':'* ]];
then
# invert ipv6 address e.g.: 2a00:1450:4001:824::2003 to 3.0.0.2.0.0.0.0.0.0.0.0.0.0.0.0.4.2.8.0.1.0.0.4.0.5.4.1.0.0.a.2.
# @see lsowen https://gist.github.com/lsowen/4447d916fd19cbb7fce4
echo "$1" | awk -F: 'BEGIN {OFS=""; }{addCount = 9 - NF; for(i=1; i<=NF;i++){if(length($i) == 0){ for(j=1;j<=addCount;j++){$i = ($i "0000");} } else { $i = substr(("0000" $i), length($i)+5-4);}}; print}' | rev | sed -e "s/./&./g" | echo "$(</dev/stdin)ip6.arpa."
else
# invert ipv6 address e.g.: 216.58.207.35 to 35.207.58.216.in-addr.arpa
# @see Stéphane Chazelas /unix/132779/how-to-read-an-ip-address-backwards
echo $(printf %s "$1." | tac -s.)in-addr.arpa
fi

0

用更少的工具代替mattbianco的答案的较短替代方法,但是使用pcregrep可以是:

$ dig -x 194.68.208.240 | pcregrep -o1 '^;(\S+)\s+IN\s+PTR$'
240.208.68.194.in-addr.arpa.

$ dig -x 2001:db8:dc61:2a61::1 | pcregrep -o1 '^;(\S+)\s+IN\s+PTR$'
1.0.0.0.0.0.0.0.0.0.0.0.0.0.0.0.1.6.a.2.1.6.c.d.8.b.d.0.1.0.0.2.ip6.arpa.

0

使用以下一行shell:

echo '192.168.1.1' | tr '.' '\n' | tac | paste -s -d '.' -
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.