2021-08-27 14:32:44 +02:00
|
|
|
import { Server as TrackerServer } from 'bittorrent-tracker'
|
|
|
|
import express from 'express'
|
|
|
|
import { createServer } from 'http'
|
2023-05-22 17:04:39 +02:00
|
|
|
import { LRUCache } from 'lru-cache'
|
2021-08-27 14:32:44 +02:00
|
|
|
import proxyAddr from 'proxy-addr'
|
2021-08-27 16:42:17 +02:00
|
|
|
import { WebSocketServer } from 'ws'
|
2020-06-25 16:27:35 +02:00
|
|
|
import { logger } from '../helpers/logger'
|
|
|
|
import { CONFIG } from '../initializers/config'
|
2022-12-23 13:38:28 +01:00
|
|
|
import { LRU_CACHE, TRACKER_RATE_LIMITS } from '../initializers/constants'
|
2018-08-14 11:00:03 +02:00
|
|
|
import { VideoFileModel } from '../models/video/video-file'
|
2019-01-29 08:37:25 +01:00
|
|
|
import { VideoStreamingPlaylistModel } from '../models/video/video-streaming-playlist'
|
2018-06-26 16:53:24 +02:00
|
|
|
|
|
|
|
const trackerRouter = express.Router()
|
|
|
|
|
2022-12-23 13:38:28 +01:00
|
|
|
const blockedIPs = new LRUCache<string, boolean>({
|
|
|
|
max: LRU_CACHE.TRACKER_IPS.MAX_SIZE,
|
|
|
|
ttl: TRACKER_RATE_LIMITS.BLOCK_IP_LIFETIME
|
|
|
|
})
|
|
|
|
|
2018-06-26 16:53:24 +02:00
|
|
|
let peersIps = {}
|
|
|
|
let peersIpInfoHash = {}
|
|
|
|
runPeersChecker()
|
|
|
|
|
|
|
|
const trackerServer = new TrackerServer({
|
|
|
|
http: false,
|
|
|
|
udp: false,
|
|
|
|
ws: false,
|
2019-01-29 08:37:25 +01:00
|
|
|
filter: async function (infoHash, params, cb) {
|
2019-04-10 09:23:18 +02:00
|
|
|
if (CONFIG.TRACKER.ENABLED === false) {
|
|
|
|
return cb(new Error('Tracker is disabled on this instance.'))
|
|
|
|
}
|
|
|
|
|
2018-06-26 16:53:24 +02:00
|
|
|
let ip: string
|
|
|
|
|
|
|
|
if (params.type === 'ws') {
|
2021-01-11 15:45:04 +01:00
|
|
|
ip = params.ip
|
2018-06-26 16:53:24 +02:00
|
|
|
} else {
|
|
|
|
ip = params.httpReq.ip
|
|
|
|
}
|
|
|
|
|
|
|
|
const key = ip + '-' + infoHash
|
|
|
|
|
2020-01-31 16:56:52 +01:00
|
|
|
peersIps[ip] = peersIps[ip] ? peersIps[ip] + 1 : 1
|
|
|
|
peersIpInfoHash[key] = peersIpInfoHash[key] ? peersIpInfoHash[key] + 1 : 1
|
2018-06-26 16:53:24 +02:00
|
|
|
|
2020-01-31 16:56:52 +01:00
|
|
|
if (CONFIG.TRACKER.REJECT_TOO_MANY_ANNOUNCES && peersIpInfoHash[key] > TRACKER_RATE_LIMITS.ANNOUNCES_PER_IP_PER_INFOHASH) {
|
|
|
|
return cb(new Error(`Too many requests (${peersIpInfoHash[key]} of ip ${ip} for torrent ${infoHash}`))
|
2018-06-26 16:53:24 +02:00
|
|
|
}
|
|
|
|
|
2019-01-29 08:37:25 +01:00
|
|
|
try {
|
2019-04-10 09:23:18 +02:00
|
|
|
if (CONFIG.TRACKER.PRIVATE === false) return cb()
|
|
|
|
|
2020-01-03 13:47:45 +01:00
|
|
|
const videoFileExists = await VideoFileModel.doesInfohashExistCached(infoHash)
|
2019-01-29 08:37:25 +01:00
|
|
|
if (videoFileExists === true) return cb()
|
2018-08-14 11:00:03 +02:00
|
|
|
|
2020-10-29 15:03:31 +01:00
|
|
|
const playlistExists = await VideoStreamingPlaylistModel.doesInfohashExistCached(infoHash)
|
2019-01-29 08:37:25 +01:00
|
|
|
if (playlistExists === true) return cb()
|
|
|
|
|
2020-06-25 16:27:35 +02:00
|
|
|
cb(new Error(`Unknown infoHash ${infoHash} requested by ip ${ip}`))
|
|
|
|
|
|
|
|
// Close socket connection and block IP for a few time
|
|
|
|
if (params.type === 'ws') {
|
2022-12-23 13:38:28 +01:00
|
|
|
blockedIPs.set(ip, true)
|
2020-06-25 16:27:35 +02:00
|
|
|
|
|
|
|
// setTimeout to wait filter response
|
|
|
|
setTimeout(() => params.socket.close(), 0)
|
|
|
|
}
|
2019-01-29 08:37:25 +01:00
|
|
|
} catch (err) {
|
|
|
|
logger.error('Error in tracker filter.', { err })
|
|
|
|
return cb(err)
|
|
|
|
}
|
2018-06-26 16:53:24 +02:00
|
|
|
}
|
|
|
|
})
|
|
|
|
|
2019-04-10 09:23:18 +02:00
|
|
|
if (CONFIG.TRACKER.ENABLED !== false) {
|
|
|
|
trackerServer.on('error', function (err) {
|
|
|
|
logger.error('Error in tracker.', { err })
|
|
|
|
})
|
|
|
|
|
|
|
|
trackerServer.on('warning', function (err) {
|
2022-07-11 14:29:57 +02:00
|
|
|
const message = err.message || ''
|
|
|
|
|
|
|
|
if (CONFIG.LOG.LOG_TRACKER_UNKNOWN_INFOHASH === false && message.includes('Unknown infoHash')) {
|
|
|
|
return
|
2022-06-30 09:13:11 +02:00
|
|
|
}
|
|
|
|
|
2019-04-10 09:23:18 +02:00
|
|
|
logger.warn('Warning in tracker.', { err })
|
|
|
|
})
|
|
|
|
}
|
2018-06-26 16:53:24 +02:00
|
|
|
|
|
|
|
const onHttpRequest = trackerServer.onHttpRequest.bind(trackerServer)
|
|
|
|
trackerRouter.get('/tracker/announce', (req, res) => onHttpRequest(req, res, { action: 'announce' }))
|
|
|
|
trackerRouter.get('/tracker/scrape', (req, res) => onHttpRequest(req, res, { action: 'scrape' }))
|
|
|
|
|
2018-12-26 10:36:24 +01:00
|
|
|
function createWebsocketTrackerServer (app: express.Application) {
|
2021-08-27 14:32:44 +02:00
|
|
|
const server = createServer(app)
|
2019-01-08 15:51:52 +01:00
|
|
|
const wss = new WebSocketServer({ noServer: true })
|
|
|
|
|
2018-06-26 16:53:24 +02:00
|
|
|
wss.on('connection', function (ws, req) {
|
2019-01-08 15:51:52 +01:00
|
|
|
ws['ip'] = proxyAddr(req, CONFIG.TRUST_PROXY)
|
2018-06-26 16:53:24 +02:00
|
|
|
|
|
|
|
trackerServer.onWebSocketConnection(ws)
|
|
|
|
})
|
|
|
|
|
2020-01-31 16:56:52 +01:00
|
|
|
server.on('upgrade', (request: express.Request, socket, head) => {
|
2020-02-25 16:27:35 +01:00
|
|
|
if (request.url === '/tracker/socket') {
|
2020-06-25 16:27:35 +02:00
|
|
|
const ip = proxyAddr(request, CONFIG.TRUST_PROXY)
|
|
|
|
|
2022-12-23 13:38:28 +01:00
|
|
|
if (blockedIPs.has(ip)) {
|
|
|
|
logger.debug('Blocking IP %s from tracker.', ip)
|
2020-06-25 16:27:35 +02:00
|
|
|
|
2022-12-23 13:38:28 +01:00
|
|
|
socket.write('HTTP/1.1 403 Forbidden\r\n\r\n')
|
|
|
|
socket.destroy()
|
|
|
|
return
|
|
|
|
}
|
2020-06-25 16:27:35 +02:00
|
|
|
|
2022-12-23 13:38:28 +01:00
|
|
|
// FIXME: typings
|
|
|
|
return wss.handleUpgrade(request, socket as any, head, ws => wss.emit('connection', ws, request))
|
2019-01-08 15:51:52 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Don't destroy socket, we have Socket.IO too
|
|
|
|
})
|
|
|
|
|
2023-01-05 10:19:51 +01:00
|
|
|
return { server, trackerServer }
|
2018-06-26 16:53:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
export {
|
|
|
|
trackerRouter,
|
2018-12-26 10:36:24 +01:00
|
|
|
createWebsocketTrackerServer
|
2018-06-26 16:53:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
function runPeersChecker () {
|
|
|
|
setInterval(() => {
|
|
|
|
logger.debug('Checking peers.')
|
|
|
|
|
|
|
|
for (const ip of Object.keys(peersIpInfoHash)) {
|
|
|
|
if (peersIps[ip] > TRACKER_RATE_LIMITS.ANNOUNCES_PER_IP) {
|
|
|
|
logger.warn('Peer %s made abnormal requests (%d).', ip, peersIps[ip])
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
peersIpInfoHash = {}
|
|
|
|
peersIps = {}
|
|
|
|
}, TRACKER_RATE_LIMITS.INTERVAL)
|
|
|
|
}
|