]> git.immae.eu Git - github/Chocobozzz/PeerTube.git/blame - server/controllers/tracker.ts
Refactor video playlist middlewares
[github/Chocobozzz/PeerTube.git] / server / controllers / tracker.ts
CommitLineData
9b67da3d
C
1import { logger } from '../helpers/logger'
2import * as express from 'express'
3import * as http from 'http'
4import * as bitTorrentTracker from 'bittorrent-tracker'
5import * as proxyAddr from 'proxy-addr'
6import { Server as WebSocketServer } from 'ws'
7import { CONFIG, TRACKER_RATE_LIMITS } from '../initializers/constants'
cc43831a 8import { VideoFileModel } from '../models/video/video-file'
89ada4e2 9import { parse } from 'url'
09209296 10import { VideoStreamingPlaylistModel } from '../models/video/video-streaming-playlist'
9b67da3d
C
11
12const TrackerServer = bitTorrentTracker.Server
13
14const trackerRouter = express.Router()
15
16let peersIps = {}
17let peersIpInfoHash = {}
18runPeersChecker()
19
20const trackerServer = new TrackerServer({
21 http: false,
22 udp: false,
23 ws: false,
24 dht: false,
09209296 25 filter: async function (infoHash, params, cb) {
9b67da3d
C
26 let ip: string
27
28 if (params.type === 'ws') {
29 ip = params.socket.ip
30 } else {
31 ip = params.httpReq.ip
32 }
33
34 const key = ip + '-' + infoHash
35
09209296
C
36 peersIps[ ip ] = peersIps[ ip ] ? peersIps[ ip ] + 1 : 1
37 peersIpInfoHash[ key ] = peersIpInfoHash[ key ] ? peersIpInfoHash[ key ] + 1 : 1
9b67da3d 38
09209296 39 if (peersIpInfoHash[ key ] > TRACKER_RATE_LIMITS.ANNOUNCES_PER_IP_PER_INFOHASH) {
9b67da3d
C
40 return cb(new Error(`Too many requests (${peersIpInfoHash[ key ]} of ip ${ip} for torrent ${infoHash}`))
41 }
42
09209296
C
43 try {
44 const videoFileExists = await VideoFileModel.doesInfohashExist(infoHash)
45 if (videoFileExists === true) return cb()
cc43831a 46
09209296
C
47 const playlistExists = await VideoStreamingPlaylistModel.doesInfohashExist(infoHash)
48 if (playlistExists === true) return cb()
49
50 return cb(new Error(`Unknown infoHash ${infoHash}`))
51 } catch (err) {
52 logger.error('Error in tracker filter.', { err })
53 return cb(err)
54 }
9b67da3d
C
55 }
56})
57
58trackerServer.on('error', function (err) {
59 logger.error('Error in tracker.', { err })
60})
61
62trackerServer.on('warning', function (err) {
63 logger.warn('Warning in tracker.', { err })
64})
65
66const onHttpRequest = trackerServer.onHttpRequest.bind(trackerServer)
67trackerRouter.get('/tracker/announce', (req, res) => onHttpRequest(req, res, { action: 'announce' }))
68trackerRouter.get('/tracker/scrape', (req, res) => onHttpRequest(req, res, { action: 'scrape' }))
69
cef534ed 70function createWebsocketTrackerServer (app: express.Application) {
9b67da3d 71 const server = http.createServer(app)
89ada4e2
C
72 const wss = new WebSocketServer({ noServer: true })
73
9b67da3d 74 wss.on('connection', function (ws, req) {
89ada4e2 75 ws['ip'] = proxyAddr(req, CONFIG.TRUST_PROXY)
9b67da3d
C
76
77 trackerServer.onWebSocketConnection(ws)
78 })
79
89ada4e2
C
80 server.on('upgrade', (request, socket, head) => {
81 const pathname = parse(request.url).pathname
82
83 if (pathname === '/tracker/socket') {
84 wss.handleUpgrade(request, socket, head, ws => wss.emit('connection', ws, request))
85 }
86
87 // Don't destroy socket, we have Socket.IO too
88 })
89
9b67da3d
C
90 return server
91}
92
93// ---------------------------------------------------------------------------
94
95export {
96 trackerRouter,
cef534ed 97 createWebsocketTrackerServer
9b67da3d
C
98}
99
100// ---------------------------------------------------------------------------
101
102function runPeersChecker () {
103 setInterval(() => {
104 logger.debug('Checking peers.')
105
106 for (const ip of Object.keys(peersIpInfoHash)) {
107 if (peersIps[ip] > TRACKER_RATE_LIMITS.ANNOUNCES_PER_IP) {
108 logger.warn('Peer %s made abnormal requests (%d).', ip, peersIps[ip])
109 }
110 }
111
112 peersIpInfoHash = {}
113 peersIps = {}
114 }, TRACKER_RATE_LIMITS.INTERVAL)
115}