1
0
Fork 0
peertube/server/controllers/tracker.ts

150 lines
4.5 KiB
TypeScript
Raw Normal View History

2021-08-27 08:32:44 -04:00
import { Server as TrackerServer } from 'bittorrent-tracker'
import express from 'express'
import { createServer } from 'http'
import proxyAddr from 'proxy-addr'
import { WebSocketServer } from 'ws'
2020-06-25 10:27:35 -04:00
import { Redis } from '@server/lib/redis'
import { logger } from '../helpers/logger'
import { CONFIG } from '../initializers/config'
2019-04-11 05:33:44 -04:00
import { TRACKER_RATE_LIMITS } from '../initializers/constants'
2018-08-14 05:00:03 -04:00
import { VideoFileModel } from '../models/video/video-file'
2019-01-29 02:37:25 -05:00
import { VideoStreamingPlaylistModel } from '../models/video/video-streaming-playlist'
2018-06-26 10:53:24 -04:00
const trackerRouter = express.Router()
let peersIps = {}
let peersIpInfoHash = {}
runPeersChecker()
const trackerServer = new TrackerServer({
http: false,
udp: false,
ws: false,
2019-01-29 02:37:25 -05:00
filter: async function (infoHash, params, cb) {
2019-04-10 03:23:18 -04:00
if (CONFIG.TRACKER.ENABLED === false) {
return cb(new Error('Tracker is disabled on this instance.'))
}
2018-06-26 10:53:24 -04:00
let ip: string
if (params.type === 'ws') {
2021-01-11 09:45:04 -05:00
ip = params.ip
2018-06-26 10:53:24 -04:00
} else {
ip = params.httpReq.ip
}
const key = ip + '-' + infoHash
2020-01-31 10:56:52 -05:00
peersIps[ip] = peersIps[ip] ? peersIps[ip] + 1 : 1
peersIpInfoHash[key] = peersIpInfoHash[key] ? peersIpInfoHash[key] + 1 : 1
2018-06-26 10:53:24 -04:00
2020-01-31 10:56:52 -05: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 10:53:24 -04:00
}
2019-01-29 02:37:25 -05:00
try {
2019-04-10 03:23:18 -04:00
if (CONFIG.TRACKER.PRIVATE === false) return cb()
2020-01-03 07:47:45 -05:00
const videoFileExists = await VideoFileModel.doesInfohashExistCached(infoHash)
2019-01-29 02:37:25 -05:00
if (videoFileExists === true) return cb()
2018-08-14 05:00:03 -04:00
2020-10-29 10:03:31 -04:00
const playlistExists = await VideoStreamingPlaylistModel.doesInfohashExistCached(infoHash)
2019-01-29 02:37:25 -05:00
if (playlistExists === true) return cb()
2020-06-25 10:27:35 -04: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') {
Redis.Instance.setTrackerBlockIP(ip)
.catch(err => logger.error('Cannot set tracker block ip.', { err }))
// setTimeout to wait filter response
setTimeout(() => params.socket.close(), 0)
}
2019-01-29 02:37:25 -05:00
} catch (err) {
logger.error('Error in tracker filter.', { err })
return cb(err)
}
2018-06-26 10:53:24 -04:00
}
})
2019-04-10 03:23:18 -04: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 08:29:57 -04:00
const message = err.message || ''
if (CONFIG.LOG.LOG_TRACKER_UNKNOWN_INFOHASH === false && message.includes('Unknown infoHash')) {
return
}
2019-04-10 03:23:18 -04:00
logger.warn('Warning in tracker.', { err })
})
}
2018-06-26 10:53:24 -04: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 04:36:24 -05:00
function createWebsocketTrackerServer (app: express.Application) {
2021-08-27 08:32:44 -04:00
const server = createServer(app)
2019-01-08 09:51:52 -05:00
const wss = new WebSocketServer({ noServer: true })
2018-06-26 10:53:24 -04:00
wss.on('connection', function (ws, req) {
2019-01-08 09:51:52 -05:00
ws['ip'] = proxyAddr(req, CONFIG.TRUST_PROXY)
2018-06-26 10:53:24 -04:00
trackerServer.onWebSocketConnection(ws)
})
2020-01-31 10:56:52 -05:00
server.on('upgrade', (request: express.Request, socket, head) => {
2020-02-25 10:27:35 -05:00
if (request.url === '/tracker/socket') {
2020-06-25 10:27:35 -04:00
const ip = proxyAddr(request, CONFIG.TRUST_PROXY)
Redis.Instance.doesTrackerBlockIPExist(ip)
.then(result => {
if (result === true) {
logger.debug('Blocking IP %s from tracker.', ip)
socket.write('HTTP/1.1 403 Forbidden\r\n\r\n')
socket.destroy()
return
}
2021-08-06 09:51:45 -04:00
// FIXME: typings
return wss.handleUpgrade(request, socket as any, head, ws => wss.emit('connection', ws, request))
2020-06-25 10:27:35 -04:00
})
.catch(err => logger.error('Cannot check if tracker block ip exists.', { err }))
2019-01-08 09:51:52 -05:00
}
// Don't destroy socket, we have Socket.IO too
})
2018-06-26 10:53:24 -04:00
return server
}
// ---------------------------------------------------------------------------
export {
trackerRouter,
2018-12-26 04:36:24 -05:00
createWebsocketTrackerServer
2018-06-26 10:53:24 -04: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)
}