Skip to content
This repository was archived by the owner on Mar 10, 2020. It is now read-only.

Commit bcbf0d2

Browse files
olizillaAlan Shaw
authored and
Alan Shaw
committed
fix: throw on invalid multiaddr to constructor (#934)
- dont assume that invalid multiaddrs are valid hosts - if multiaddrOrHost starts with a /, assume its a multiaddr - throw if that multiaddr is invalid. Previously, we'd try and use invalid multiaddrs as the host, which would lead to requests like http:///dnsaddr/foobar.com:5001/api License: MIT Signed-off-by: Oli Evans <oli@tableflip.io>
1 parent 12ddaa3 commit bcbf0d2

File tree

2 files changed

+118
-65
lines changed

2 files changed

+118
-65
lines changed

src/index.js

+33-24
Original file line numberDiff line numberDiff line change
@@ -7,35 +7,35 @@ const getConfig = require('./utils/default-config')
77
const sendRequest = require('./utils/send-request')
88

99
function ipfsClient (hostOrMultiaddr, port, opts) {
10-
const config = getConfig()
11-
12-
try {
13-
const maddr = multiaddr(hostOrMultiaddr).nodeAddress()
14-
config.host = maddr.address
15-
config.port = maddr.port
16-
} catch (e) {
17-
if (typeof hostOrMultiaddr === 'string') {
18-
config.host = hostOrMultiaddr
19-
config.port = port && typeof port !== 'object' ? port : config.port
10+
// convert all three params to objects that we can merge.
11+
let hostAndPort = {}
12+
13+
if (!hostOrMultiaddr) {
14+
// autoconfigure host and port in browser
15+
if (typeof self !== 'undefined') {
16+
const split = self.location.host.split(':')
17+
hostAndPort.host = split[0]
18+
hostAndPort.port = split[1]
19+
}
20+
} else if (multiaddr.isMultiaddr(hostOrMultiaddr)) {
21+
hostAndPort = toHostAndPort(hostOrMultiaddr)
22+
} else if (typeof hostOrMultiaddr === 'object') {
23+
hostAndPort = hostOrMultiaddr
24+
} else if (typeof hostOrMultiaddr === 'string') {
25+
if (hostOrMultiaddr[0] === '/') {
26+
// throws if multiaddr is malformed or can't be converted to a nodeAddress
27+
hostAndPort = toHostAndPort(multiaddr(hostOrMultiaddr))
28+
} else {
29+
// hostOrMultiaddr is domain or ip address as a string
30+
hostAndPort.host = hostOrMultiaddr
2031
}
2132
}
2233

23-
let lastIndex = arguments.length
24-
while (!opts && lastIndex-- > 0) {
25-
opts = arguments[lastIndex]
26-
if (opts) break
27-
}
28-
29-
Object.assign(config, opts)
30-
31-
// autoconfigure in browser
32-
if (!config.host &&
33-
typeof self !== 'undefined') {
34-
const split = self.location.host.split(':')
35-
config.host = split[0]
36-
config.port = split[1]
34+
if (port && typeof port !== 'object') {
35+
port = { port: port }
3736
}
3837

38+
const config = Object.assign(getConfig(), hostAndPort, port, opts)
3939
const requestAPI = sendRequest(config)
4040
const cmds = loadCommands(requestAPI, config)
4141
cmds.send = requestAPI
@@ -44,4 +44,13 @@ function ipfsClient (hostOrMultiaddr, port, opts) {
4444
return cmds
4545
}
4646

47+
// throws if multiaddr can't be converted to a nodeAddress
48+
function toHostAndPort (multiaddr) {
49+
const nodeAddr = multiaddr.nodeAddress()
50+
return {
51+
host: nodeAddr.address,
52+
port: nodeAddr.port
53+
}
54+
}
55+
4756
module.exports = ipfsClient

test/constructor.spec.js

+85-41
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
/* eslint-env mocha */
22
'use strict'
33

4+
const multiaddr = require('multiaddr')
45
const chai = require('chai')
56
const dirtyChai = require('dirty-chai')
67
const expect = chai.expect
@@ -9,18 +10,74 @@ chai.use(dirtyChai)
910
const f = require('./utils/factory')
1011
const ipfsClient = require('../src/index.js')
1112

12-
function clientWorks (client, done) {
13-
client.id((err, id) => {
14-
expect(err).to.not.exist()
13+
describe('ipfs-http-client constructor tests', () => {
14+
describe('parameter permuations', () => {
15+
it('none', () => {
16+
const ipfs = ipfsClient()
17+
if (typeof self !== 'undefined') {
18+
const { hostname, port } = self.location
19+
expectConfig(ipfs, { host: hostname, port })
20+
} else {
21+
expectConfig(ipfs, {})
22+
}
23+
})
1524

16-
expect(id).to.have.a.property('id')
17-
expect(id).to.have.a.property('publicKey')
18-
done()
25+
it('opts', () => {
26+
const host = 'wizard.world'
27+
const port = '999'
28+
const protocol = 'https'
29+
const ipfs = ipfsClient({ host, port, protocol })
30+
expectConfig(ipfs, { host, port, protocol })
31+
})
32+
33+
it('mutliaddr dns4 string, opts', () => {
34+
const host = 'foo.com'
35+
const port = '1001'
36+
const protocol = 'https'
37+
const addr = `/dns4/${host}/tcp/${port}`
38+
const ipfs = ipfsClient(addr, { protocol })
39+
expectConfig(ipfs, { host, port, protocol })
40+
})
41+
42+
it('mutliaddr ipv4 string', () => {
43+
const host = '101.101.101.101'
44+
const port = '1001'
45+
const addr = `/ip4/${host}/tcp/${port}`
46+
const ipfs = ipfsClient(addr)
47+
expectConfig(ipfs, { host, port })
48+
})
49+
50+
it('mutliaddr instance', () => {
51+
const host = 'ace.place'
52+
const port = '1001'
53+
const addr = multiaddr(`/dns4/${host}/tcp/${port}`)
54+
const ipfs = ipfsClient(addr)
55+
expectConfig(ipfs, { host, port })
56+
})
57+
58+
it('host and port strings', () => {
59+
const host = '1.1.1.1'
60+
const port = '9999'
61+
const ipfs = ipfsClient(host, port)
62+
expectConfig(ipfs, { host, port })
63+
})
64+
65+
it('host, port and api path', () => {
66+
const host = '10.100.100.255'
67+
const port = '9999'
68+
const apiPath = '/future/api/v1/'
69+
const ipfs = ipfsClient(host, port, { 'api-path': apiPath })
70+
expectConfig(ipfs, { host, port, apiPath })
71+
})
72+
73+
it('throws on invalid mutliaddr', () => {
74+
expect(() => ipfsClient('/dns4')).to.throw('invalid address')
75+
expect(() => ipfsClient('/hello')).to.throw('no protocol with name')
76+
expect(() => ipfsClient('/dns4/ipfs.io')).to.throw('multiaddr must have a valid format')
77+
})
1978
})
20-
}
2179

22-
describe('ipfs-http-client constructor tests', () => {
23-
describe('parameter permuations', () => {
80+
describe('integration', () => {
2481
let apiAddr
2582
let ipfsd
2683

@@ -40,39 +97,26 @@ describe('ipfs-http-client constructor tests', () => {
4097
ipfsd.stop(done)
4198
})
4299

43-
it('opts', (done) => {
44-
const splitted = apiAddr.split('/')
45-
clientWorks(ipfsClient({
46-
host: splitted[2],
47-
port: splitted[4],
48-
protocol: 'http'
49-
}), done)
50-
})
51-
52-
it('mutliaddr, opts', (done) => {
53-
clientWorks(ipfsClient(apiAddr, { protocol: 'http' }), done)
54-
})
55-
56-
it('host, port', (done) => {
57-
const splitted = apiAddr.split('/')
58-
59-
clientWorks(ipfsClient(splitted[2], splitted[4]), done)
60-
})
61-
62-
it('specify host, port and api path', (done) => {
63-
const splitted = apiAddr.split('/')
64-
65-
clientWorks(ipfsClient({
66-
host: splitted[2],
67-
port: splitted[4],
68-
'api-path': '/api/v0/'
69-
}), done)
100+
it('can connect to an ipfs http api', (done) => {
101+
clientWorks(ipfsClient(apiAddr), done)
70102
})
103+
})
104+
})
71105

72-
it('host, port, opts', (done) => {
73-
const splitted = apiAddr.split('/')
106+
function clientWorks (client, done) {
107+
client.id((err, id) => {
108+
expect(err).to.not.exist()
74109

75-
clientWorks(ipfsClient(splitted[2], splitted[4], { protocol: 'http' }), done)
76-
})
110+
expect(id).to.have.a.property('id')
111+
expect(id).to.have.a.property('publicKey')
112+
done()
77113
})
78-
})
114+
}
115+
116+
function expectConfig (ipfs, { host, port, protocol, apiPath }) {
117+
const conf = ipfs.util.getEndpointConfig()
118+
expect(conf.host).to.equal(host || 'localhost')
119+
expect(conf.port).to.equal(port || '5001')
120+
expect(conf.protocol).to.equal(protocol || 'http')
121+
expect(conf['api-path']).to.equal(apiPath || '/api/v0/')
122+
}

0 commit comments

Comments
 (0)