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

Commit bffe080

Browse files
authored
feat: preload content (#1464)
refs #1459 This PR adds a new config property `preload`: ```js new IPFS({ preload: { enabled: false, addresses: ['/multiaddr/api/address'] } }) ``` * `preload.enabled` (default `false`) enables/disabled preloading - **should the default be false?** * `preload.addresses` array of node API addresses to preload content on. This are the addresses we make a `/api/v0/refs?arg=QmHash` request to, to initiate the preload **This PR upgrades the following APIs to preload content**. After adding content with `ipfs.files.add` (for example), we make a request to the first preload gateway addresses (providing `preload.enabled` is true), and will fall back to the second etc. * [x] `dag.put` * [x] `block.put` * [x] `object.new` * [x] `object.put` * [x] `object.patch.*` * [x] `mfs.*` MFS preloading is slightly different - we periodically submit your MFS root to the preload nodes when it changes. NOTE: this PR adds an option to `dag`, `block` and `object` APIs allowing users to opt out of preloading by specifying `preload: false` in their options object. License: MIT Signed-off-by: Alan Shaw <alan@tableflip.io>
1 parent 45b80a0 commit bffe080

27 files changed

+824
-30
lines changed

.aegir.js

+27-4
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,11 @@
11
'use strict'
22

3-
const createServer = require('ipfsd-ctl').createServer
3+
const IPFSFactory = require('ipfsd-ctl')
4+
const parallel = require('async/parallel')
5+
const MockPreloadNode = require('./test/utils/mock-preload-node')
46

5-
const server = createServer()
7+
const ipfsdServer = IPFSFactory.createServer()
8+
const preloadNode = MockPreloadNode.createNode()
69

710
module.exports = {
811
webpack: {
@@ -21,9 +24,29 @@ module.exports = {
2124
singleRun: true
2225
},
2326
hooks: {
27+
node: {
28+
pre: (cb) => preloadNode.start(cb),
29+
post: (cb) => preloadNode.stop(cb)
30+
},
2431
browser: {
25-
pre: server.start.bind(server),
26-
post: server.stop.bind(server)
32+
pre: (cb) => {
33+
parallel([
34+
(cb) => {
35+
ipfsdServer.start()
36+
cb()
37+
},
38+
(cb) => preloadNode.start(cb)
39+
], cb)
40+
},
41+
post: (cb) => {
42+
parallel([
43+
(cb) => {
44+
ipfsdServer.stop()
45+
cb()
46+
},
47+
(cb) => preloadNode.stop(cb)
48+
], cb)
49+
}
2750
}
2851
}
2952
}

README.md

+3
Original file line numberDiff line numberDiff line change
@@ -231,6 +231,9 @@ Creates and returns an instance of an IPFS node. Use the `options` argument to s
231231
- `enabled` (boolean): Make this node a relay (other nodes can connect *through* it). (Default: `false`)
232232
- `active` (boolean): Make this an *active* relay node. Active relay nodes will attempt to dial a destination peer even if that peer is not yet connected to the relay. (Default: `false`)
233233

234+
- `preload` (object): Configure external nodes that will preload content added to this node
235+
- `enabled` (boolean): Enable content preloading (Default: `true`)
236+
- `addresses` (array): Multiaddr API addresses of nodes that should preload content. NOTE: nodes specified here should also be added to your node's bootstrap address list at `config.Boostrap`
234237
- `EXPERIMENTAL` (object): Enable and configure experimental features.
235238
- `pubsub` (boolean): Enable libp2p pub-sub. (Default: `false`)
236239
- `sharding` (boolean): Enable directory sharding. Directories that have many child objects will be represented by multiple DAG nodes instead of just one. It can improve lookup performance when a directory has several thousand files or more. (Default: `false`)

package.json

+2
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,7 @@
1111
"./src/core/components/init-assets.js": false,
1212
"./src/core/runtime/config-nodejs.js": "./src/core/runtime/config-browser.js",
1313
"./src/core/runtime/libp2p-nodejs.js": "./src/core/runtime/libp2p-browser.js",
14+
"./src/core/runtime/preload-nodejs.js": "./src/core/runtime/preload-browser.js",
1415
"./src/core/runtime/repo-nodejs.js": "./src/core/runtime/repo-browser.js",
1516
"./src/core/runtime/dns-nodejs.js": "./src/core/runtime/dns-browser.js",
1617
"./test/utils/create-repo-nodejs.js": "./test/utils/create-repo-browser.js",
@@ -140,6 +141,7 @@
140141
"mime-types": "^2.1.18",
141142
"mkdirp": "~0.5.1",
142143
"multiaddr": "^5.0.0",
144+
"multiaddr-to-uri": "^4.0.0",
143145
"multibase": "~0.4.0",
144146
"multihashes": "~0.4.13",
145147
"once": "^1.4.0",

src/core/components/block.js

+5
Original file line numberDiff line numberDiff line change
@@ -52,6 +52,11 @@ module.exports = function block (self) {
5252
if (err) {
5353
return cb(err)
5454
}
55+
56+
if (options.preload !== false) {
57+
self._preload(block.cid)
58+
}
59+
5560
cb(null, block)
5661
})
5762
], callback)

