web-dev-qa-db-ja.com

Python 3:CIDR表記から可能なIPアドレスのリストを作成する

python(3.1)で、CIDR表記を取り、可能なIPアドレスのリストを返す関数を作成するタスクを渡されました。python.orgを調べて、これを見つけました。 http://docs.python.org/dev/py3k/library/ipaddr.html

しかし、私はこのニーズを満たすものを見たことがありません...誰かが私の道を蹴るのを気にかけているどんな援助にも非常に感謝します。前もって感謝します。 :-)

13
MadSc13ntist

組み込みモジュールの使用に慣れていない場合は、 netaddr というプロジェクトがあります。これは、IPネットワークでの作業に使用した中で最高のモジュールです。

IPチュートリアル をご覧ください。これは、ネットワークでの作業とIPの識別がいかに簡単であるかを示しています。簡単な例:

>>> from netaddr import IPNetwork
>>> for ip in IPNetwork('192.0.2.0/23'):
...    print '%s' % ip
...
192.0.2.0
192.0.2.1
192.0.2.2
192.0.2.3
...
192.0.3.252
192.0.3.253
192.0.3.254
192.0.3.255
35
jathanism

Python 3と同じくらい簡単

>>> import ipaddress
>>> [str(ip) for ip in ipaddress.IPv4Network('192.0.2.0/28')]
['192.0.2.0', '192.0.2.1', '192.0.2.2',
'192.0.2.3', '192.0.2.4', '192.0.2.5',
'192.0.2.6', '192.0.2.7', '192.0.2.8',
'192.0.2.9', '192.0.2.10', '192.0.2.11',
'192.0.2.12', '192.0.2.13', '192.0.2.14',
'192.0.2.15']
21
Petr Javorik

私は外部モジュールをインストールするよりも少し数学をしたいのですが、誰も私と同じ趣味を持っていませんか?

#!/usr/bin/env python
# python cidr.py 192.168.1.1/24

import sys, struct, socket

(ip, cidr) = sys.argv[1].split('/')
cidr = int(cidr) 
Host_bits = 32 - cidr
i = struct.unpack('>I', socket.inet_aton(ip))[0] # note the endianness
start = (i >> Host_bits) << Host_bits # clear the Host bits
end = start | ((1 << Host_bits) - 1)

# excludes the first and last address in the subnet
for i in range(start, end):
    print(socket.inet_ntoa(struct.pack('>I',i)))
9
jfly

チェックアウトしましたか iptools ?かなりフィットしているようです。

1
Hank Gay

ドキュメントにはありませんが、ソースを参照すると、ipaddr__iter__iterhostsを実装していることがわかります。これはまさにあなたが望むものです。


エラー、気にしないでください。

  1. ipaddr.pyが3.1ベータでstdlibに追加されたようですが、3.1rcによって削除されました。
  2. 私はオリジナルの ipaddr.py からのソースを見ていました。これは、python.orgのコピーとは別に進化したようです。

後者をバンドルすることもできます。

0
ephemient

以下のコードは、IPとサブネットを提供する際にIPの範囲を生成します。 (255.255.255.0)のようにCIDR表記を展開します

from netaddr import *

def getFirstIp(ipAddress,subnet):
  ipBin = IPNetwork(ipAddress).ip.bits().split('.')
  subBin = IPNetwork(subnet).ip.bits().split('.')
  zipped = Zip(ipBin,subBin)
  netIdList = []
  for octets in zipped:
    netIdList.append(''.join(str(b) for b in (map((lambda x: int(x[0])*int(x[1])),Zip(list(octets[0]),list(octets[1]))))))
  firstIp = ''
  firstIp = '.'.join(str(int(oct,2)) for oct in netIdList)
  return firstIp


def getLastIp(ipAddress,subnet):
  ipBin = IPNetwork(ipAddress).ip.bits().split('.')
  subBin = IPNetwork(subnet).ip.bits().split('.')
  #print ipBin
  #print subBin
  revsubBin = []
  for octets in subBin:
    revB = ''.join('1' if(b == '0') else '0' for b in octets)
    revsubBin.append(revB)
  zipped = Zip(ipBin,revsubBin)
  netIdList = []
  for octets in zipped:
    netIdList.append(''.join(str(b) for b in (map((lambda x: 0 if(int(x[0]) == 0 and int(x[1]) == 0) else 1),Zip(list(octets[0]),list(octets[1]))))))
  #print netIdList
  lastIp = ''
  lastIp = '.'.join(str(int(oct,2)) for oct in netIdList)
  return lastIp

def getRangeOfIps(firstIp,lastIp):
  start= int(IPAddress(firstIp))
  end = int(IPAddress(lastIp))
  ipList = []
  for ip in range(start,end+1):
    ipList.append(str(IPAddress(ip)))
  return ipList

def manipulateIP():
 firstIp = getFirstIp(ipAddress,subnet)
 lastIp = getLastIp(ipAddress,subnet)
 ipList = getRangeOfIps(firstIp,lastIp)  
 print ipList 
0
shyamtg

CIDRを指定してすべてのパブリックIPアドレスを生成する

https://github.com/stephenlb/geo-ip は、地域を含む有効なIPパブリックアドレスのリストを生成します。

'1.0.0.0/8'から'191.0.0.0/8'は、予約済みのプライベートIPアドレスを除く有効なパブリックIPアドレスの範囲です。

IPジェネレーター

IPアドレスと関連する地理情報のJSONダンプを生成します。有効なパブリックIPアドレスの範囲は'1.0.0.0/8'から'191.0.0.0/8'であることに注意してください。ただし、このreadmeの下部に示されている予約済みのプライベートIPアドレスの範囲は除きます。

docker build -t geo-ip .
docker run -e IPRANGE='54.0.0.0/30' geo-ip               ## a few IPs
docker run -e IPRANGE='54.0.0.0/26' geo-ip               ## a few more IPs
docker run -e IPRANGE='54.0.0.0/16' geo-ip               ## a lot more IPs
docker run -e IPRANGE='0.0.0.0/0'   geo-ip               ## ALL IPs ( slooooowwwwww )
docker run -e IPRANGE='0.0.0.0/0'   geo-ip > geo-ip.json ## ALL IPs saved to JSON File
docker run geo-ip 

すべての有効なパブリックアドレスをスキャンするためのもう少し高速なオプション:

for i in $(seq 1 191); do \
    docker run -e IPRANGE="$i.0.0.0/8" geo-ip; \
    sleep 1; \ 
done

これにより、4,228,250,625未満のJSON行がSTDOUTに出力されます。次の行の例を次に示します。

{"city": "Palo Alto", "ip": "0.0.0.0", "longitude": -122.1274,
 "continent": "North America", "continent_code": "NA",
 "state": "California", "country": "United States", "latitude": 37.418,
 "iso_code": "US", "state_code": "CA", "aso": "PubNub",
 "asn": "11404", "Zip_code": "94107"}

プライベートおよび予約済みIP範囲

上記のリポジトリのdockerfileは、ウィキペディアの記事のガイドに従って使用できないIPアドレスを除外します。 https://en.wikipedia.org/wiki/Reserved_IP_addresses

MaxMind Geo IP

Dockerfileは、 https://www.maxmind.com/en/home によって提供される無料の公開データベースをインポートします。

0
Stephen Blum