Como ler um endereço IP ao contrário?

19

Se tivermos esta sequência ( endereço IP ):192.168.1.1

Como posso derivar o ( formulário de registro reverso do DNS ) dessa string, para que seja mostrado como 1.1.168.192.in-addr.arpausar um script de shell?

Networker
fonte
Tem que ser com a concha?
Braiam 30/05
1
Pare de fazer isso
Michael Mrozek
1
Com essa quantidade de respostas isso deve ser melhor em codegolf;)
tkausl
@tkausl, compartilhá-lo lá, se você quiser :)
Networker

Respostas:

29

Você pode fazer isso com o AWK . Existem maneiras mais agradáveis ​​de fazer isso, mas acho que é a mais simples.

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

Isso reverterá a ordem do endereço IP.

Apenas para salvar algumas teclas, como Mikel sugeriu, podemos reduzir ainda mais a declaração superior:

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

OU

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

OU

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

O AWK é bastante flexível. :)

Neven
fonte
Se você deseja salvar algumas teclas, isso -F .deve ser equivalente a BEGIN{FS="."}.
Mikel
33

Apenas pelo valor da curiosidade ... usando tacdo GNU coreutils: dada uma variável ipna forma 192.168.1.1então

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

ie

$ ip=192.168.1.1
$ rr=$(printf %s "$ip." | tac -s.)in-addr.arpa
$ echo "$rr"
1.1.168.192.in-addr.arpa
chave de aço
fonte
9
+1 Esta é definitivamente a minha resposta favorita aqui. Obrigado por me apresentar tac!
Jonathon Reinhart
Graças são também devido à @ StéphaneChazelas para o elegante printfedição (I originalmente um feio echo -n)
steeldriver
1
Claramente mais simples:printf 'arpa.in-addr.%s.' "$ip" | tac -s.
Isaac
17

Se você quiser usar apenas Shell ( zsh, ksh93, bash), aqui está uma outra maneira:

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

Ou em concha velha lisa:

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

Facilmente com Perl, assim:

$ echo 192.168.1.1|perl -nle 'print join ".",reverse(split /\./,$_)'
1.1.168.192
JRFerguson
fonte
6
Pode ser ainda mais compacto:perl -F'\.' -lane '$,=".";print reverse @F'
Joseph R.
6

Para finalizar, Ruby:

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

Que também suporta 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
Matt
fonte
você pode tirar o "require" do script:ruby -r ipaddr -e 'puts ...'
glenn jackman
5

Através do 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

Inverte qualquer formato de endereço IPv4.

Exemplo:

$ 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
Avinash Raj
fonte
1
Esta é a solução mais rápida! IPs de 45k convertidos em apenas 0.794s.
Petr Javorik 04/10
4

Usando a biblioteca padrão do Python :

>>> ipaddress.ip_address('192.168.1.1').reverse_pointer
'1.1.168.192.in-addr.arpa'
Josh Lee
fonte
1
Isso está na biblioteca padrão 3.5 e está programado para ser lançado em setembro de 2015. A partir de um script, você pode fazer: python3.5 -c "import ipaddress; ipaddress.ip_address('192.168.1.1').reverse_pointer"(todos em uma linha)
Anthon
3

Com zsh:

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

Esses são sinalizadores de expansão variável:

  • s:.:: s plit on.
  • Oa: Inversa o rder a um rray
  • j:.:: j oin on.
Stéphane Chazelas
fonte
3

Outra possibilidade é usar a ferramenta de linha de comando "dig" com a opção "-x".

Na verdade, ele faz uma solicitação na entrada PTR, mas se você filtrar em "PTR", será exibida uma linha comentada (a solicitação) e talvez algumas respostas.

O uso de "dig" pode ser útil para escrever rapidamente o nome do PTR, sem a necessidade de escrever um pequeno script. Especialmente se você precisar interativamente (para cortar e colar o resultado). Também funciona no IPv6.

Pablo Saratxaga
fonte
2

Se você quiser que ele funcione com IPv6 também, você pode usá-lo dig -x.

Por exemplo:

$ 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.
MattBianco
fonte
2

Em 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'
c4f4t0r
fonte
2
Ou mais "{}.in-addr.arpa".format(".".join(reversed(a.split("."))))
linguisticamente
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
mikeserv
fonte
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

Dessa forma, você pode digitar um endereço e clicar em retornar para o seu resultado.

uwotm8
fonte
1

Com o hostcomando de dnsutils:

$ host -t ptr 192.168.1.1 | cut -d' ' -f 2
1.1.168.192.in-addr.arpa.
cuonglm
fonte
Isso funciona apenas para endereços que você pode procurar no seu DNS. Se você tentar um endereço desconhecido, receberá uma mensagem de erro (que inclui o endereço reverso, mas precisa de um pós-processamento ligeiramente diferente).
Alexander Remesch
1

Assumindo uma var contém o ip: ip=192.168.2.1. Se o valor precisar ser atribuído a uma nova variável, apenas inclua qualquer solução interna $()e atribua-a à var rr=$(...).

Algumas soluções são possíveis:

: printf 'arpa.in-addr.%s.' "$ip" | tac -s.
Mais simples A maioria das conchas : IFS=. eval 'set -- $ip'; echo "$4.$3.$2.$1.in-addr.arpa"
Algumas conchas : 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

As soluções Dig e Host funcionam com IPv6.

Isaac
fonte
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
Lukas Gottschall
fonte
0

Uma alternativa mais curta à resposta de mattbianco com menos ferramentas, mas o uso de pcregrep pode ser:

$ 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.
mback2k
fonte
0

Use o seguinte shell de uma linha:

echo '192.168.1.1' | tr '.' '\n' | tac | paste -s -d '.' -
Hongbo Liu
fonte