Skip to content

Commit e6033db

Browse files
committed
Merge branch 'server-ipv6'
2 parents 37203fb + 83b4ecc commit e6033db

File tree

4 files changed

+39
-14
lines changed

4 files changed

+39
-14
lines changed

lib/common.js

Lines changed: 3 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,9 @@
44

55
var querystring = require('querystring')
66

7+
exports.IPV4_RE = /^[\d\.]+$/
8+
exports.IPV6_RE = /^[\da-fA-F:]+$/
9+
710
exports.NUM_ANNOUNCE_PEERS = 50
811
exports.MAX_ANNOUNCE_PEERS = 82
912

lib/parse_http.js

Lines changed: 3 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@ module.exports = parseHttpRequest
22

33
var common = require('./common')
44

5-
var REMOVE_IPV6_RE = /^::ffff:/
5+
var REMOVE_IPV4_MAPPED_IPV6_RE = /^::ffff:/
66

77
function parseHttpRequest (req, options) {
88
var s = req.url.split('?')
@@ -30,8 +30,8 @@ function parseHttpRequest (req, options) {
3030

3131
params.ip = options.trustProxy
3232
? req.headers['x-forwarded-for'] || req.connection.remoteAddress
33-
: req.connection.remoteAddress.replace(REMOVE_IPV6_RE, '') // force ipv4
34-
params.addr = params.ip + ':' + params.port // TODO: ipv6 brackets?
33+
: req.connection.remoteAddress.replace(REMOVE_IPV4_MAPPED_IPV6_RE, '') // force ipv4
34+
params.addr = (common.IPV6_RE.test(params.ip) ? '[' + params.ip + ']' : params.ip) + ':' + params.port
3535

3636
} else if (s[0] === '/scrape') { // unofficial scrape message
3737
params.action = common.ACTIONS.SCRAPE

server.js

Lines changed: 18 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -94,7 +94,10 @@ Server.prototype.listen = function (port, onlistening) {
9494
function onPort (err, port) {
9595
if (err) return self.emit('error', err)
9696
self.port = port
97-
self._httpServer && self._httpServer.listen(port.http || port)
97+
// ATTENTION:
98+
// binding to :: only receives IPv4 connections if the bindv6only
99+
// sysctl is set 0, which is the default on many operating systems.
100+
self._httpServer && self._httpServer.listen(port.http || port, '::')
98101
self._udpSocket && self._udpSocket.bind(port.udp || port)
99102
}
100103

@@ -207,11 +210,23 @@ Server.prototype._onAnnounce = function (params, cb) {
207210
if (response) {
208211
if (!response.action) response.action = common.ACTIONS.ANNOUNCE
209212
if (!response.intervalMs) response.intervalMs = self._intervalMs
213+
210214
if (params.compact === 1) {
211-
response.peers = string2compact(response.peers.map(function (peer) {
212-
return peer.ip + ':' + peer.port // TODO: ipv6 brackets
215+
var peers = response.peers
216+
// Find IPv4 peers
217+
response.peers = string2compact(peers.filter(function (peer) {
218+
return common.IPV4_RE.test(peer.ip)
219+
}).map(function (peer) {
220+
return peer.ip + ':' + peer.port
221+
}))
222+
// Find IPv6 peers
223+
response.peers6 = string2compact(peers.filter(function (peer) {
224+
return common.IPV6_RE.test(peer.ip)
225+
}).map(function (peer) {
226+
return '[' + peer.ip + ']:' + peer.port
213227
}))
214228
}
229+
// IPv6 peers are not separate for non-compact responses
215230
}
216231
cb(err, response)
217232
})

test/server.js

Lines changed: 15 additions & 8 deletions
Original file line numberDiff line numberDiff line change
@@ -7,11 +7,14 @@ var peerId = new Buffer('01234567890123456789')
77
var peerId2 = new Buffer('12345678901234567890')
88
var torrentLength = 50000
99

10-
function serverTest (t, serverType) {
10+
function serverTest (t, serverType, serverFamily) {
1111
t.plan(26)
1212

1313
var opts = serverType === 'http' ? { udp: false } : { http: false }
1414
var server = new Server(opts)
15+
var serverAddr = serverFamily === 'inet6' ? '[::1]' : '127.0.0.1'
16+
var clientAddr = serverFamily === 'inet6' ? '[::1]' : '127.0.0.1'
17+
var clientIp = serverFamily === 'inet6' ? '::1' : '127.0.0.1'
1518

1619
server.on('error', function (err) {
1720
t.fail(err.message)
@@ -26,7 +29,7 @@ function serverTest (t, serverType) {
2629
})
2730

2831
server.listen(function (port) {
29-
var announceUrl = serverType + '://127.0.0.1:' + port + '/announce'
32+
var announceUrl = serverType + '://' + serverAddr + ':' + port + '/announce'
3033

3134
var client = new Client(peerId, 6881, {
3235
infoHash: infoHash,
@@ -49,8 +52,8 @@ function serverTest (t, serverType) {
4952
t.equal(server.getSwarm(infoHash).complete, 0)
5053
t.equal(server.getSwarm(infoHash).incomplete, 1)
5154
t.equal(Object.keys(server.getSwarm(infoHash).peers).length, 1)
52-
t.deepEqual(server.getSwarm(infoHash).peers['127.0.0.1:6881'], {
53-
ip: '127.0.0.1',
55+
t.deepEqual(server.getSwarm(infoHash).peers[clientAddr + ':6881'], {
56+
ip: clientIp,
5457
port: 6881,
5558
peerId: peerId.toString('hex')
5659
})
@@ -83,7 +86,7 @@ function serverTest (t, serverType) {
8386
})
8487

8588
client2.once('peer', function (addr) {
86-
t.equal(addr, '127.0.0.1:6881')
89+
t.equal(addr, clientAddr + ':6881')
8790

8891
client2.stop()
8992
client2.once('update', function (data) {
@@ -109,10 +112,14 @@ function serverTest (t, serverType) {
109112
})
110113
}
111114

112-
test('http server', function (t) {
113-
serverTest(t, 'http')
115+
test('http ipv4 server', function (t) {
116+
serverTest(t, 'http', 'inet')
117+
})
118+
119+
test('http ipv6 server', function (t) {
120+
serverTest(t, 'http', 'inet6')
114121
})
115122

116123
test('udp server', function (t) {
117-
serverTest(t, 'udp')
124+
serverTest(t, 'udp', 'inet')
118125
})

0 commit comments

Comments
 (0)