⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 rerequester.py

📁 bittorrent source by python. please enjoy
💻 PY
📖 第 1 页 / 共 2 页
字号:
# The contents of this file are subject to the BitTorrent Open Source License# Version 1.1 (the License).  You may not copy or use this file, in either# source code or executable form, except in compliance with the License.  You# may obtain a copy of the License at http://www.bittorrent.com/license/.## Software distributed under the License is distributed on an AS IS basis,# WITHOUT WARRANTY OF ANY KIND, either express or implied.  See the License# for the specific language governing rights and limitations under the# License.# Written by Bram Cohen, Uoti Urpalaimport sysimport randomimport structimport socketimport loggingfrom binascii import b2a_hexfrom BitTorrent.translation import _import BitTorrent.stackthreading as threadingfrom BitTorrent import versionfrom BitTorrent.platform import bttimefrom BitTorrent.zurllib import urlopen, quote, Request, URLErrorfrom BitTorrent.btformats import check_peersfrom BitTorrent.bencode import bencode, bdecodefrom BitTorrent.defer import ThreadedDeferredfrom BitTorrent.yielddefer import _wrap_taskfrom BitTorrent import BTFailurefrom BitTorrent.prefs import Preferences# TEMPimport threadclass Rerequester(object):    STATES = ['started', 'completed', 'stopped']    def __init__(self, url, announce_list, config, sched, externalsched, rawserver,                 howmany, connect,                 amount_left, up, down, port, myid, infohash, errorfunc, doneflag,                 upratefunc, downratefunc, ever_got_incoming, diefunc, sfunc):        """         @param url:       tracker's announce URL.         @param announce_list: ?         @param config:    preferences obj storing BitTorrent-wide                           configuration.         @param sched:     used to schedule events from inside rawserver's                           thread.  (Oh boy.  externalsched and sched.                           We expect Rerequester to                           recognize the difference between rawserver's                           thread and yet we go through the trouble of                           abstracting away rawserver using a callback...                           So what was the point?  --Dave)         @param externalsched: see sched.  This one is called from outside                           rawserver's thread.         @param howmany:   callback to get the number of complete connections.         @param connect:   callback to establish a connection to a peer                           obtained from the tracker.         @param amount_left: callback to obtain the number of bytes left to                           download for this torrent.         @param up:        callback to obtain the total number of bytes sent                           for this torrent.         @param down:      callback to obtain the total number of bytes                           received for this torrent.         @param port:      port to report to the tracker.  If the local peer                           is behind a NAT then this is the local peer's port                           on the NAT facing the outside world.         @param myid:      local peer's unique (self-generated) id.         @param infohash:  hash of the info section of the metainfo file.         @param errorfunc: callback to report errors.         @param doneflag:  when set all threads cleanup and then terminate.         @param upratefunc: callback to obtain moving average rate on the                           uplink for this torrent.         @param downratefunc: callback to obtain moving average rate on the                           downlink for this torrent.         @param ever_got_incoming: callback to determine if this torrent                           has ever received any mesages from other peers.         @param diefunc:   callback that is called when announce fails to find                           any peers.         @param sfunc:     success function?  With regard to what?  --Dave        """        assert isinstance(url, str)        assert isinstance(config, Preferences)        assert type(port) in (int,long) and port > 0 and port < 65536, "Port: %s" % repr(port)        assert callable(connect)        assert callable(externalsched)        assert callable(amount_left)        assert callable(errorfunc)        assert isinstance(doneflag, threading._Event)        assert callable(upratefunc)        assert callable(downratefunc)        assert callable(ever_got_incoming)        assert callable(diefunc)        assert callable(sfunc)        self.rawserver = rawserver        self.dead = False        self.baseurl = url        self.announce_list = None        if announce_list:            # shuffle a new copy of the whole set only once            shuffled_announce_list = []            for tier in announce_list:                if not tier:                    # strip blank lists                    continue                shuffled_tier = list(tier)                random.shuffle(shuffled_tier)                shuffled_announce_list.append(shuffled_tier)            if shuffled_announce_list:                self.announce_list = shuffled_announce_list                self.tier = 0                self.announce_i = 0                self.baseurl = self.announce_list_next()        self.announce_infohash = infohash        self.peerid = None        self.wanted_peerid = myid        self.port = port        self.url = None        self.config = config        self.last = None        self.trackerid = None        self.announce_interval = 30 * 60        self.sched = sched        self.howmany = howmany        self.connect = connect        self.externalsched = externalsched        self.amount_left = amount_left        self.up = up        self.down = down        self.errorfunc = errorfunc        self.doneflag = doneflag        self.upratefunc = upratefunc        self.downratefunc = downratefunc        self.ever_got_incoming = ever_got_incoming        self.diefunc = diefunc        self.successfunc = sfunc        self.finish = False        self.running_df = None        self.current_started = None        self.fail_wait = None        self.last_time = bttime()        self.previous_down = 0        self.previous_up = 0        self.tracker_num_peers = None        self.tracker_num_seeds = None    def _makeurl(self, peerid, port):        return ('%s?info_hash=%s&peer_id=%s&port=%s&key=%s' %                (self.baseurl, quote(self.announce_infohash), quote(peerid), str(port),                 b2a_hex(''.join([chr(random.randrange(256)) for i in xrange(4)]))))    def change_port(self, peerid, port):        assert thread.get_ident() == self.rawserver.ident        self.wanted_peerid = peerid        self.port = port        self.last = None        self.trackerid = None        self._check()    def begin(self):        if self.sched:            self.sched(10, self.begin)            self._check()    def announce_list_success(self):        tmp = self.announce_list[self.tier].pop(self.announce_i)        self.announce_list[self.tier].insert(0, tmp)        self.tier = 0        self.announce_i = 0    def announce_list_fail(self):        """returns True if the announce-list was restarted"""        self.announce_i += 1        if self.announce_i == len(self.announce_list[self.tier]):            self.announce_i = 0            self.tier += 1            if self.tier == len(self.announce_list):                self.tier = 0                return True        return False    def announce_list_next(self):        return self.announce_list[self.tier][self.announce_i]    def announce_finish(self):        if self.dead:            return        self.finish = True        self._check()    def announce_stop(self):        if self.dead:            return        self._announce('stopped')    def _check(self):        assert thread.get_ident() == self.rawserver.ident        assert not self.dead        #self.errorfunc(logging.INFO, 'check: ' + str(self.current_started))        if self.current_started is not None:            if self.current_started <= bttime() - 58:                self.errorfunc(logging.WARNING,                               _("Tracker announce still not complete "                                 "%d seconds after starting it") %                               int(bttime() - self.current_started))            return        if self.peerid is None:            self.peerid = self.wanted_peerid            self.url = self._makeurl(self.peerid, self.port)            self._announce('started')            return        if self.peerid != self.wanted_peerid:            # _announce will clean up these            up = self.up            down = self.down            self._announce('stopped')            self.peerid = None            self.previous_up = up()            self.previous_down = down()            return        if self.finish:            self.finish = False            self._announce('completed')            return        if self.fail_wait is not None:            if self.last_time + self.fail_wait <= bttime():                self._announce()            return        if self.last_time > bttime() - self.config['rerequest_interval']:            return        if self.ever_got_incoming():            getmore = self.howmany() <= self.config['min_peers'] / 3        else:            getmore = self.howmany() < self.config['min_peers']        if getmore or bttime() - self.last_time > self.announce_interval:            self._announce()    def get_next_announce_time_est(self):        # I'm sure this is wrong, but _check is confusing        return bttime() - (self.last_time + self.announce_interval)

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -