亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

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

 更新時間:2016年03月19日 11:51:33   作者:TAN9LE  
這篇文章主要介紹了使用Python編寫基于DHT協(xié)議的BT資源爬蟲的方法,文中對于DHT協(xié)議的相關知識也作了補充說明,需要的朋友可以參考下

關于DHT協(xié)議

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

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

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

DHT協(xié)議中的主要幾個操作:

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

ping:檢查一個節(jié)點是否“存活”

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

find_node:向一個節(jié)點發(fā)送查找節(jié)點的請求

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

get_peers:向一個節(jié)點發(fā)送查找資源的請求

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

2016319114959666.png (204×120)

announce_peer:向一個節(jié)點發(fā)送自己已經(jīng)開始下載某個資源的通知

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

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

#!/usr/bin/env python
# encoding: utf-8

import socket
from hashlib import sha1
from random import randint
from struct import unpack
from socket import inet_ntoa
from threading import Timer, Thread
from time import sleep
from collections import deque

from bencode import bencode, bdecode

BOOTSTRAP_NODES = (
  ("router.bittorrent.com", 6881),
  ("dht.transmissionbt.com", 6881),
  ("router.utorrent.com", 6881)
)
TID_LENGTH = 2
RE_JOIN_DHT_INTERVAL = 3
TOKEN_LENGTH = 2


def 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 n


def 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 = port


class 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:
      pass


class 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 example
if __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協(xié)議中有幾個重點的需要澄清的地方:

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

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

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

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

2016319115044824.jpg (490×417)

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

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

相關文章

  • 跟老齊學Python之賦值,簡單也不簡單

    跟老齊學Python之賦值,簡單也不簡單

    在《初識永遠強大的函數(shù)》一文中,有一節(jié)專門討論“取名字的學問”,就是有關變量名稱的問題,本溫故而知新的原則,這里要復習一下
    2014-09-09
  • python讀取文件列表并排序的實現(xiàn)示例

    python讀取文件列表并排序的實現(xiàn)示例

    本文主要介紹了python讀取文件列表并排序的實現(xiàn)示例,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2022-07-07
  • django 2.2和mysql使用的常見問題

    django 2.2和mysql使用的常見問題

    這篇文章主要介紹了django 2.2和mysql使用的常見問題,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
    2019-07-07
  • 利用Python將社交網(wǎng)絡進行可視化

    利用Python將社交網(wǎng)絡進行可視化

    這篇文章介紹了利用Python將社交網(wǎng)絡進行可視化,主要是一些Python的第三方庫來進行社交網(wǎng)絡的可視化,利用領英(Linkedin)的社交關系數(shù)據(jù)展開介紹,內(nèi)容可當學習練習題有一定的參考價值,需要的小伙伴可以參考一下
    2022-06-06
  • Python redis操作實例分析【連接、管道、發(fā)布和訂閱等】

    Python redis操作實例分析【連接、管道、發(fā)布和訂閱等】

    這篇文章主要介紹了Python redis操作,結合實例形式分析了Python redis的連接、管道、發(fā)布和訂閱等相關概念、原理及操作技巧,需要的朋友可以參考下
    2019-05-05
  • Python的Socket編程過程中實現(xiàn)UDP端口復用的實例分享

    Python的Socket編程過程中實現(xiàn)UDP端口復用的實例分享

    這篇文章主要介紹了Python的Socket編程過程中實現(xiàn)UDP端口復用的實例分享,文中作者用到了Python的twisted異步框架,需要的朋友可以參考下
    2016-03-03
  • Django REST framework 限流功能的使用

    Django REST framework 限流功能的使用

    DRF常用功能的案例基本用法都有講解,關于限流(Throttling)這個功能其實在真實的業(yè)務場景中能真正用到的其實不算多。今天說這個話題其實一方面是討論功能,另一方面也是希望換個角度去審視我們的開發(fā)過程,希望大家可以在使用DRF功能的同時,也了解一下功能背后的實現(xiàn)
    2021-06-06
  • Python matplotlib繪制xkcd動漫風格的圖表

    Python matplotlib繪制xkcd動漫風格的圖表

    xkcd是蘭道爾·門羅(Randall Munroe)的網(wǎng)名,又是他所創(chuàng)作的漫畫的名稱。本文將用matplotlib庫繪制xkcd動漫風格的圖表,感興趣的可以了解一下
    2022-03-03
  • jupyter notebook 多環(huán)境conda kernel配置方式

    jupyter notebook 多環(huán)境conda kernel配置方式

    這篇文章主要介紹了jupyter notebook 多環(huán)境conda kernel配置方式,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-04-04
  • Flask利用自定義接口實現(xiàn)mock應用詳解

    Flask利用自定義接口實現(xiàn)mock應用詳解

    后端接口已提供,前端需要依賴后端接口返回的數(shù)據(jù)進行前端頁面的開發(fā),如何配合前端?這篇就來介紹一下Flask如何利用自定義接口實現(xiàn)mock應用,需要的可以參考一下
    2023-03-03

最新評論