modernize client and index using lebab

- Uses class, const let, etc
- Removed self variable
- Creating a new Client without `new` keyword don’t work anymore
- Replaced inherits with class extends
This commit is contained in:
Jimmy Wärting 2018-10-03 12:09:02 +02:00
parent 65b2bdc804
commit 447423fc46
2 changed files with 204 additions and 220 deletions

268
client.js
View File

@ -1,21 +1,16 @@
module.exports = Client
const { Buffer } = require('safe-buffer')
const debug = require('debug')('bittorrent-tracker:client')
const EventEmitter = require('events')
const once = require('once')
const parallel = require('run-parallel')
const Peer = require('simple-peer')
const uniq = require('uniq')
const url = require('url')
var Buffer = require('safe-buffer').Buffer
var debug = require('debug')('bittorrent-tracker:client')
var EventEmitter = require('events').EventEmitter
var inherits = require('inherits')
var once = require('once')
var parallel = require('run-parallel')
var Peer = require('simple-peer')
var uniq = require('uniq')
var url = require('url')
var common = require('./lib/common')
var HTTPTracker = require('./lib/client/http-tracker') // empty object in browser
var UDPTracker = require('./lib/client/udp-tracker') // empty object in browser
var WebSocketTracker = require('./lib/client/websocket-tracker')
inherits(Client, EventEmitter)
const common = require('./lib/common')
const HTTPTracker = require('./lib/client/http-tracker') // empty object in browser
const UDPTracker = require('./lib/client/udp-tracker') // empty object in browser
const WebSocketTracker = require('./lib/client/websocket-tracker')
/**
* BitTorrent tracker client.
@ -32,48 +27,46 @@ inherits(Client, EventEmitter)
* @param {number} opts.userAgent User-Agent header for http requests
* @param {number} opts.wrtc custom webrtc impl (useful in node.js)
*/
function Client (opts) {
var self = this
if (!(self instanceof Client)) return new Client(opts)
EventEmitter.call(self)
if (!opts) opts = {}
class Client extends EventEmitter {
constructor (opts = {}) {
super()
if (!opts.peerId) throw new Error('Option `peerId` is required')
if (!opts.infoHash) throw new Error('Option `infoHash` is required')
if (!opts.announce) throw new Error('Option `announce` is required')
if (!process.browser && !opts.port) throw new Error('Option `port` is required')
self.peerId = typeof opts.peerId === 'string'
this.peerId = typeof opts.peerId === 'string'
? opts.peerId
: opts.peerId.toString('hex')
self._peerIdBuffer = Buffer.from(self.peerId, 'hex')
self._peerIdBinary = self._peerIdBuffer.toString('binary')
this._peerIdBuffer = Buffer.from(this.peerId, 'hex')
this._peerIdBinary = this._peerIdBuffer.toString('binary')
self.infoHash = typeof opts.infoHash === 'string'
this.infoHash = typeof opts.infoHash === 'string'
? opts.infoHash.toLowerCase()
: opts.infoHash.toString('hex')
self._infoHashBuffer = Buffer.from(self.infoHash, 'hex')
self._infoHashBinary = self._infoHashBuffer.toString('binary')
this._infoHashBuffer = Buffer.from(this.infoHash, 'hex')
this._infoHashBinary = this._infoHashBuffer.toString('binary')
debug('new client %s', self.infoHash)
debug('new client %s', this.infoHash)
self.destroyed = false
this.destroyed = false
self._port = opts.port
self._getAnnounceOpts = opts.getAnnounceOpts
self._rtcConfig = opts.rtcConfig
self._userAgent = opts.userAgent
this._port = opts.port
this._getAnnounceOpts = opts.getAnnounceOpts
this._rtcConfig = opts.rtcConfig
this._userAgent = opts.userAgent
// Support lazy 'wrtc' module initialization
// See: https://github.com/webtorrent/webtorrent-hybrid/issues/46
self._wrtc = typeof opts.wrtc === 'function' ? opts.wrtc() : opts.wrtc
this._wrtc = typeof opts.wrtc === 'function' ? opts.wrtc() : opts.wrtc
var announce = typeof opts.announce === 'string'
let announce = typeof opts.announce === 'string'
? [ opts.announce ]
: opts.announce == null ? [] : opts.announce
// Remove trailing slash from trackers to catch duplicates
announce = announce.map(function (announceUrl) {
announce = announce.map(announceUrl => {
announceUrl = announceUrl.toString()
if (announceUrl[announceUrl.length - 1] === '/') {
announceUrl = announceUrl.substring(0, announceUrl.length - 1)
@ -82,86 +75,36 @@ function Client (opts) {
})
announce = uniq(announce)
var webrtcSupport = self._wrtc !== false && (!!self._wrtc || Peer.WEBRTC_SUPPORT)
const webrtcSupport = this._wrtc !== false && (!!this._wrtc || Peer.WEBRTC_SUPPORT)
self._trackers = announce
.map(function (announceUrl) {
var protocol = url.parse(announceUrl).protocol
const nextTickWarn = err => {
process.nextTick(() => {
this.emit('warning', err)
})
}
this._trackers = announce
.map(announceUrl => {
const protocol = url.parse(announceUrl).protocol
if ((protocol === 'http:' || protocol === 'https:') &&
typeof HTTPTracker === 'function') {
return new HTTPTracker(self, announceUrl)
return new HTTPTracker(this, announceUrl)
} else if (protocol === 'udp:' && typeof UDPTracker === 'function') {
return new UDPTracker(self, announceUrl)
return new UDPTracker(this, announceUrl)
} else if ((protocol === 'ws:' || protocol === 'wss:') && webrtcSupport) {
// Skip ws:// trackers on https:// sites because they throw SecurityError
if (protocol === 'ws:' && typeof window !== 'undefined' &&
window.location.protocol === 'https:') {
nextTickWarn(new Error('Unsupported tracker protocol: ' + announceUrl))
nextTickWarn(new Error(`Unsupported tracker protocol: ${announceUrl}`))
return null
}
return new WebSocketTracker(self, announceUrl)
return new WebSocketTracker(this, announceUrl)
} else {
nextTickWarn(new Error('Unsupported tracker protocol: ' + announceUrl))
nextTickWarn(new Error(`Unsupported tracker protocol: ${announceUrl}`))
return null
}
})
.filter(Boolean)
function nextTickWarn (err) {
process.nextTick(function () {
self.emit('warning', err)
})
}
}
/**
* Simple convenience function to scrape a tracker for an info hash without needing to
* create a Client, pass it a parsed torrent, etc. Support scraping a tracker for multiple
* torrents at the same time.
* @params {Object} opts
* @param {string|Array.<string>} opts.infoHash
* @param {string} opts.announce
* @param {function} cb
*/
Client.scrape = function (opts, cb) {
cb = once(cb)
if (!opts.infoHash) throw new Error('Option `infoHash` is required')
if (!opts.announce) throw new Error('Option `announce` is required')
var clientOpts = Object.assign({}, opts, {
infoHash: Array.isArray(opts.infoHash) ? opts.infoHash[0] : opts.infoHash,
peerId: Buffer.from('01234567890123456789'), // dummy value
port: 6881 // dummy value
})
var client = new Client(clientOpts)
client.once('error', cb)
client.once('warning', cb)
var len = Array.isArray(opts.infoHash) ? opts.infoHash.length : 1
var results = {}
client.on('scrape', function (data) {
len -= 1
results[data.infoHash] = data
if (len === 0) {
client.destroy()
var keys = Object.keys(results)
if (keys.length === 1) {
cb(null, results[keys[0]])
} else {
cb(null, results)
}
}
})
opts.infoHash = Array.isArray(opts.infoHash)
? opts.infoHash.map(function (infoHash) {
return Buffer.from(infoHash, 'hex')
})
: Buffer.from(opts.infoHash, 'hex')
client.scrape({ infoHash: opts.infoHash })
return client
}
/**
@ -171,15 +114,14 @@ Client.scrape = function (opts, cb) {
* @param {number=} opts.downloaded
* @param {number=} opts.left (if not set, calculated automatically)
*/
Client.prototype.start = function (opts) {
var self = this
start (opts) {
debug('send `start`')
opts = self._defaultAnnounceOpts(opts)
opts = this._defaultAnnounceOpts(opts)
opts.event = 'started'
self._announce(opts)
this._announce(opts)
// start announcing on intervals
self._trackers.forEach(function (tracker) {
this._trackers.forEach(tracker => {
tracker.setInterval()
})
}
@ -192,12 +134,11 @@ Client.prototype.start = function (opts) {
* @param {number=} opts.numwant
* @param {number=} opts.left (if not set, calculated automatically)
*/
Client.prototype.stop = function (opts) {
var self = this
stop (opts) {
debug('send `stop`')
opts = self._defaultAnnounceOpts(opts)
opts = this._defaultAnnounceOpts(opts)
opts.event = 'stopped'
self._announce(opts)
this._announce(opts)
}
/**
@ -208,13 +149,12 @@ Client.prototype.stop = function (opts) {
* @param {number=} opts.numwant
* @param {number=} opts.left (if not set, calculated automatically)
*/
Client.prototype.complete = function (opts) {
var self = this
complete (opts) {
debug('send `complete`')
if (!opts) opts = {}
opts = self._defaultAnnounceOpts(opts)
opts = this._defaultAnnounceOpts(opts)
opts.event = 'completed'
self._announce(opts)
this._announce(opts)
}
/**
@ -225,17 +165,15 @@ Client.prototype.complete = function (opts) {
* @param {number=} opts.numwant
* @param {number=} opts.left (if not set, calculated automatically)
*/
Client.prototype.update = function (opts) {
var self = this
update (opts) {
debug('send `update`')
opts = self._defaultAnnounceOpts(opts)
opts = this._defaultAnnounceOpts(opts)
if (opts.event) delete opts.event
self._announce(opts)
this._announce(opts)
}
Client.prototype._announce = function (opts) {
var self = this
self._trackers.forEach(function (tracker) {
_announce (opts) {
this._trackers.forEach(tracker => {
// tracker should not modify `opts` object, it's passed to all trackers
tracker.announce(opts)
})
@ -245,51 +183,97 @@ Client.prototype._announce = function (opts) {
* Send a scrape request to the trackers.
* @param {Object} opts
*/
Client.prototype.scrape = function (opts) {
var self = this
scrape (opts) {
debug('send `scrape`')
if (!opts) opts = {}
self._trackers.forEach(function (tracker) {
this._trackers.forEach(tracker => {
// tracker should not modify `opts` object, it's passed to all trackers
tracker.scrape(opts)
})
}
Client.prototype.setInterval = function (intervalMs) {
var self = this
setInterval (intervalMs) {
debug('setInterval %d', intervalMs)
self._trackers.forEach(function (tracker) {
this._trackers.forEach(tracker => {
tracker.setInterval(intervalMs)
})
}
Client.prototype.destroy = function (cb) {
var self = this
if (self.destroyed) return
self.destroyed = true
destroy (cb) {
if (this.destroyed) return
this.destroyed = true
debug('destroy')
var tasks = self._trackers.map(function (tracker) {
return function (cb) {
const tasks = this._trackers.map(tracker => cb => {
tracker.destroy(cb)
}
})
parallel(tasks, cb)
self._trackers = []
self._getAnnounceOpts = null
this._trackers = []
this._getAnnounceOpts = null
}
Client.prototype._defaultAnnounceOpts = function (opts) {
var self = this
if (!opts) opts = {}
_defaultAnnounceOpts (opts = {}) {
if (opts.numwant == null) opts.numwant = common.DEFAULT_ANNOUNCE_PEERS
if (opts.uploaded == null) opts.uploaded = 0
if (opts.downloaded == null) opts.downloaded = 0
if (self._getAnnounceOpts) opts = Object.assign({}, opts, self._getAnnounceOpts())
if (this._getAnnounceOpts) opts = Object.assign({}, opts, this._getAnnounceOpts())
return opts
}
}
/**
* Simple convenience function to scrape a tracker for an info hash without needing to
* create a Client, pass it a parsed torrent, etc. Support scraping a tracker for multiple
* torrents at the same time.
* @params {Object} opts
* @param {string|Array.<string>} opts.infoHash
* @param {string} opts.announce
* @param {function} cb
*/
Client.scrape = (opts, cb) => {
cb = once(cb)
if (!opts.infoHash) throw new Error('Option `infoHash` is required')
if (!opts.announce) throw new Error('Option `announce` is required')
const clientOpts = Object.assign({}, opts, {
infoHash: Array.isArray(opts.infoHash) ? opts.infoHash[0] : opts.infoHash,
peerId: Buffer.from('01234567890123456789'), // dummy value
port: 6881 // dummy value
})
const client = new Client(clientOpts)
client.once('error', cb)
client.once('warning', cb)
let len = Array.isArray(opts.infoHash) ? opts.infoHash.length : 1
const results = {}
client.on('scrape', data => {
len -= 1
results[data.infoHash] = data
if (len === 0) {
client.destroy()
const keys = Object.keys(results)
if (keys.length === 1) {
cb(null, results[keys[0]])
} else {
cb(null, results)
}
}
})
opts.infoHash = Array.isArray(opts.infoHash)
? opts.infoHash.map(infoHash => {
return Buffer.from(infoHash, 'hex')
})
: Buffer.from(opts.infoHash, 'hex')
client.scrape({ infoHash: opts.infoHash })
return client
}
module.exports = Client

View File

@ -1,5 +1,5 @@
var Client = require('./client')
var Server = require('./server')
const Client = require('./client')
const Server = require('./server')
module.exports = Client
module.exports.Client = Client