亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > Python > 正文

使用Python編寫基于DHT協議的BT資源爬蟲

2020-01-04 17:35:51
字體:
來源:轉載
供稿:網友
這篇文章主要介紹了使用Python編寫基于DHT協議的BT資源爬蟲的方法,文中對于DHT協議的相關知識也作了補充說明,需要的朋友可以參考下
 

關于DHT協議

DHT協議作為BT協議的一個輔助,是非常好玩的。它主要是為了在BT正式下載時得到種子或者BT資源。傳統的網絡,需要一臺中央服務器存放種子或者BT資源,不僅浪費服務器資源,還容易出現單點的各種問題,而DHT網絡則是為了去中心化,也就是說任意時刻,這個網絡總有節點是亮的,你可以去詢問問這些亮的節點,從而將自己加入DHT網絡。

要實現DHT協議的網絡爬蟲,主要分3步,第一步是得到資源信息(infohash,160bit,20字節,可以編碼為40字節的十六進制字符串),第二步是確認這些infohash是有效的,第三步是通過有效的infohash下載到BT的種子文件,從而得到對這個資源的完整描述。

其中第一步是其他節點用DHT協議中的get_peers方法向爬蟲發送請求得到的,第二步是其他節點用DHT協議中的announce_peer向爬蟲發送請求得到的,第三步可以有幾種方式得到,比如可以去一些保存種子的網站根據infohash直接下載到,或者通過announce_peer的節點來下載到,具體如何實現,可以取決于你自己的爬蟲。

DHT協議中的主要幾個操作:

主要負責通過UDP與外部節點交互,封裝4種基本操作的請求以及相應。

ping:檢查一個節點是否“存活”

在一個爬蟲里主要有兩個地方用到ping,第一是初始路由表時,第二是驗證節點是否存活時

find_node:向一個節點發送查找節點的請求

在一個爬蟲中主要也是兩個地方用到find_node,第一是初始路由表時,第二是驗證桶是否存活時

get_peers:向一個節點發送查找資源的請求

在爬蟲中有節點向自己請求時不僅像個正常節點一樣做出回應,還需要以此資源的info_hash為機會盡可能多的去認識更多的節點。如圖,get_peers實際上最后一步是announce_peer,但是因為爬蟲不能announce_peer,所以實際上get_peers退化成了find_node操作。

Python,DHT協議,爬蟲

announce_peer:向一個節點發送自己已經開始下載某個資源的通知

爬蟲中不能用announce_peer,因為這就相當于通報虛假資源,對方很容易從上下文中判斷你是否通報了虛假資源從而把你禁掉。

基于Python的DHT爬蟲
修改自github開源爬蟲,原作者名字有些。。,這里直接將項目地址列出:https://github.com/Fuck-You-GFW/simDHT,有github帳號的請給原作者star,后續我將結果放入db,外加用tornado做一個簡單的查詢界面出來放在github上,先備份一下代碼

#!/usr/bin/env python# encoding: utf-8import socketfrom hashlib import sha1from random import randintfrom struct import unpackfrom socket import inet_ntoafrom threading import Timer, Threadfrom time import sleepfrom collections import dequefrom bencode import bencode, bdecodeBOOTSTRAP_NODES = (  ("router.bittorrent.com", 6881),  ("dht.transmissionbt.com", 6881),  ("router.utorrent.com", 6881))TID_LENGTH = 2RE_JOIN_DHT_INTERVAL = 3TOKEN_LENGTH = 2def entropy(length):  return "".join(chr(randint(0, 255)) for _ in xrange(length))def random_id():  h = sha1()  h.update(entropy(20))  return h.digest()def decode_nodes(nodes):  n = []  length = len(nodes)  if (length % 26) != 0:    return n  for i in range(0, length, 26):    nid = nodes[i:i+20]    ip = inet_ntoa(nodes[i+20:i+24])    port = unpack("!H", nodes[i+24:i+26])[0]    n.append((nid, ip, port))  return ndef timer(t, f):  Timer(t, f).start()def get_neighbor(target, nid, end=10):  return target[:end]+nid[end:]class KNode(object):  def __init__(self, nid, ip, port):    self.nid = nid    self.ip = ip    self.port = portclass DHTClient(Thread):  def __init__(self, max_node_qsize):    Thread.__init__(self)    self.setDaemon(True)    self.max_node_qsize = max_node_qsize    self.nid = random_id()    self.nodes = deque(maxlen=max_node_qsize)  def send_krpc(self, msg, address):    try:      self.ufd.sendto(bencode(msg), address)    except Exception:      pass  def send_find_node(self, address, nid=None):    nid = get_neighbor(nid, self.nid) if nid else self.nid    tid = entropy(TID_LENGTH)    msg = {      "t": tid,      "y": "q",      "q": "find_node",      "a": {        "id": nid,        "target": random_id()      }    }    self.send_krpc(msg, address)  def join_DHT(self):    for address in BOOTSTRAP_NODES:      self.send_find_node(address)  def re_join_DHT(self):    if len(self.nodes) == 0:      self.join_DHT()    timer(RE_JOIN_DHT_INTERVAL, self.re_join_DHT)  def auto_send_find_node(self):    wait = 1.0 / self.max_node_qsize    while True:      try:        node = self.nodes.popleft()        self.send_find_node((node.ip, node.port), node.nid)      except IndexError:        pass      sleep(wait)  def process_find_node_response(self, msg, address):    nodes = decode_nodes(msg["r"]["nodes"])    for node in nodes:      (nid, ip, port) = node      if len(nid) != 20: continue      if ip == self.bind_ip: continue      if port < 1 or port > 65535: continue      n = KNode(nid, ip, port)      self.nodes.append(n)class DHTServer(DHTClient):  def __init__(self, master, bind_ip, bind_port, max_node_qsize):    DHTClient.__init__(self, max_node_qsize)    self.master = master    self.bind_ip = bind_ip    self.bind_port = bind_port    self.process_request_actions = {      "get_peers": self.on_get_peers_request,      "announce_peer": self.on_announce_peer_request,    }    self.ufd = socket.socket(socket.AF_INET, socket.SOCK_DGRAM, socket.IPPROTO_UDP)    self.ufd.bind((self.bind_ip, self.bind_port))    timer(RE_JOIN_DHT_INTERVAL, self.re_join_DHT)  def run(self):    self.re_join_DHT()    while True:      try:        (data, address) = self.ufd.recvfrom(65536)        msg = bdecode(data)        self.on_message(msg, address)      except Exception:        pass  def on_message(self, msg, address):    try:      if msg["y"] == "r":        if msg["r"].has_key("nodes"):          self.process_find_node_response(msg, address)      elif msg["y"] == "q":        try:          self.process_request_actions[msg["q"]](msg, address)        except KeyError:          self.play_dead(msg, address)    except KeyError:      pass  def on_get_peers_request(self, msg, address):    try:      infohash = msg["a"]["info_hash"]      tid = msg["t"]      nid = msg["a"]["id"]      token = infohash[:TOKEN_LENGTH]      msg = {        "t": tid,        "y": "r",        "r": {          "id": get_neighbor(infohash, self.nid),          "nodes": "",          "token": token        }      }      self.send_krpc(msg, address)    except KeyError:      pass  def on_announce_peer_request(self, msg, address):    try:      infohash = msg["a"]["info_hash"]      #print msg["a"]      tname = msg["a"]["name"]      token = msg["a"]["token"]      nid = msg["a"]["id"]      tid = msg["t"]      if infohash[:TOKEN_LENGTH] == token:        if msg["a"].has_key("implied_port") and msg["a"]["implied_port"] != 0:          port = address[1]        else:          port = msg["a"]["port"]          if port < 1 or port > 65535: return        self.master.log(infohash, (address[0], port),tname)    except Exception:      pass    finally:      self.ok(msg, address)  def play_dead(self, msg, address):    try:      tid = msg["t"]      msg = {        "t": tid,        "y": "e",        "e": [202, "Server Error"]      }      self.send_krpc(msg, address)    except KeyError:      pass  def ok(self, msg, address):    try:      tid = msg["t"]      nid = msg["a"]["id"]      msg = {        "t": tid,        "y": "r",        "r": {          "id": get_neighbor(nid, self.nid)        }      }      self.send_krpc(msg, address)    except KeyError:      passclass Master(object):  def log(self, infohash,address=None,tname=None):    hexinfohash = infohash.encode("hex")    print "info_hash is: %s,name is: %s from %s:%s" % (      hexinfohash,tname, address[0], address[1]  )    print "magnet:?xt=urn:btih:%s&dn=%s" % (hexinfohash, tname)# using exampleif __name__ == "__main__":  # max_node_qsize bigger, bandwith bigger, speed higher  dht = DHTServer(Master(), "0.0.0.0", 6882, max_node_qsize=200)  dht.start()  dht.auto_send_find_node()

