Skip to content
This repository was archived by the owner on Feb 12, 2024. It is now read-only.

Commit 00fd709

Browse files
authored
feat: allow passing the id of a network peer to ipfs.id (#3386)
Adds parity with go-IPFS and allows looking up known peer IDs using the `ipfs.id` command.
1 parent bff33c4 commit 00fd709

File tree

14 files changed

+151
-44
lines changed

14 files changed

+151
-44
lines changed

docs/core-api/MISCELLANEOUS.md

+2-1
Original file line numberDiff line numberDiff line change
@@ -47,6 +47,7 @@ An optional object which may have the following keys:
4747
| ---- | ---- | ------- | ----------- |
4848
| timeout | `Number` | `undefined` | A timeout in ms |
4949
| signal | [AbortSignal][] | `undefined` | Can be used to cancel any long running requests started as a result of this call |
50+
| peerId | `string` | `undefined` | Look up the identity for this peer instead of the current node |
5051

5152
### Returns
5253

@@ -300,4 +301,4 @@ A great source of [examples](https://github.com/ipfs/js-ipfs/blob/master/package
300301
[rs]: https://www.npmjs.com/package/readable-stream
301302
[ps]: https://www.npmjs.com/package/pull-stream
302303
[cid]: https://www.npmjs.com/package/cids
303-
[AbortSignal]: https://developer.mozilla.org/en-US/docs/Web/API/AbortSignal
304+
[AbortSignal]: https://developer.mozilla.org/en-US/docs/Web/API/AbortSignal

examples/custom-libp2p/package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,7 @@
1111
"license": "MIT",
1212
"dependencies": {
1313
"ipfs": "^0.55.1",
14-
"libp2p": "^0.31.0",
14+
"libp2p": "^0.31.5",
1515
"libp2p-bootstrap": "^0.12.3",
1616
"libp2p-kad-dht": "^0.22.0",
1717
"libp2p-mdns": "^0.16.0",

packages/interface-ipfs-core/package.json

+1
Original file line numberDiff line numberDiff line change
@@ -70,6 +70,7 @@
7070
"nanoid": "^3.1.12",
7171
"native-abort-controller": "^1.0.3",
7272
"p-map": "^4.0.0",
73+
"p-retry": "^4.5.0",
7374
"peer-id": "^0.14.1",
7475
"readable-stream": "^3.4.0",
7576
"uint8arrays": "^2.1.3"

packages/interface-ipfs-core/src/miscellaneous/id.js

+30
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@ const { getDescribe, getIt, expect } = require('../utils/mocha')
55
const { Multiaddr } = require('multiaddr')
66
const CID = require('cids')
77
const { isWebWorker } = require('ipfs-utils/src/env')
8+
const retry = require('p-retry')
89

910
/** @typedef { import("ipfsd-ctl/src/factory") } Factory */
1011
/**
@@ -66,5 +67,34 @@ module.exports = (common, options) => {
6667

6768
await expect(ipfs.id()).to.eventually.have.property('addresses').that.is.not.empty()
6869
})
70+
71+
it('should get the id of another node in the swarm', async function () {
72+
if (isWebWorker) {
73+
// TODO: https://github.com/libp2p/js-libp2p-websockets/issues/129
74+
return this.skip()
75+
}
76+
77+
const ipfsB = (await common.spawn()).api
78+
await ipfs.swarm.connect(ipfsB.peerId.addresses[0])
79+
80+
// have to wait for identify to complete before protocols etc are available for remote hosts
81+
await retry(async () => {
82+
const result = await ipfs.id({
83+
peerId: ipfsB.peerId.id
84+
})
85+
86+
expect(result).to.deep.equal(ipfsB.peerId)
87+
}, { retries: 5 })
88+
})
89+
90+
it('should get our own id when passed as an option', async function () {
91+
const res = await ipfs.id()
92+
93+
const result = await ipfs.id({
94+
peerId: res.id
95+
})
96+
97+
expect(result).to.deep.equal(res)
98+
})
6999
})
70100
}

packages/ipfs-cli/src/commands/id.js

+9-3
Original file line numberDiff line numberDiff line change
@@ -3,11 +3,15 @@
33
const { default: parseDuration } = require('parse-duration')
44

55
module.exports = {
6-
command: 'id',
6+
command: 'id [peerId]',
77

88
describe: 'Shows IPFS Node ID info',
99

1010
builder: {
11+
peerid: {
12+
type: 'string',
13+
describe: 'Peer.ID of node to look up'
14+
},
1115
format: {
1216
alias: 'f',
1317
type: 'string',
@@ -24,10 +28,12 @@ module.exports = {
2428
* @param {import('../types').Context} argv.ctx
2529
* @param {string} argv.format
2630
* @param {number} argv.timeout
31+
* @param {string} [argv.peerId]
2732
*/
28-
async handler ({ ctx: { ipfs, print }, format, timeout }) {
33+
async handler ({ ctx: { ipfs, print }, format, timeout, peerId }) {
2934
const id = await ipfs.id({
30-
timeout
35+
timeout,
36+
peerId
3137
})
3238

3339
if (format) {

packages/ipfs-cli/test/id.js

+20-1
Original file line numberDiff line numberDiff line change
@@ -4,9 +4,11 @@
44
const { expect } = require('aegir/utils/chai')
55
const cli = require('./utils/cli')
66
const sinon = require('sinon')
7+
const PeerId = require('peer-id')
78

89
const defaultOptions = {
9-
timeout: undefined
10+
timeout: undefined,
11+
peerId: undefined
1012
}
1113

1214
describe('id', () => {
@@ -54,4 +56,21 @@ describe('id', () => {
5456
expect(res).to.have.property('id', 'id')
5557
expect(res).to.have.property('publicKey', 'publicKey')
5658
})
59+
60+
it('get the id of another peer', async () => {
61+
const peerId = PeerId.createFromB58String('QmSnuWmxptJZdLJpKRarxBMS2Ju2oANVrgbr2xWbie9b2D')
62+
63+
ipfs.id.withArgs({
64+
...defaultOptions,
65+
peerId: peerId.toString()
66+
}).resolves({
67+
id: 'id',
68+
publicKey: 'publicKey'
69+
})
70+
71+
const out = await cli(`id ${peerId}`, { ipfs })
72+
const res = JSON.parse(out)
73+
expect(res).to.have.property('id', 'id')
74+
expect(res).to.have.property('publicKey', 'publicKey')
75+
})
5776
})

packages/ipfs-core-types/src/root.d.ts

+5-1
Original file line numberDiff line numberDiff line change
@@ -40,7 +40,7 @@ export interface API<OptionExtension = {}> {
4040
* console.log(identity)
4141
* ```
4242
*/
43-
id: (options?: AbortOptions & OptionExtension) => Promise<IDResult>
43+
id: (options?: IDOptions & OptionExtension) => Promise<IDResult>
4444

4545
/**
4646
* Returns the implementation version
@@ -289,6 +289,10 @@ export interface ListOptions extends AbortOptions, PreloadOptions {
289289
includeContent?: boolean
290290
}
291291

292+
export interface IDOptions extends AbortOptions {
293+
peerId?: string
294+
}
295+
292296
export interface IDResult {
293297
id: string
294298
publicKey: string

packages/ipfs-core/package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -94,7 +94,7 @@
9494
"it-map": "^1.0.4",
9595
"it-pipe": "^1.1.0",
9696
"just-safe-set": "^2.2.1",
97-
"libp2p": "^0.31.2",
97+
"libp2p": "^0.31.5",
9898
"libp2p-bootstrap": "^0.12.3",
9999
"libp2p-crypto": "^0.19.3",
100100
"libp2p-floodsub": "^0.25.1",

packages/ipfs-core/src/components/id.js

+39-30
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,8 @@ const pkgversion = require('../../package.json').version
44
const { Multiaddr } = require('multiaddr')
55
const withTimeoutOption = require('ipfs-core-utils/src/with-timeout-option')
66
const uint8ArrayToString = require('uint8arrays/to-string')
7+
const PeerId = require('peer-id')
8+
const { NotStartedError } = require('../errors')
79

810
/**
911
* @param {Object} config
@@ -14,54 +16,61 @@ module.exports = ({ peerId, network }) => {
1416
/**
1517
* @type {import('ipfs-core-types/src/root').API["id"]}
1618
*/
17-
async function id (_options = {}) { // eslint-disable-line require-await
18-
const id = peerId.toB58String()
19-
/** @type {Multiaddr[]} */
20-
let addresses = []
21-
/** @type {string[]} */
22-
let protocols = []
19+
async function id (options = {}) { // eslint-disable-line require-await
20+
if (options.peerId === peerId.toB58String()) {
21+
delete options.peerId
22+
}
2323

2424
const net = network.try()
2525

26-
if (net) {
27-
const { libp2p } = net
28-
// only available while the node is running
29-
addresses = libp2p.multiaddrs
30-
protocols = Array.from(libp2p.upgrader.protocols.keys())
26+
if (!net) {
27+
if (options.peerId) {
28+
throw new NotStartedError()
29+
}
30+
31+
const idStr = peerId.toB58String()
32+
33+
return {
34+
id: idStr,
35+
publicKey: uint8ArrayToString(peerId.pubKey.bytes, 'base64pad'),
36+
addresses: [],
37+
agentVersion: `js-ipfs/${pkgversion}`,
38+
protocolVersion: '9000',
39+
protocols: []
40+
}
3141
}
3242

43+
const id = options.peerId ? PeerId.createFromB58String(options.peerId.toString()) : peerId
44+
const { libp2p } = net
45+
46+
const publicKey = options.peerId ? libp2p.peerStore.keyBook.get(id) : id.pubKey
47+
const addresses = options.peerId ? libp2p.peerStore.addressBook.getMultiaddrsForPeer(id) : libp2p.multiaddrs
48+
const protocols = options.peerId ? libp2p.peerStore.protoBook.get(id) : Array.from(libp2p.upgrader.protocols.keys())
49+
const agentVersion = uint8ArrayToString(libp2p.peerStore.metadataBook.getValue(id, 'AgentVersion') || new Uint8Array())
50+
const protocolVersion = uint8ArrayToString(libp2p.peerStore.metadataBook.getValue(id, 'ProtocolVersion') || new Uint8Array())
51+
const idStr = id.toB58String()
52+
3353
return {
34-
id,
35-
publicKey: uint8ArrayToString(peerId.pubKey.bytes, 'base64pad'),
36-
addresses: addresses
54+
id: idStr,
55+
publicKey: uint8ArrayToString(publicKey.bytes, 'base64pad'),
56+
addresses: (addresses || [])
3757
.map(ma => {
3858
const str = ma.toString()
3959

4060
// some relay-style transports add our peer id to the ma for us
4161
// so don't double-add
42-
if (str.endsWith(`/p2p/${id}`)) {
62+
if (str.endsWith(`/p2p/${idStr}`)) {
4363
return str
4464
}
4565

46-
return `${str}/p2p/${id}`
66+
return `${str}/p2p/${idStr}`
4767
})
4868
.sort()
4969
.map(ma => new Multiaddr(ma)),
50-
agentVersion: `js-ipfs/${pkgversion}`,
51-
protocolVersion: '9000',
52-
protocols: protocols.sort()
70+
agentVersion,
71+
protocolVersion,
72+
protocols: (protocols || []).sort()
5373
}
5474
}
5575
return withTimeoutOption(id)
5676
}
57-
58-
/**
59-
* @typedef {object} ID
60-
* The Peer identity
61-
* @property {string} id - the Peer ID
62-
* @property {string} publicKey - the public key of the peer as a base64 encoded string
63-
* @property {Multiaddr[]} addresses - A list of multiaddrs this node is listening on
64-
* @property {string} agentVersion - The agent version
65-
* @property {string} protocolVersion - The supported protocol version
66-
* @property {string[]} protocols - The supported protocols
67-
*/

packages/ipfs-core/src/components/libp2p.js

+4
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@ const get = require('dlv')
44
const mergeOptions = require('merge-options')
55
const errCode = require('err-code')
66
const PubsubRouters = require('../runtime/libp2p-pubsub-routers-nodejs')
7+
const pkgversion = require('../../package.json').version
78

89
/**
910
* @typedef {Object} KeychainConfig
@@ -134,6 +135,9 @@ function getLibp2pOptions ({ options, config, datastore, keys, keychainConfig, p
134135
keychain: {
135136
datastore: keys,
136137
...keychainConfig
138+
},
139+
host: {
140+
agentVersion: `js-ipfs/${pkgversion}`
137141
}
138142
}
139143

packages/ipfs-daemon/package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -40,7 +40,7 @@
4040
"ipfs-http-server": "^0.4.0",
4141
"ipfs-utils": "^7.0.0",
4242
"just-safe-set": "^2.2.1",
43-
"libp2p": "^0.31.2",
43+
"libp2p": "^0.31.5",
4444
"libp2p-delegated-content-routing": "^0.10.0",
4545
"libp2p-delegated-peer-routing": "^0.9.0",
4646
"libp2p-webrtc-star": "^0.22.2",

packages/ipfs-http-client/src/id.js

+4-1
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,10 @@ module.exports = configure(api => {
1818
const res = await api.post('id', {
1919
timeout: options.timeout,
2020
signal: options.signal,
21-
searchParams: toUrlSearchParams(options),
21+
searchParams: toUrlSearchParams({
22+
arg: options.peerId ? options.peerId.toString() : undefined,
23+
...options
24+
}),
2225
headers: options.headers
2326
})
2427
const data = await res.json()

packages/ipfs-http-server/src/api/resources/id.js

+10-3
Original file line numberDiff line numberDiff line change
@@ -10,8 +10,13 @@ module.exports = {
1010
stripUnknown: true
1111
},
1212
query: Joi.object().keys({
13-
timeout: Joi.timeout()
13+
timeout: Joi.timeout(),
14+
peerId: Joi.string()
1415
})
16+
.rename('arg', 'peerId', {
17+
override: true,
18+
ignoreUndefined: true
19+
})
1520
}
1621
},
1722
/**
@@ -29,13 +34,15 @@ module.exports = {
2934
}
3035
},
3136
query: {
32-
timeout
37+
timeout,
38+
peerId
3339
}
3440
} = request
3541

3642
const id = await ipfs.id({
3743
signal,
38-
timeout
44+
timeout,
45+
peerId
3946
})
4047
return h.response({
4148
ID: id.id,

packages/ipfs-http-server/test/inject/id.js

+24-1
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,8 @@ const { AbortSignal } = require('native-abort-controller')
99

1010
const defaultOptions = {
1111
signal: sinon.match.instanceOf(AbortSignal),
12-
timeout: undefined
12+
timeout: undefined,
13+
peerId: undefined
1314
}
1415

1516
describe('/id', () => {
@@ -65,4 +66,26 @@ describe('/id', () => {
6566

6667
expect(res).to.have.property('statusCode', 200)
6768
})
69+
70+
it('get the id of another peer', async () => {
71+
const peerId = 'QmSnuWmxptJZdLJpKRarxBMS2Ju2oANVrgbr2xWbie9b2D'
72+
73+
ipfs.id.withArgs({
74+
...defaultOptions,
75+
peerId
76+
}).returns({
77+
id: 'id',
78+
publicKey: 'publicKey',
79+
addresses: 'addresses',
80+
agentVersion: 'agentVersion',
81+
protocolVersion: 'protocolVersion'
82+
})
83+
84+
const res = await http({
85+
method: 'POST',
86+
url: `/api/v0/id?peerId=${peerId}`
87+
}, { ipfs })
88+
89+
expect(res).to.have.property('statusCode', 200)
90+
})
6891
})

0 commit comments

Comments
 (0)