]> git.immae.eu Git - github/Chocobozzz/PeerTube.git/blame - server/controllers/tracker.ts
Increase test timeout
[github/Chocobozzz/PeerTube.git] / server / controllers / tracker.ts
CommitLineData
41fb13c3
C
1import { Server as TrackerServer } from 'bittorrent-tracker'
2import express from 'express'
3import { createServer } from 'http'
4import proxyAddr from 'proxy-addr'
e874edd9 5import { WebSocketServer } from 'ws'
db48de85
C
6import { Redis } from '@server/lib/redis'
7import { logger } from '../helpers/logger'
8import { CONFIG } from '../initializers/config'
6dd9de95 9import { TRACKER_RATE_LIMITS } from '../initializers/constants'
cc43831a 10import { VideoFileModel } from '../models/video/video-file'
09209296 11import { VideoStreamingPlaylistModel } from '../models/video/video-streaming-playlist'
9b67da3d 12
9b67da3d
C
13const trackerRouter = express.Router()
14
15let peersIps = {}
16let peersIpInfoHash = {}
17runPeersChecker()
18
19const trackerServer = new TrackerServer({
20 http: false,
21 udp: false,
22 ws: false,
09209296 23 filter: async function (infoHash, params, cb) {
31b6ddf8
C
24 if (CONFIG.TRACKER.ENABLED === false) {
25 return cb(new Error('Tracker is disabled on this instance.'))
26 }
27
9b67da3d
C
28 let ip: string
29
30 if (params.type === 'ws') {
b14e8e46 31 ip = params.ip
9b67da3d
C
32 } else {
33 ip = params.httpReq.ip
34 }
35
36 const key = ip + '-' + infoHash
37
a1587156
C
38 peersIps[ip] = peersIps[ip] ? peersIps[ip] + 1 : 1
39 peersIpInfoHash[key] = peersIpInfoHash[key] ? peersIpInfoHash[key] + 1 : 1
9b67da3d 40
a1587156
C
41 if (CONFIG.TRACKER.REJECT_TOO_MANY_ANNOUNCES && peersIpInfoHash[key] > TRACKER_RATE_LIMITS.ANNOUNCES_PER_IP_PER_INFOHASH) {
42 return cb(new Error(`Too many requests (${peersIpInfoHash[key]} of ip ${ip} for torrent ${infoHash}`))
9b67da3d
C
43 }
44
09209296 45 try {
31b6ddf8
C
46 if (CONFIG.TRACKER.PRIVATE === false) return cb()
47
35f28e94 48 const videoFileExists = await VideoFileModel.doesInfohashExistCached(infoHash)
09209296 49 if (videoFileExists === true) return cb()
cc43831a 50
d988e9a2 51 const playlistExists = await VideoStreamingPlaylistModel.doesInfohashExistCached(infoHash)
09209296
C
52 if (playlistExists === true) return cb()
53
db48de85
C
54 cb(new Error(`Unknown infoHash ${infoHash} requested by ip ${ip}`))
55
56 // Close socket connection and block IP for a few time
57 if (params.type === 'ws') {
58 Redis.Instance.setTrackerBlockIP(ip)
59 .catch(err => logger.error('Cannot set tracker block ip.', { err }))
60
61 // setTimeout to wait filter response
62 setTimeout(() => params.socket.close(), 0)
63 }
09209296
C
64 } catch (err) {
65 logger.error('Error in tracker filter.', { err })
66 return cb(err)
67 }
9b67da3d
C
68 }
69})
70
31b6ddf8 71if (CONFIG.TRACKER.ENABLED !== false) {
31b6ddf8
C
72 trackerServer.on('error', function (err) {
73 logger.error('Error in tracker.', { err })
74 })
75
76 trackerServer.on('warning', function (err) {
a4152bed
C
77 if (CONFIG.LOG.LOG_TRACKER_UNKNOWN_INFOHASH) {
78 const message = err.message || ''
79 if (message.includes('Uknown infoHash')) return
80 }
81
31b6ddf8
C
82 logger.warn('Warning in tracker.', { err })
83 })
84}
9b67da3d
C
85
86const onHttpRequest = trackerServer.onHttpRequest.bind(trackerServer)
87trackerRouter.get('/tracker/announce', (req, res) => onHttpRequest(req, res, { action: 'announce' }))
88trackerRouter.get('/tracker/scrape', (req, res) => onHttpRequest(req, res, { action: 'scrape' }))
89
cef534ed 90function createWebsocketTrackerServer (app: express.Application) {
41fb13c3 91 const server = createServer(app)
89ada4e2
C
92 const wss = new WebSocketServer({ noServer: true })
93
9b67da3d 94 wss.on('connection', function (ws, req) {
89ada4e2 95 ws['ip'] = proxyAddr(req, CONFIG.TRUST_PROXY)
9b67da3d
C
96
97 trackerServer.onWebSocketConnection(ws)
98 })
99
a1587156 100 server.on('upgrade', (request: express.Request, socket, head) => {
4832e415 101 if (request.url === '/tracker/socket') {
db48de85
C
102 const ip = proxyAddr(request, CONFIG.TRUST_PROXY)
103
104 Redis.Instance.doesTrackerBlockIPExist(ip)
105 .then(result => {
106 if (result === true) {
107 logger.debug('Blocking IP %s from tracker.', ip)
108
109 socket.write('HTTP/1.1 403 Forbidden\r\n\r\n')
110 socket.destroy()
111 return
112 }
113
31aa391d
C
114 // FIXME: typings
115 return wss.handleUpgrade(request, socket as any, head, ws => wss.emit('connection', ws, request))
db48de85
C
116 })
117 .catch(err => logger.error('Cannot check if tracker block ip exists.', { err }))
89ada4e2
C
118 }
119
120 // Don't destroy socket, we have Socket.IO too
121 })
122
9b67da3d
C
123 return server
124}
125
126// ---------------------------------------------------------------------------
127
128export {
129 trackerRouter,
cef534ed 130 createWebsocketTrackerServer
9b67da3d
C
131}
132
133// ---------------------------------------------------------------------------
134
135function runPeersChecker () {
136 setInterval(() => {
137 logger.debug('Checking peers.')
138
139 for (const ip of Object.keys(peersIpInfoHash)) {
140 if (peersIps[ip] > TRACKER_RATE_LIMITS.ANNOUNCES_PER_IP) {
141 logger.warn('Peer %s made abnormal requests (%d).', ip, peersIps[ip])
142 }
143 }
144
145 peersIpInfoHash = {}
146 peersIps = {}
147 }, TRACKER_RATE_LIMITS.INTERVAL)
148}