PS:  DHT協議中有幾個重點的需要澄清的地方:

1. node與infohash同樣使用160bit的表示方式,160bit意味著整個節點空間有2^160 = 730750818665451459101842416358141509827966271488,是48位10進制,也就是說有百億億億億億個節點空間,這么大的節點空間,是足夠存放你的主機節點以及任意的資源信息的。

2. 每個節點有張路由表。每張路由表由一堆K桶組成,所謂K桶,就是桶中最多只能放K個節點,默認是8個。而桶的保存則是類似一顆前綴樹的方式。相當于一張8桶的路由表中最多有160-4個K桶。

3. 根據DHT協議的規定,每個infohash都是有位置的,因此,兩個infohash之間就有距離一說,而兩個infohash的距離就可以用異或來表示,即infohash1 xor infohash2,也就是說,高位一樣的話,他們的距離就近,反之則遠,這樣可以快速的計算兩個節點的距離。計算這個距離有什么用呢,在DHT網絡中,如果一個資源的infohash與一個節點的infohash越近則該節點越有可能擁有該資源的信息,為什么呢?可以想象,因為人人都用同樣的距離算法去遞歸的詢問離資源接近的節點,并且只要該節點做出了回應,那么就會得到一個announce信息,也就是說跟資源infohash接近的節點就有更大的概率拿到該資源的infohash

4. 根據上述算法,DHT中的查詢是跳躍式查詢,可以迅速的跨越的的節點桶而接近目標節點桶。之所以在遠處能夠大幅度跳躍,而在近處只能小幅度跳躍,原因是每個節點的路由表中離自身越接近的節點保存得越多,如下圖

Python,DHT協議,爬蟲

5. 在一個DHT網絡中當爬蟲并不容易,不像普通爬蟲一樣,看到資源就可以主動爬下來,相反,因為得到資源的方式(get_peers, announce_peer)都是被動的,所以爬蟲的方式就有些變化了,爬蟲所要做的事就是像個正常節點一樣去響應其他節點的查詢,并且得到其他節點的回應,把其中的數據收集下來就算是完成工作了。而爬蟲唯一能做的,是盡可能的去多認識其他節點,這樣,才能有更多其他節點來向你詢問。

