tracker.ts 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144
  1. import { Server as TrackerServer } from 'bittorrent-tracker'
  2. import express from 'express'
  3. import { createServer } from 'http'
  4. import proxyAddr from 'proxy-addr'
  5. import { WebSocketServer } from 'ws'
  6. import { Redis } from '@server/lib/redis'
  7. import { logger } from '../helpers/logger'
  8. import { CONFIG } from '../initializers/config'
  9. import { TRACKER_RATE_LIMITS } from '../initializers/constants'
  10. import { VideoFileModel } from '../models/video/video-file'
  11. import { VideoStreamingPlaylistModel } from '../models/video/video-streaming-playlist'
  12. const trackerRouter = express.Router()
  13. let peersIps = {}
  14. let peersIpInfoHash = {}
  15. runPeersChecker()
  16. const trackerServer = new TrackerServer({
  17. http: false,
  18. udp: false,
  19. ws: false,
  20. filter: async function (infoHash, params, cb) {
  21. if (CONFIG.TRACKER.ENABLED === false) {
  22. return cb(new Error('Tracker is disabled on this instance.'))
  23. }
  24. let ip: string
  25. if (params.type === 'ws') {
  26. ip = params.ip
  27. } else {
  28. ip = params.httpReq.ip
  29. }
  30. const key = ip + '-' + infoHash
  31. peersIps[ip] = peersIps[ip] ? peersIps[ip] + 1 : 1
  32. peersIpInfoHash[key] = peersIpInfoHash[key] ? peersIpInfoHash[key] + 1 : 1
  33. if (CONFIG.TRACKER.REJECT_TOO_MANY_ANNOUNCES && peersIpInfoHash[key] > TRACKER_RATE_LIMITS.ANNOUNCES_PER_IP_PER_INFOHASH) {
  34. return cb(new Error(`Too many requests (${peersIpInfoHash[key]} of ip ${ip} for torrent ${infoHash}`))
  35. }
  36. try {
  37. if (CONFIG.TRACKER.PRIVATE === false) return cb()
  38. const videoFileExists = await VideoFileModel.doesInfohashExistCached(infoHash)
  39. if (videoFileExists === true) return cb()
  40. const playlistExists = await VideoStreamingPlaylistModel.doesInfohashExistCached(infoHash)
  41. if (playlistExists === true) return cb()
  42. cb(new Error(`Unknown infoHash ${infoHash} requested by ip ${ip}`))
  43. // Close socket connection and block IP for a few time
  44. if (params.type === 'ws') {
  45. Redis.Instance.setTrackerBlockIP(ip)
  46. .catch(err => logger.error('Cannot set tracker block ip.', { err }))
  47. // setTimeout to wait filter response
  48. setTimeout(() => params.socket.close(), 0)
  49. }
  50. } catch (err) {
  51. logger.error('Error in tracker filter.', { err })
  52. return cb(err)
  53. }
  54. }
  55. })
  56. if (CONFIG.TRACKER.ENABLED !== false) {
  57. trackerServer.on('error', function (err) {
  58. logger.error('Error in tracker.', { err })
  59. })
  60. trackerServer.on('warning', function (err) {
  61. logger.warn('Warning in tracker.', { err })
  62. })
  63. }
  64. const onHttpRequest = trackerServer.onHttpRequest.bind(trackerServer)
  65. trackerRouter.get('/tracker/announce', (req, res) => onHttpRequest(req, res, { action: 'announce' }))
  66. trackerRouter.get('/tracker/scrape', (req, res) => onHttpRequest(req, res, { action: 'scrape' }))
  67. function createWebsocketTrackerServer (app: express.Application) {
  68. const server = createServer(app)
  69. const wss = new WebSocketServer({ noServer: true })
  70. wss.on('connection', function (ws, req) {
  71. ws['ip'] = proxyAddr(req, CONFIG.TRUST_PROXY)
  72. trackerServer.onWebSocketConnection(ws)
  73. })
  74. server.on('upgrade', (request: express.Request, socket, head) => {
  75. if (request.url === '/tracker/socket') {
  76. const ip = proxyAddr(request, CONFIG.TRUST_PROXY)
  77. Redis.Instance.doesTrackerBlockIPExist(ip)
  78. .then(result => {
  79. if (result === true) {
  80. logger.debug('Blocking IP %s from tracker.', ip)
  81. socket.write('HTTP/1.1 403 Forbidden\r\n\r\n')
  82. socket.destroy()
  83. return
  84. }
  85. // FIXME: typings
  86. return wss.handleUpgrade(request, socket as any, head, ws => wss.emit('connection', ws, request))
  87. })
  88. .catch(err => logger.error('Cannot check if tracker block ip exists.', { err }))
  89. }
  90. // Don't destroy socket, we have Socket.IO too
  91. })
  92. return server
  93. }
  94. // ---------------------------------------------------------------------------
  95. export {
  96. trackerRouter,
  97. createWebsocketTrackerServer
  98. }
  99. // ---------------------------------------------------------------------------
  100. function runPeersChecker () {
  101. setInterval(() => {
  102. logger.debug('Checking peers.')
  103. for (const ip of Object.keys(peersIpInfoHash)) {
  104. if (peersIps[ip] > TRACKER_RATE_LIMITS.ANNOUNCES_PER_IP) {
  105. logger.warn('Peer %s made abnormal requests (%d).', ip, peersIps[ip])
  106. }
  107. }
  108. peersIpInfoHash = {}
  109. peersIps = {}
  110. }, TRACKER_RATE_LIMITS.INTERVAL)
  111. }