forked from mafintosh/peerflix
-
Notifications
You must be signed in to change notification settings - Fork 0
/
index.js
211 lines (172 loc) · 6.22 KB
/
index.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
var torrentStream = require('torrent-stream')
var http = require('http')
var fs = require('fs')
var rangeParser = require('range-parser')
var xtend = require('xtend')
var url = require('url')
var mime = require('mime')
var pump = require('pump')
var parseBlocklist = function (filename) {
// TODO: support gzipped files
var blocklistData = fs.readFileSync(filename, { encoding: 'utf8' })
var blocklist = []
blocklistData.split('\n').forEach(function (line) {
var match = null
if ((match = /^\s*[^#].*?\s*:\s*([a-f0-9.:]+?)\s*-\s*([a-f0-9.:]+?)\s*$/.exec(line))) {
blocklist.push({
start: match[1],
end: match[2]
})
}
})
return blocklist
}
var truthy = function () {
return true
}
var createServer = function (e, opts) {
var server = http.createServer()
var index = opts.index
var getType = opts.type || mime.lookup.bind(mime)
var filter = opts.filter || truthy
var onready = function () {
if (typeof index !== 'number') {
index = e.files.reduce(function (a, b) {
return a.length > b.length ? a : b
})
index = e.files.indexOf(index)
}
e.files[index].select()
server.index = e.files[index]
server.infoHash = e.infoHash
if (opts.sort) e.files.sort(opts.sort)
}
if (e.torrent) onready()
else e.on('ready', onready)
server.on('request', function (request, response) {
var u = url.parse(request.url)
var host = request.headers.host || 'localhost'
var toPlaylist = function () {
var toEntry = function (file, i) {
return '#EXTINF:-1,' + file.path + '\n' + 'http://' + host + '/' + i
}
return '#EXTM3U\n' + e.files.filter(filter).map(toEntry).join('\n')
}
var toJSON = function () {
var totalPeers = e.swarm.wires
var activePeers = totalPeers.filter(function (wire) {
return !wire.peerChoking
})
var totalLength = e.files.reduce(function (prevFileLength, currFile) {
return prevFileLength + currFile.length
}, 0)
var toEntry = function (file, i) {
return {
name: file.name,
url: 'http://' + host + '/' + i,
length: file.length
}
}
var swarmStats = {
totalLength: totalLength,
downloaded: e.swarm.downloaded,
uploaded: e.swarm.uploaded,
downloadSpeed: parseInt(e.swarm.downloadSpeed(), 10),
uploadSpeed: parseInt(e.swarm.uploadSpeed(), 10),
totalPeers: totalPeers.length,
activePeers: activePeers.length,
files: e.files.filter(filter).map(toEntry)
}
return JSON.stringify(swarmStats, null, ' ')
}
// Allow CORS requests to specify arbitrary headers, e.g. 'Range',
// by responding to the OPTIONS preflight request with the specified
// origin and requested headers.
if (request.method === 'OPTIONS' && request.headers['access-control-request-headers']) {
response.setHeader('Access-Control-Allow-Origin', request.headers.origin)
response.setHeader('Access-Control-Allow-Methods', 'POST, GET, OPTIONS')
response.setHeader(
'Access-Control-Allow-Headers',
request.headers['access-control-request-headers'])
response.setHeader('Access-Control-Max-Age', '1728000')
response.end()
return
}
if (request.headers.origin) response.setHeader('Access-Control-Allow-Origin', request.headers.origin)
if (u.pathname === '/') u.pathname = '/' + index
if (u.pathname === '/favicon.ico') {
response.statusCode = 404
response.end()
return
}
if (u.pathname === '/.json') {
var json = toJSON()
response.setHeader('Content-Type', 'application/json; charset=utf-8')
response.setHeader('Content-Length', Buffer.byteLength(json))
response.end(json)
return
}
if (u.pathname === '/.m3u') {
var playlist = toPlaylist()
response.setHeader('Content-Type', 'application/x-mpegurl; charset=utf-8')
response.setHeader('Content-Length', Buffer.byteLength(playlist))
response.end(playlist)
return
}
e.files.forEach(function (file, i) {
if (u.pathname.slice(1) === file.name) u.pathname = '/' + i
})
var i = Number(u.pathname.slice(1))
if (isNaN(i) || i >= e.files.length) {
response.statusCode = 404
response.end()
return
}
var file = e.files[i]
var range = request.headers.range
range = range && rangeParser(file.length, range)[0]
response.setHeader('Accept-Ranges', 'bytes')
response.setHeader('Content-Type', getType(file.name))
response.setHeader('transferMode.dlna.org', 'Streaming')
response.setHeader('contentFeatures.dlna.org', 'DLNA.ORG_OP=01;DLNA.ORG_CI=0;DLNA.ORG_FLAGS=017000 00000000000000000000000000')
if (!range) {
response.setHeader('Content-Length', file.length)
if (request.method === 'HEAD') return response.end()
pump(file.createReadStream(), response)
return
}
response.statusCode = 206
response.setHeader('Content-Length', range.end - range.start + 1)
response.setHeader('Content-Range', 'bytes ' + range.start + '-' + range.end + '/' + file.length)
response.setHeader('transferMode.dlna.org', 'Streaming')
response.setHeader('contentFeatures.dlna.org', 'DLNA.ORG_OP=01;DLNA.ORG_CI=0;DLNA.ORG_FLAGS=017000 00000000000000000000000000')
if (request.method === 'HEAD') return response.end()
pump(file.createReadStream(range), response)
})
server.on('connection', function (socket) {
socket.setTimeout(36000000)
})
return server
}
module.exports = function (torrent, opts) {
if (!opts) opts = {}
// Parse blocklist
if (opts.blocklist) opts.blocklist = parseBlocklist(opts.blocklist)
var engine = torrentStream(torrent, xtend(opts, {port: opts.peerPort}))
// Just want torrent-stream to list files.
if (opts.list) return engine
// Pause/Resume downloading as needed
engine.on('uninterested', function () {
engine.swarm.pause()
})
engine.on('interested', function () {
engine.swarm.resume()
})
engine.server = createServer(engine, opts)
// Listen when torrent-stream is ready, by default a random port.
engine.on('ready', function () {
engine.server.listen(opts.port || 0, opts.hostname)
})
engine.listen()
return engine
}