6. 有人說,那么我把DHT爬蟲的K桶中的容量K增大是不是就能增加得到資源的機會,其實不然,之前也分析過了,DHT爬蟲最重要的信息來源全是被動的,因為你不能增大別人的K,所以距離遠的節點保存你自身的概率就越小,當然距離遠的節點去請求你的概率相對也比較小。


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
亚洲免费电影一区| 色噜噜久久综合伊人一本| 久久99国产综合精品女同| 欧美激情视频播放| 国产成人一区二区| 欧美激情视频一区| 日韩在线播放一区| 亚洲一区二区三区sesese| 亚洲成人激情小说| 久久久久久久电影一区| 亚洲视频在线播放| 国产精品黄页免费高清在线观看| 国产精品视频99| 久久久免费av| 欧美激情精品久久久久久大尺度| 日韩有码在线观看| 最新中文字幕亚洲| 三级精品视频久久久久| 久久综合国产精品台湾中文娱乐网| 亚洲成人网av| 日韩一中文字幕| 狠狠色狠色综合曰曰| 欧美日韩亚洲一区二区| 欧美香蕉大胸在线视频观看| 国产精品视频精品| 国产成人97精品免费看片| 日韩有码在线电影| 欧美激情免费观看| 成人在线播放av| 欧美黑人巨大精品一区二区| 中文国产亚洲喷潮| 久久精品国产清自在天天线| 亚洲乱码一区av黑人高潮| 日韩欧美一区二区三区| 欧美黑人xxxⅹ高潮交| 8x拔播拔播x8国产精品| www.欧美免费| 国产精品影片在线观看| 亚洲аv电影天堂网| 久久精品国产成人精品| 欧美成人免费网| 亚洲最大的成人网| 亚洲天堂色网站| 一区二区三区视频免费| 亚洲精品网站在线播放gif| 夜夜嗨av色综合久久久综合网| 国产精品视频不卡| 91成人在线视频| 日韩欧美a级成人黄色| 亚洲精品电影网在线观看| 日韩电影第一页| 18久久久久久| 国产男人精品视频| 成人疯狂猛交xxx| 欧美精品videossex性护士| 亚洲高清免费观看高清完整版| 九九热在线精品视频| 亚洲精品影视在线观看| 久久久成人av| 成人啪啪免费看| 日日狠狠久久偷偷四色综合免费| 久久久女女女女999久久| 日本亚洲欧洲色| 国产欧美亚洲视频| 国产精品免费电影| 欧美理论电影在线观看| 97视频在线观看成人| 欧美福利视频网站| 国产精品成人免费视频| 国产精品入口福利| 亚洲国产第一页| 久久99热这里只有精品国产| 日日骚久久av| 红桃视频成人在线观看| 亚洲精品国产欧美| 91精品国产91久久久久久| 欧美激情a在线| 97高清免费视频| 欧美激情视频给我| 亚洲一区二区三区四区在线播放| 日韩有码视频在线| 中文综合在线观看| 欧美激情中文字幕乱码免费| 操91在线视频| 亚洲国产美女精品久久久久∴| 国产噜噜噜噜噜久久久久久久久| 奇米一区二区三区四区久久| 欧美日韩一区二区免费视频| 九色成人免费视频| 日韩欧美一区视频| 91综合免费在线| 亚洲精品国产美女| 日本精品一区二区三区在线| 日韩在线观看网址| 国产精品精品视频| 97人人做人人爱| 日韩精品免费视频| 亚洲无限av看| 欧美电影免费观看高清| 日韩经典中文字幕| www.日韩.com| 国产在线视频一区| 欧美精品videos性欧美| 亚洲丝袜在线视频| 久久久久久久999精品视频| 美女久久久久久久| 欧美午夜女人视频在线| 日韩亚洲一区二区| 欧美日韩亚洲视频一区| 精品亚洲一区二区三区在线观看| 亚洲福利精品在线| 久久精品99国产精品酒店日本| 国产98色在线| 欧美精品在线观看| 欧美在线性爱视频| 亚洲成**性毛茸茸| 91久久中文字幕| 久久99热精品| 亚洲自拍中文字幕| 亚洲欧美视频在线| 国产亚洲成av人片在线观看桃| 欧洲日本亚洲国产区| 成人在线视频福利| 久久成人亚洲精品| 97超碰国产精品女人人人爽| 亚洲一区二区三区乱码aⅴ| 欧美成人第一页| 日韩精品久久久久久福利| 97免费在线视频| 色综合五月天导航| 成人黄色av网站| 欧美日韩高清区| 视频一区视频二区国产精品| 国产亚洲欧洲高清一区| 亚洲精品福利资源站| 韩国日本不卡在线| 日韩精品高清在线| 国产成人亚洲综合91| 国产精品精品视频| 亚洲欧美精品伊人久久| 亚洲男人天堂九九视频| 日本韩国在线不卡| 欧美限制级电影在线观看| 欧美性xxxxxxxxx| 亚洲国产欧美自拍| 激情懂色av一区av二区av| 国产成人一区二区三区| 久久免费视频这里只有精品| 欧美在线观看网站| 美日韩在线视频| 久热国产精品视频| 久久综合亚洲社区| 日韩免费在线免费观看| 国产精品大陆在线观看| 97免费视频在线| 91国产中文字幕| 欧美国产在线视频| 欧美国产日韩一区二区| 色综合久久天天综线观看| 久久九九全国免费精品观看| 九色精品美女在线| 国产97在线亚洲| 亚洲一区二区久久| 国产精品video|