src/core/components/dag.js

+9-1
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,15 @@ module.exports = function dag (self) {
2424

2525
options = options.cid ? options : Object.assign({}, optionDefaults, options)
2626

27-
self._ipld.put(dagNode, options, callback)
27+
self._ipld.put(dagNode, options, (err, cid) => {
28+
if (err) return callback(err)
29+
30+
if (options.preload !== false) {
31+
self._preload(cid)
32+
}
33+
34+
callback(null, cid)
35+
})
2836
}),
2937

3038
get: promisify((cid, path, options, callback) => {

src/core/components/files.js

+15
Original file line numberDiff line numberDiff line change
@@ -89,6 +89,20 @@ function normalizeContent (opts, content) {
8989
})
9090
}
9191

92+
function preloadFile (self, opts, file) {
93+
const isRootFile = opts.wrapWithDirectory
94+
? file.path === ''
95+
: !file.path.includes('/')
96+
97+
const shouldPreload = isRootFile && !opts.onlyHash && opts.preload !== false
98+
99+
if (shouldPreload) {
100+
self._preload(file.hash)
101+
}
102+
103+
return file
104+
}
105+
92106
function pinFile (self, opts, file, cb) {
93107
// Pin a file if it is the root dir of a recursive add or the single file
94108
// of a direct add.
@@ -158,6 +172,7 @@ module.exports = function files (self) {
158172
pull.flatten(),
159173
importer(self._ipld, opts),
160174
pull.asyncMap(prepareFile.bind(null, self, opts)),
175+
pull.map(preloadFile.bind(null, self, opts)),
161176
pull.asyncMap(pinFile.bind(null, self, opts))
162177
)
163178
}

src/core/components/index.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -26,4 +26,4 @@ exports.dht = require('./dht')
2626
exports.dns = require('./dns')
2727
exports.key = require('./key')
2828
exports.stats = require('./stats')
29-
exports.mfs = require('ipfs-mfs/core')
29+
exports.mfs = require('./mfs')

src/core/components/mfs.js

+24
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
'use strict'
2+
3+
const promisify = require('promisify-es6')
4+
const mfs = require('ipfs-mfs/core')
5+
6+
module.exports = self => {
7+
const mfsSelf = Object.assign({}, self)
8+
9+
// A patched dag API to ensure preload doesn't happen for MFS operations
10+
mfsSelf.dag = Object.assign({}, self.dag, {
11+
put: promisify((node, opts, cb) => {
12+
if (typeof opts === 'function') {
13+
cb = opts
14+
opts = {}
15+
}
16+
17+
opts = Object.assign({}, opts, { preload: false })
18+
19+
return self.dag.put(node, opts, cb)
20+
})
21+
})
22+
23+
return mfs(mfsSelf, mfsSelf._options)
24+
}

src/core/components/object.js

+37-11
Original file line numberDiff line numberDiff line change
@@ -80,10 +80,17 @@ module.exports = function object (self) {
8080
if (err) {
8181
return cb(err)
8282
}
83-
self._ipld.put(node, {
84-
cid: new CID(node.multihash)
85-
}, (err) => {
86-
cb(err, node)
83+
84+
const cid = new CID(node.multihash)
85+
86+
self._ipld.put(node, { cid }, (err) => {
87+
if (err) return cb(err)
88+
89+
if (options.preload !== false) {
90+
self._preload(cid)
91+
}
92+
93+
cb(null, node)
8794
})
8895
})
8996
}
@@ -92,12 +99,20 @@ module.exports = function object (self) {
9299
}
93100

94101
return {
95-
new: promisify((template, callback) => {
102+
new: promisify((template, options, callback) => {
96103
if (typeof template === 'function') {
97104
callback = template
98105
template = undefined
106+
options = {}
107+
}
108+
109+
if (typeof options === 'function') {
110+
callback = options
111+
options = {}
99112
}
100113

114+
options = options || {}
115+
101116
let data
102117

103118
if (template) {
@@ -111,13 +126,18 @@ module.exports = function object (self) {
111126
if (err) {
112127
return callback(err)
113128
}
114-
self._ipld.put(node, {
115-
cid: new CID(node.multihash)
116-
}, (err) => {
129+
130+
const cid = new CID(node.multihash)
131+
132+
self._ipld.put(node, { cid }, (err) => {
117133
if (err) {
118134
return callback(err)
119135
}
120136

137+
if (options.preload !== false) {
138+
self._preload(cid)
139+
}
140+
121141
callback(null, node)
122142
})
123143
})
@@ -166,13 +186,17 @@ module.exports = function object (self) {
166186
}
167187

168188
function next () {
169-
self._ipld.put(node, {
170-
cid: new CID(node.multihash)
171-
}, (err) => {
189+
const cid = new CID(node.multihash)
190+
191+
self._ipld.put(node, { cid }, (err) => {
172192
if (err) {
173193
return callback(err)
174194
}
175195

196+
if (options.preload !== false) {
197+
self._preload(cid)
198+
}
199+
176200
self.object.get(node.multihash, callback)
177201
})
178202
}
@@ -282,6 +306,8 @@ module.exports = function object (self) {
282306
editAndSave((node, cb) => {
283307
if (DAGLink.isDAGLink(linkRef)) {
284308
linkRef = linkRef._name
309+
} else if (linkRef && linkRef.name) {
310+
linkRef = linkRef.name
285311
}
286312
DAGNode.rmLink(node, linkRef, cb)
287313
})(multihash, options, callback)

src/core/components/pin-set.js

+3-2
Original file line numberDiff line numberDiff line change
@@ -90,7 +90,7 @@ exports = module.exports = function (dag) {
9090

9191
pinSet.storeItems(pins, (err, rootNode) => {
9292
if (err) { return callback(err) }
93-
const opts = { cid: new CID(rootNode.multihash) }
93+
const opts = { cid: new CID(rootNode.multihash), preload: false }
9494
dag.put(rootNode, opts, (err, cid) => {
9595
if (err) { return callback(err) }
9696
callback(null, rootNode)
@@ -168,7 +168,8 @@ exports = module.exports = function (dag) {
168168
function storeChild (err, child, binIdx, cb) {
169169
if (err) { return cb(err) }
170170

171-
dag.put(child, { cid: new CID(child._multihash) }, err => {
171+
const opts = { cid: new CID(child._multihash), preload: false }
172+
dag.put(child, opts, err => {
172173
if (err) { return cb(err) }
173174
fanoutLinks[binIdx] = new DAGLink('', child.size, child.multihash)
174175
cb(null)

src/core/components/pin.js

+2-2
Original file line numberDiff line numberDiff line change
@@ -80,14 +80,14 @@ module.exports = (self) => {
8080
// the pin-set nodes link to a special 'empty' node, so make sure it exists
8181
cb => DAGNode.create(Buffer.alloc(0), (err, empty) => {
8282
if (err) { return cb(err) }
83-
dag.put(empty, { cid: new CID(empty.multihash) }, cb)
83+
dag.put(empty, { cid: new CID(empty.multihash), preload: false }, cb)
8484
}),
8585

8686
// create a root node with DAGLinks to the direct and recursive DAGs
8787
cb => DAGNode.create(Buffer.alloc(0), [dLink, rLink], (err, node) => {
8888
if (err) { return cb(err) }
8989
root = node
90-
dag.put(root, { cid: new CID(root.multihash) }, cb)
90+
dag.put(root, { cid: new CID(root.multihash), preload: false }, cb)
9191
}),
9292

9393
// hack for CLI tests

src/core/components/start.js

+3-1
Original file line numberDiff line numberDiff line change
@@ -42,7 +42,9 @@ module.exports = (self) => {
4242

4343
self._bitswap.start()
4444
self._blockService.setExchange(self._bitswap)
45-
cb()
45+
46+
self._preload.start()
47+
self._mfsPreload.start(cb)
4648
}
4749
], done)
4850
})

src/core/components/stop.js

+2
Original file line numberDiff line numberDiff line change
@@ -30,8 +30,10 @@ module.exports = (self) => {
3030
self.state.stop()
3131
self._blockService.unsetExchange()
3232
self._bitswap.stop()
33+
self._preload.stop()
3334

3435
series([
36+
(cb) => self._mfsPreload.stop(cb),
3537
(cb) => self.libp2p.stop(cb),
3638
(cb) => self._repo.close(cb)
3739
], done)

src/core/config.js

+4
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,10 @@ const schema = Joi.object().keys({
88
Joi.string()
99
).allow(null),
1010
repoOwner: Joi.boolean().default(true),
11+
preload: Joi.object().keys({
12+
enabled: Joi.boolean().default(true),
13+
addresses: Joi.array().items(Joi.multiaddr().options({ convert: false }))
14+
}).allow(null),
1115
init: Joi.alternatives().try(
1216
Joi.boolean(),
1317
Joi.object().keys({ bits: Joi.number().integer() })

src/core/index.js

+13-2
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,8 @@ const boot = require('./boot')
2222
const components = require('./components')
2323
// replaced by repo-browser when running in the browser
2424
const defaultRepo = require('./runtime/repo-nodejs')
25+
const preload = require('./preload')
26+
const mfsPreload = require('./mfs-preload')
2527

2628
class IPFS extends EventEmitter {
2729
constructor (options) {
@@ -30,7 +32,14 @@ class IPFS extends EventEmitter {
3032
this._options = {
3133
init: true,
3234
start: true,
33-
EXPERIMENTAL: {}
35+
EXPERIMENTAL: {},
36+
preload: {
37+
enabled: true,
38+
addresses: [
39+
'/dnsaddr/node0.preload.ipfs.io/https',
40+
'/dnsaddr/node1.preload.ipfs.io/https'
41+
]
42+
}
3443
}
3544

3645
options = config.validate(options || {})
@@ -78,6 +87,8 @@ class IPFS extends EventEmitter {
7887
this._blockService = new BlockService(this._repo)
7988
this._ipld = new Ipld(this._blockService)
8089
this._pubsub = undefined
90+
this._preload = preload(this)
91+
this._mfsPreload = mfsPreload(this)
8192

8293
// IPFS Core exposed components
8394
// - for booting up a node
@@ -134,7 +145,7 @@ class IPFS extends EventEmitter {
134145
}
135146

136147
// ipfs.files
137-
const mfs = components.mfs(this, this._options)
148+
const mfs = components.mfs(this)
138149

139150
Object.keys(mfs).forEach(key => {
140151
this.files[key] = mfs[key]

0 commit comments

Comments
 (0)