From 5c5f891251cacf2c85cf87837227a4ba2602882c Mon Sep 17 00:00:00 2001 From: Marcin Rataj Date: Fri, 26 Jul 2019 16:43:26 +0200 Subject: [PATCH 01/23] fix: limit concurrent HTTP requests in browser Adds limit of concurrent HTTP requests sent to remote API by dns and preload calls when running in web browser contexts. Browsers limit connections per host (~6). This change mitigates the problem of expensive and long running calls of one type exhausting connection pool for other uses. It additionally limits the number of DNS lookup calls by introducing time-bound cache with eviction rules following what browser already do. This is similar to: https://github.com/libp2p/js-libp2p-delegated-content-routing/issues/12 License: MIT Signed-off-by: Marcin Rataj --- package.json | 1 + src/core/runtime/dns-browser.js | 44 ++++++++++++++++++++++------- src/core/runtime/preload-browser.js | 8 +++++- 3 files changed, 42 insertions(+), 11 deletions(-) diff --git a/package.json b/package.json index e7cfec839b..dd996b786d 100644 --- a/package.json +++ b/package.json @@ -150,6 +150,7 @@ "multihashes": "~0.4.14", "multihashing-async": "~0.6.0", "node-fetch": "^2.3.0", + "p-queue": "^6.1.0", "peer-book": "~0.9.0", "peer-id": "~0.12.3", "peer-info": "~0.15.0", diff --git a/src/core/runtime/dns-browser.js b/src/core/runtime/dns-browser.js index 74d0a14cdb..7b2725e491 100644 --- a/src/core/runtime/dns-browser.js +++ b/src/core/runtime/dns-browser.js @@ -1,33 +1,57 @@ -/* global self */ +/* eslint-env browser */ 'use strict' +const TLRU = require('../../utils/tlru') +const { default: PQueue } = require('p-queue') + +// Avoid sending multiple queries for the same hostname by caching results +const cache = new TLRU(1000) +// TODO: /api/v0/dns does not return TTL yet: https://github.com/ipfs/go-ipfs/issues/5884 +// However we know browsers themselves cache DNS records for at least 1 minute, +// which acts a provisional default ttl: https://stackoverflow.com/a/36917902/11518426 +const ttl = 60 * 1000 + +// browsers limit concurrent connections per host, +// we don't want preload calls to exhaust the limit (~6) +const _httpQueue = new PQueue({ concurrency: 4 }) + +function unpackResponse (domain, response, callback) { + if (response.Path) { + return callback(null, response.Path) + } else { + const err = new Error(response.Message) + return callback(err) + } +} + module.exports = (domain, opts, callback) => { if (typeof opts === 'function') { callback = opts opts = {} } - opts = opts || {} - domain = encodeURIComponent(domain) - let url = `https://ipfs.io/api/v0/dns?arg=${domain}` + if (cache.has(domain)) { + const response = cache.get(domain) + return unpackResponse(domain, response, callback) + } + + let url = `https://ipfs.io/api/v0/dns?arg=${domain}` Object.keys(opts).forEach(prop => { url += `&${encodeURIComponent(prop)}=${encodeURIComponent(opts[prop])}` }) - self.fetch(url, { mode: 'cors' }) + _httpQueue.add(() => fetch(url, { mode: 'cors' }) .then((response) => { return response.json() }) .then((response) => { - if (response.Path) { - return callback(null, response.Path) - } else { - return callback(new Error(response.Message)) - } + cache.set(domain, response, ttl) + return unpackResponse(domain, response, callback) }) .catch((error) => { callback(error) }) + ) } diff --git a/src/core/runtime/preload-browser.js b/src/core/runtime/preload-browser.js index 81407f483f..d24c7f3eff 100644 --- a/src/core/runtime/preload-browser.js +++ b/src/core/runtime/preload-browser.js @@ -1,18 +1,23 @@ /* eslint-env browser */ 'use strict' +const { default: PQueue } = require('p-queue') const debug = require('debug') const log = debug('ipfs:preload') log.error = debug('ipfs:preload:error') +// browsers limit concurrent connections per host, +// we don't want preload calls to exhaust the limit (~6) +const _httpQueue = new PQueue({ concurrency: 4 }) + module.exports = function preload (url, callback) { log(url) const controller = new AbortController() const signal = controller.signal - fetch(url, { signal }) + _httpQueue.add(() => fetch(url, { signal }) .then(res => { if (!res.ok) { log.error('failed to preload', url, res.status, res.statusText) @@ -22,6 +27,7 @@ module.exports = function preload (url, callback) { }) .then(() => callback()) .catch(callback) + ).catch(callback) return { cancel: () => controller.abort() From 45465ffd1b449065251904dd88023904b696536e Mon Sep 17 00:00:00 2001 From: Marcin Rataj Date: Fri, 26 Jul 2019 23:44:57 +0200 Subject: [PATCH 02/23] fix: undefined in TLUR.get This fixes undefined error when getting value for a key that is not in cache. License: MIT Signed-off-by: Marcin Rataj --- src/utils/tlru.js | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/src/utils/tlru.js b/src/utils/tlru.js index ba3b26e8c6..569015c038 100644 --- a/src/utils/tlru.js +++ b/src/utils/tlru.js @@ -33,8 +33,9 @@ class TLRU { this.lru.remove(key) return undefined } + return value.value } - return value.value + return undefined } /** From 7e422e9b7ed0dce06fab6adbb7d420d6e03b5c86 Mon Sep 17 00:00:00 2001 From: Marcin Rataj Date: Sat, 27 Jul 2019 13:01:35 +0200 Subject: [PATCH 03/23] fix: dns cache per domain+opts - `opts` impact returned value, so we cache per domain+opts - skip local cache when `nocache=true` License: MIT Signed-off-by: Marcin Rataj --- src/core/runtime/dns-browser.js | 12 +++++++++--- 1 file changed, 9 insertions(+), 3 deletions(-) diff --git a/src/core/runtime/dns-browser.js b/src/core/runtime/dns-browser.js index 7b2725e491..5d8794ddd1 100644 --- a/src/core/runtime/dns-browser.js +++ b/src/core/runtime/dns-browser.js @@ -32,11 +32,17 @@ module.exports = (domain, opts, callback) => { opts = opts || {} domain = encodeURIComponent(domain) - if (cache.has(domain)) { - const response = cache.get(domain) + // `opts` impact returned value, so we cache per domain+opts + const query = `${domain}${JSON.stringify(opts)}` + + // try cache first + if (!opts.nocache && cache.has(query)) { + const response = cache.get(query) return unpackResponse(domain, response, callback) } + // fallback to sending DNSLink query to ipfs.io + // TODO: replace this with generic DNS over HTTPS: https://github.com/ipfs/js-ipfs/issues/2212 let url = `https://ipfs.io/api/v0/dns?arg=${domain}` Object.keys(opts).forEach(prop => { url += `&${encodeURIComponent(prop)}=${encodeURIComponent(opts[prop])}` @@ -47,7 +53,7 @@ module.exports = (domain, opts, callback) => { return response.json() }) .then((response) => { - cache.set(domain, response, ttl) + cache.set(query, response, ttl) return unpackResponse(domain, response, callback) }) .catch((error) => { From 717198171c2b659f8a6fa964e28b96bba98ba238 Mon Sep 17 00:00:00 2001 From: Marcin Rataj Date: Tue, 30 Jul 2019 13:06:08 +0200 Subject: [PATCH 04/23] refactor: simplify error handling License: MIT Signed-off-by: Marcin Rataj --- src/core/runtime/preload-browser.js | 4 +--- 1 file changed, 1 insertion(+), 3 deletions(-) diff --git a/src/core/runtime/preload-browser.js b/src/core/runtime/preload-browser.js index d24c7f3eff..471d667a91 100644 --- a/src/core/runtime/preload-browser.js +++ b/src/core/runtime/preload-browser.js @@ -25,9 +25,7 @@ module.exports = function preload (url, callback) { } return res.text() }) - .then(() => callback()) - .catch(callback) - ).catch(callback) + ).then(() => callback(), callback) return { cancel: () => controller.abort() From 6c404eab456d94cdf36e3efaa6faf6ad10068454 Mon Sep 17 00:00:00 2001 From: Marcin Rataj Date: Tue, 30 Jul 2019 19:59:35 +0200 Subject: [PATCH 05/23] chore: increase bundlesize maxSize This PR adds p-queue and hits maxSize limit, however the library will be included by delegated content/peer modules anyway. License: MIT Signed-off-by: Marcin Rataj --- .aegir.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.aegir.js b/.aegir.js index 6d0ed4b2dd..60f97ab578 100644 --- a/.aegir.js +++ b/.aegir.js @@ -8,7 +8,7 @@ const ipfsdServer = IPFSFactory.createServer() const preloadNode = MockPreloadNode.createNode() module.exports = { - bundlesize: { maxSize: '689kB' }, + bundlesize: { maxSize: '690kB' }, webpack: { resolve: { mainFields: ['browser', 'main'], From 3ea284d7a8c688f42471f8e68bc556c185167f2f Mon Sep 17 00:00:00 2001 From: Marcin Rataj Date: Tue, 30 Jul 2019 21:52:13 +0200 Subject: [PATCH 06/23] refactor: call callbacks inside setImmediate License: MIT Signed-off-by: Marcin Rataj --- src/core/runtime/dns-browser.js | 14 ++++---------- src/core/runtime/preload-browser.js | 4 ++-- 2 files changed, 6 insertions(+), 12 deletions(-) diff --git a/src/core/runtime/dns-browser.js b/src/core/runtime/dns-browser.js index 5d8794ddd1..93d5b9c176 100644 --- a/src/core/runtime/dns-browser.js +++ b/src/core/runtime/dns-browser.js @@ -18,10 +18,8 @@ const _httpQueue = new PQueue({ concurrency: 4 }) function unpackResponse (domain, response, callback) { if (response.Path) { return callback(null, response.Path) - } else { - const err = new Error(response.Message) - return callback(err) } + return callback(new Error(response.Message)) } module.exports = (domain, opts, callback) => { @@ -49,15 +47,11 @@ module.exports = (domain, opts, callback) => { }) _httpQueue.add(() => fetch(url, { mode: 'cors' }) - .then((response) => { - return response.json() - }) + .then((response) => response.json()) .then((response) => { cache.set(query, response, ttl) - return unpackResponse(domain, response, callback) - }) - .catch((error) => { - callback(error) + setImmediate(() => unpackResponse(domain, response, callback)) }) + .catch((err) => setImmediate(() => callback(err))) ) } diff --git a/src/core/runtime/preload-browser.js b/src/core/runtime/preload-browser.js index 471d667a91..c09e716160 100644 --- a/src/core/runtime/preload-browser.js +++ b/src/core/runtime/preload-browser.js @@ -23,9 +23,9 @@ module.exports = function preload (url, callback) { log.error('failed to preload', url, res.status, res.statusText) throw new Error(`failed to preload ${url}`) } - return res.text() + setImmediate(callback) }) - ).then(() => callback(), callback) + ).catch((err) => setImmediate(() => callback(err))) return { cancel: () => controller.abort() From d5fd304970e1f93600267ce67fe58ad2bffbdb4c Mon Sep 17 00:00:00 2001 From: Marcin Rataj Date: Tue, 30 Jul 2019 23:19:55 +0200 Subject: [PATCH 07/23] refactor: replace custom fetch with ky-universal This change simplifies code related to ad-hoc HTTP requests by switching from custom code to ky-universal. It is a drop-in replacement for fetch that works in browser and node that retries failed request License: MIT Signed-off-by: Marcin Rataj --- .aegir.js | 2 +- package.json | 4 ++-- .../components/files-regular/add-from-url.js | 7 +++++-- src/core/runtime/dns-browser.js | 14 ++++++-------- src/core/runtime/fetch-browser.js | 3 --- src/core/runtime/fetch-nodejs.js | 2 -- src/core/runtime/preload-browser.js | 18 +++++++++--------- 7 files changed, 23 insertions(+), 27 deletions(-) delete mode 100644 src/core/runtime/fetch-browser.js delete mode 100644 src/core/runtime/fetch-nodejs.js diff --git a/.aegir.js b/.aegir.js index 60f97ab578..74d6e16c2a 100644 --- a/.aegir.js +++ b/.aegir.js @@ -8,7 +8,7 @@ const ipfsdServer = IPFSFactory.createServer() const preloadNode = MockPreloadNode.createNode() module.exports = { - bundlesize: { maxSize: '690kB' }, + bundlesize: { maxSize: '692kB' }, webpack: { resolve: { mainFields: ['browser', 'main'], diff --git a/package.json b/package.json index dd996b786d..1bdb2d5489 100644 --- a/package.json +++ b/package.json @@ -19,7 +19,6 @@ "./src/core/runtime/add-from-fs-nodejs.js": "./src/core/runtime/add-from-fs-browser.js", "./src/core/runtime/config-nodejs.js": "./src/core/runtime/config-browser.js", "./src/core/runtime/dns-nodejs.js": "./src/core/runtime/dns-browser.js", - "./src/core/runtime/fetch-nodejs.js": "./src/core/runtime/fetch-browser.js", "./src/core/runtime/libp2p-nodejs.js": "./src/core/runtime/libp2p-browser.js", "./src/core/runtime/libp2p-pubsub-routers-nodejs.js": "./src/core/runtime/libp2p-pubsub-routers-browser.js", "./src/core/runtime/preload-nodejs.js": "./src/core/runtime/preload-browser.js", @@ -121,6 +120,8 @@ "iso-url": "~0.4.6", "just-safe-set": "^2.1.0", "kind-of": "^6.0.2", + "ky": "~0.13.0", + "ky-universal": "~0.3.0", "libp2p": "~0.26.1", "libp2p-bootstrap": "~0.9.3", "libp2p-crypto": "~0.16.0", @@ -149,7 +150,6 @@ "multicodec": "~0.5.5", "multihashes": "~0.4.14", "multihashing-async": "~0.6.0", - "node-fetch": "^2.3.0", "p-queue": "^6.1.0", "peer-book": "~0.9.0", "peer-id": "~0.12.3", diff --git a/src/core/components/files-regular/add-from-url.js b/src/core/components/files-regular/add-from-url.js index 73697aa2cf..31a653540d 100644 --- a/src/core/components/files-regular/add-from-url.js +++ b/src/core/components/files-regular/add-from-url.js @@ -1,7 +1,7 @@ 'use strict' const { URL } = require('iso-url') -const fetch = require('../../runtime/fetch-nodejs') +const { default: ky } = require('ky-universal') module.exports = (self) => { return async (url, options, callback) => { @@ -14,7 +14,10 @@ module.exports = (self) => { try { const parsedUrl = new URL(url) - const res = await fetch(url) + const res = await ky(url, { + timeout: 15000, + retry: 3 + }) if (!res.ok) { throw new Error('unexpected status code: ' + res.status) diff --git a/src/core/runtime/dns-browser.js b/src/core/runtime/dns-browser.js index 93d5b9c176..bcca14e64e 100644 --- a/src/core/runtime/dns-browser.js +++ b/src/core/runtime/dns-browser.js @@ -3,6 +3,7 @@ const TLRU = require('../../utils/tlru') const { default: PQueue } = require('p-queue') +const { default: ky } = require('ky-universal') // Avoid sending multiple queries for the same hostname by caching results const cache = new TLRU(1000) @@ -46,12 +47,9 @@ module.exports = (domain, opts, callback) => { url += `&${encodeURIComponent(prop)}=${encodeURIComponent(opts[prop])}` }) - _httpQueue.add(() => fetch(url, { mode: 'cors' }) - .then((response) => response.json()) - .then((response) => { - cache.set(query, response, ttl) - setImmediate(() => unpackResponse(domain, response, callback)) - }) - .catch((err) => setImmediate(() => callback(err))) - ) + _httpQueue.add(async () => { + const response = await ky(url, { mode: 'cors' }).json() + cache.set(query, response, ttl) + setImmediate(() => unpackResponse(domain, response, callback)) + }).catch((err) => setImmediate(() => callback(err))) } diff --git a/src/core/runtime/fetch-browser.js b/src/core/runtime/fetch-browser.js deleted file mode 100644 index b2f604db7c..0000000000 --- a/src/core/runtime/fetch-browser.js +++ /dev/null @@ -1,3 +0,0 @@ -/* eslint-env browser */ -'use strict' -module.exports = fetch diff --git a/src/core/runtime/fetch-nodejs.js b/src/core/runtime/fetch-nodejs.js deleted file mode 100644 index ca77ad8310..0000000000 --- a/src/core/runtime/fetch-nodejs.js +++ /dev/null @@ -1,2 +0,0 @@ -'use strict' -module.exports = require('node-fetch') diff --git a/src/core/runtime/preload-browser.js b/src/core/runtime/preload-browser.js index c09e716160..a9d7ecae80 100644 --- a/src/core/runtime/preload-browser.js +++ b/src/core/runtime/preload-browser.js @@ -2,6 +2,7 @@ 'use strict' const { default: PQueue } = require('p-queue') +const { default: ky } = require('ky-universal') const debug = require('debug') const log = debug('ipfs:preload') @@ -17,15 +18,14 @@ module.exports = function preload (url, callback) { const controller = new AbortController() const signal = controller.signal - _httpQueue.add(() => fetch(url, { signal }) - .then(res => { - if (!res.ok) { - log.error('failed to preload', url, res.status, res.statusText) - throw new Error(`failed to preload ${url}`) - } - setImmediate(callback) - }) - ).catch((err) => setImmediate(() => callback(err))) + _httpQueue.add(async () => { + const res = await ky(url, { signal }) + if (!res.ok) { + log.error('failed to preload', url, res.status, res.statusText) + throw new Error(`failed to preload ${url}`) + } + setImmediate(callback) + }).catch((err) => setImmediate(() => callback(err))) return { cancel: () => controller.abort() From a6d09c197377fddea56445e48aad41f0a342f228 Mon Sep 17 00:00:00 2001 From: Marcin Rataj Date: Wed, 31 Jul 2019 15:54:40 +0200 Subject: [PATCH 08/23] refactor: simplify http client code with ky License: MIT Signed-off-by: Marcin Rataj --- .../components/files-regular/add-from-url.js | 16 +------- src/core/runtime/dns-browser.js | 38 +++++++++++-------- src/core/runtime/preload-browser.js | 6 +-- 3 files changed, 26 insertions(+), 34 deletions(-) diff --git a/src/core/components/files-regular/add-from-url.js b/src/core/components/files-regular/add-from-url.js index 31a653540d..148250ad22 100644 --- a/src/core/components/files-regular/add-from-url.js +++ b/src/core/components/files-regular/add-from-url.js @@ -11,22 +11,10 @@ module.exports = (self) => { } let files - try { - const parsedUrl = new URL(url) - const res = await ky(url, { - timeout: 15000, - retry: 3 - }) - - if (!res.ok) { - throw new Error('unexpected status code: ' + res.status) - } - - // TODO: use res.body when supported + const res = await ky.get(url) + const path = decodeURIComponent(new URL(res.url).pathname.split('/').pop()) const content = Buffer.from(await res.arrayBuffer()) - const path = decodeURIComponent(parsedUrl.pathname.split('/').pop()) - files = await self.add({ content, path }, options) } catch (err) { if (callback) { diff --git a/src/core/runtime/dns-browser.js b/src/core/runtime/dns-browser.js index bcca14e64e..0a202d3ef6 100644 --- a/src/core/runtime/dns-browser.js +++ b/src/core/runtime/dns-browser.js @@ -16,7 +16,23 @@ const ttl = 60 * 1000 // we don't want preload calls to exhaust the limit (~6) const _httpQueue = new PQueue({ concurrency: 4 }) -function unpackResponse (domain, response, callback) { +// Delegated HTTP resolver sending DNSLink queries to ipfs.io +// TODO: replace hardcoded host with configurable DNS over HTTPS: https://github.com/ipfs/js-ipfs/issues/2212 +const api = ky.create({ + prefixUrl: 'https://ipfs.io/api/v0/', + hooks: { + afterResponse: [ + async response => { + const query = new URLSearchParams(new URL(response.url).search).toString() + const json = await response.json() + cache.set(query, json, ttl) + return json + } + ] + } +}) + +function unpackResponse (response, callback) { if (response.Path) { return callback(null, response.Path) } @@ -29,27 +45,19 @@ module.exports = (domain, opts, callback) => { opts = {} } opts = opts || {} - domain = encodeURIComponent(domain) - // `opts` impact returned value, so we cache per domain+opts - const query = `${domain}${JSON.stringify(opts)}` + const searchParams = new URLSearchParams(opts) + searchParams.set('arg', domain) // try cache first + const query = searchParams.toString() if (!opts.nocache && cache.has(query)) { const response = cache.get(query) - return unpackResponse(domain, response, callback) + return setImmediate(() => unpackResponse(response, callback)) } - // fallback to sending DNSLink query to ipfs.io - // TODO: replace this with generic DNS over HTTPS: https://github.com/ipfs/js-ipfs/issues/2212 - let url = `https://ipfs.io/api/v0/dns?arg=${domain}` - Object.keys(opts).forEach(prop => { - url += `&${encodeURIComponent(prop)}=${encodeURIComponent(opts[prop])}` - }) - _httpQueue.add(async () => { - const response = await ky(url, { mode: 'cors' }).json() - cache.set(query, response, ttl) - setImmediate(() => unpackResponse(domain, response, callback)) + const response = await api.get('dns', { searchParams }) + setImmediate(() => unpackResponse(response, callback)) }).catch((err) => setImmediate(() => callback(err))) } diff --git a/src/core/runtime/preload-browser.js b/src/core/runtime/preload-browser.js index a9d7ecae80..ca7eb188f8 100644 --- a/src/core/runtime/preload-browser.js +++ b/src/core/runtime/preload-browser.js @@ -19,11 +19,7 @@ module.exports = function preload (url, callback) { const signal = controller.signal _httpQueue.add(async () => { - const res = await ky(url, { signal }) - if (!res.ok) { - log.error('failed to preload', url, res.status, res.statusText) - throw new Error(`failed to preload ${url}`) - } + await ky.get(url, { signal }) setImmediate(callback) }).catch((err) => setImmediate(() => callback(err))) From 1999de7364a1cfca40e725718231fb87d9f927bf Mon Sep 17 00:00:00 2001 From: Marcin Rataj Date: Wed, 31 Jul 2019 16:14:01 +0200 Subject: [PATCH 09/23] refactor: promise first addFromURL License: MIT Signed-off-by: Marcin Rataj --- .../components/files-regular/add-from-url.js | 36 +++++++------------ 1 file changed, 13 insertions(+), 23 deletions(-) diff --git a/src/core/components/files-regular/add-from-url.js b/src/core/components/files-regular/add-from-url.js index 148250ad22..eb9e5836a6 100644 --- a/src/core/components/files-regular/add-from-url.js +++ b/src/core/components/files-regular/add-from-url.js @@ -2,31 +2,21 @@ const { URL } = require('iso-url') const { default: ky } = require('ky-universal') +const nodeify = require('promise-nodeify') -module.exports = (self) => { - return async (url, options, callback) => { - if (typeof options === 'function') { - callback = options - options = {} - } - - let files - try { - const res = await ky.get(url) - const path = decodeURIComponent(new URL(res.url).pathname.split('/').pop()) - const content = Buffer.from(await res.arrayBuffer()) - files = await self.add({ content, path }, options) - } catch (err) { - if (callback) { - return callback(err) - } - throw err - } +module.exports = (ipfs) => { + const addFromURL = async (url, opts = {}) => { + const res = await ky.get(url) + const path = decodeURIComponent(new URL(res.url).pathname.split('/').pop()) + const content = Buffer.from(await res.arrayBuffer()) + return ipfs.add({ content, path }, opts) + } - if (callback) { - callback(null, files) + return (name, opts = {}, cb) => { + if (typeof opts === 'function') { + cb = opts + opts = {} } - - return files + return nodeify(addFromURL(name, opts), cb) } } From c4d0a83d4e4dde70561902ac3056ed05338d8bb3 Mon Sep 17 00:00:00 2001 From: Marcin Rataj Date: Wed, 31 Jul 2019 20:25:41 +0200 Subject: [PATCH 10/23] refactor: promise first dns in browser License: MIT Signed-off-by: Marcin Rataj --- src/core/runtime/dns-browser.js | 43 ++++++++++++++++----------------- 1 file changed, 21 insertions(+), 22 deletions(-) diff --git a/src/core/runtime/dns-browser.js b/src/core/runtime/dns-browser.js index 0a202d3ef6..ab96e88867 100644 --- a/src/core/runtime/dns-browser.js +++ b/src/core/runtime/dns-browser.js @@ -4,6 +4,7 @@ const TLRU = require('../../utils/tlru') const { default: PQueue } = require('p-queue') const { default: ky } = require('ky-universal') +const nodeify = require('promise-nodeify') // Avoid sending multiple queries for the same hostname by caching results const cache = new TLRU(1000) @@ -26,38 +27,36 @@ const api = ky.create({ const query = new URLSearchParams(new URL(response.url).search).toString() const json = await response.json() cache.set(query, json, ttl) - return json } ] } }) -function unpackResponse (response, callback) { - if (response.Path) { - return callback(null, response.Path) - } - return callback(new Error(response.Message)) +const ipfsPath = (response) => { + if (response.Path) return response.Path + throw new Error(response.Message) } -module.exports = (domain, opts, callback) => { +module.exports = (fqdn, opts = {}, cb) => { if (typeof opts === 'function') { - callback = opts + cb = opts opts = {} } - opts = opts || {} - - const searchParams = new URLSearchParams(opts) - searchParams.set('arg', domain) - - // try cache first - const query = searchParams.toString() - if (!opts.nocache && cache.has(query)) { - const response = cache.get(query) - return setImmediate(() => unpackResponse(response, callback)) + const resolveDnslink = async (fqdn, opts = {}) => { + const searchParams = new URLSearchParams(opts) + searchParams.set('arg', fqdn) + + // try cache first + const query = searchParams.toString() + if (!opts.nocache && cache.has(query)) { + const response = cache.get(query) + return ipfsPath(response) + } + + // fallback to delegated DNS resolver + const response = await _httpQueue.add(() => api.get('dns', { searchParams }).json()) + return ipfsPath(response) } - _httpQueue.add(async () => { - const response = await api.get('dns', { searchParams }) - setImmediate(() => unpackResponse(response, callback)) - }).catch((err) => setImmediate(() => callback(err))) + return nodeify(resolveDnslink(fqdn, opts), cb) } From a074cbb83e75b117b68b262062cc77fd65529fc8 Mon Sep 17 00:00:00 2001 From: Marcin Rataj Date: Thu, 1 Aug 2019 22:16:01 +0200 Subject: [PATCH 11/23] fix(ci): disable preload when NODE_ENV is 'test' Preload tests failed in browser because preload is enabled in browser context by default and while js-ipfs tests disable preload in tests, interface-ipfs-core does not. This means interface-ipfs-core tests triggered dozens of slow/hanging requests to `nodeX.preload.iofs.io/api/v0/refs` and js-ipfs executed only 4 at a time. By the time preload tests got executed, the queue was long and it did not finish before timeout. This change detects js-ipfs is running in test environment (`process.env.NODE_ENV === 'test'`) and disables preload by default. This way we are still be able to test preload by explicitly enabling it in config, but we won't have preload overhead when running unrelated interface-ipfs-core tests. License: MIT Signed-off-by: Marcin Rataj --- src/cli/commands/daemon.js | 2 +- src/core/config.js | 5 ++++- src/core/index.js | 2 +- 3 files changed, 6 insertions(+), 3 deletions(-) diff --git a/src/cli/commands/daemon.js b/src/cli/commands/daemon.js index f3f61f80e6..aac35d39c2 100644 --- a/src/cli/commands/daemon.js +++ b/src/cli/commands/daemon.js @@ -32,7 +32,7 @@ module.exports = { }) .option('enable-preload', { type: 'boolean', - default: true + default: process.env.NODE_ENV !== 'test' // preload by default, unless in test env }) }, diff --git a/src/core/config.js b/src/core/config.js index 53231bcc73..367535c562 100644 --- a/src/core/config.js +++ b/src/core/config.js @@ -31,7 +31,10 @@ const configSchema = s({ enabled: 'boolean?', addresses: optional(s(['multiaddr'])), interval: 'number?' - }, { enabled: true, interval: 30 * 1000 }), + }, { // defaults + enabled: process.env.NODE_ENV !== 'test', // preload by default, unless in test env + interval: 30 * 1000 + }), pubsub: optional(s({ enabled: 'boolean?' })), diff --git a/src/core/index.js b/src/core/index.js index 4016aa47fc..bdfc249f21 100644 --- a/src/core/index.js +++ b/src/core/index.js @@ -51,7 +51,7 @@ class IPFS extends EventEmitter { enabled: false }, preload: { - enabled: true, + enabled: process.env.NODE_ENV !== 'test', // preload by default, unless in test env addresses: [ '/dnsaddr/node0.preload.ipfs.io/https', '/dnsaddr/node1.preload.ipfs.io/https' From 4af85605d31ecd4045403f44318ddaa836bae07b Mon Sep 17 00:00:00 2001 From: Marcin Rataj Date: Fri, 2 Aug 2019 14:17:27 +0200 Subject: [PATCH 12/23] fix: use node-fetch in electron-renderer `fetch` needs to be polyfilled before importing ky-universal: ky-universal won't use node-fetch because electron-renderer already has global.fetch defined and we can't use the one from electron-renderer as it returns different errors and makes HTTPS mocking impossible in tests License: MIT Signed-off-by: Marcin Rataj --- src/core/components/files-regular/add-from-url.js | 11 ++++++++++- 1 file changed, 10 insertions(+), 1 deletion(-) diff --git a/src/core/components/files-regular/add-from-url.js b/src/core/components/files-regular/add-from-url.js index eb9e5836a6..8142db5d9b 100644 --- a/src/core/components/files-regular/add-from-url.js +++ b/src/core/components/files-regular/add-from-url.js @@ -1,8 +1,17 @@ 'use strict' const { URL } = require('iso-url') -const { default: ky } = require('ky-universal') const nodeify = require('promise-nodeify') +const { isElectronRenderer } = require('ipfs-utils/src/env') +if (isElectronRenderer) { + // `fetch` needs to be polyfilled before importing ky-universal: + // ky-universal won't use node-fetch because electron-renderer already has global.fetch defined + // and we can't use the one from electron-renderer as it returns different + // errors and makes HTTPS mocking impossible in tests + // TODO: remove when upstream fix lands + global.fetch = require('node-fetch') +} +const { default: ky } = require('ky-universal') module.exports = (ipfs) => { const addFromURL = async (url, opts = {}) => { From b5038ecdb460805e1122ca5db5f1007d4795703f Mon Sep 17 00:00:00 2001 From: Marcin Rataj Date: Wed, 28 Aug 2019 01:01:46 +0200 Subject: [PATCH 13/23] fix: enable tests of addFromURL in browser Context: https://github.com/ipfs/interface-js-ipfs-core/pull/514 License: MIT Signed-off-by: Marcin Rataj --- .aegir.js | 31 ++++++++----------- package.json | 2 +- .../components/files-regular/add-from-url.js | 9 ------ test/core/interface.spec.js | 3 -- 4 files changed, 14 insertions(+), 31 deletions(-) diff --git a/.aegir.js b/.aegir.js index 74d6e16c2a..cb8573ff79 100644 --- a/.aegir.js +++ b/.aegir.js @@ -3,9 +3,20 @@ const IPFSFactory = require('ipfsd-ctl') const parallel = require('async/parallel') const MockPreloadNode = require('./test/utils/mock-preload-node') +const EchoHttpServer = require('interface-ipfs-core/src/utils/echo-http-server') const ipfsdServer = IPFSFactory.createServer() const preloadNode = MockPreloadNode.createNode() +const httpEchoServer = EchoHttpServer.createServer() // used by addFromURL + +const batch = (call, done, ...srvs) => parallel(srvs.map(srv => cb => { + if (srv === ipfsdServer) { + srv[call]() + cb() + } else { + srv[call](cb) + } +}), done) module.exports = { bundlesize: { maxSize: '692kB' }, @@ -30,24 +41,8 @@ module.exports = { post: (cb) => preloadNode.stop(cb) }, browser: { - pre: (cb) => { - parallel([ - (cb) => { - ipfsdServer.start() - cb() - }, - (cb) => preloadNode.start(cb) - ], cb) - }, - post: (cb) => { - parallel([ - (cb) => { - ipfsdServer.stop() - cb() - }, - (cb) => preloadNode.stop(cb) - ], cb) - } + pre: (cb) => batch('start', cb, ipfsdServer, preloadNode, httpEchoServer), + post: (cb) => batch('stop', cb, ipfsdServer, preloadNode, httpEchoServer) } } } diff --git a/package.json b/package.json index 1bdb2d5489..fe20b09841 100644 --- a/package.json +++ b/package.json @@ -196,7 +196,7 @@ "form-data": "^2.5.1", "hat": "0.0.3", "ipfsd-ctl": "~0.45.0", - "interface-ipfs-core": "^0.111.1", + "interface-ipfs-core": "^0.113.0", "libp2p-websocket-star": "~0.10.2", "ncp": "^2.0.0", "p-event": "^4.1.0", diff --git a/src/core/components/files-regular/add-from-url.js b/src/core/components/files-regular/add-from-url.js index 8142db5d9b..c9207c98cc 100644 --- a/src/core/components/files-regular/add-from-url.js +++ b/src/core/components/files-regular/add-from-url.js @@ -2,15 +2,6 @@ const { URL } = require('iso-url') const nodeify = require('promise-nodeify') -const { isElectronRenderer } = require('ipfs-utils/src/env') -if (isElectronRenderer) { - // `fetch` needs to be polyfilled before importing ky-universal: - // ky-universal won't use node-fetch because electron-renderer already has global.fetch defined - // and we can't use the one from electron-renderer as it returns different - // errors and makes HTTPS mocking impossible in tests - // TODO: remove when upstream fix lands - global.fetch = require('node-fetch') -} const { default: ky } = require('ky-universal') module.exports = (ipfs) => { diff --git a/test/core/interface.spec.js b/test/core/interface.spec.js index 4da19bc846..66cc6966d1 100644 --- a/test/core/interface.spec.js +++ b/test/core/interface.spec.js @@ -64,9 +64,6 @@ describe('interface-ipfs-core tests', function () { }, { name: 'addFromFs', reason: 'Not designed to run in the browser' - }, { - name: 'addFromURL', - reason: 'Not designed to run in the browser' }] }) From 857769a5d4d8fe95e13775a0884c97539d607359 Mon Sep 17 00:00:00 2001 From: Marcin Rataj Date: Fri, 30 Aug 2019 12:14:41 +0200 Subject: [PATCH 14/23] refactor: run echo-http-server via aegir This simplifies code in interface-js-ipfs-core as both Node an Browser init it the same way Context: https://github.com/ipfs/interface-js-ipfs-core/pull/514#issuecomment-526225635 License: MIT Signed-off-by: Marcin Rataj --- .aegir.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/.aegir.js b/.aegir.js index cb8573ff79..b04d05facc 100644 --- a/.aegir.js +++ b/.aegir.js @@ -37,8 +37,8 @@ module.exports = { }, hooks: { node: { - pre: (cb) => preloadNode.start(cb), - post: (cb) => preloadNode.stop(cb) + pre: (cb) => batch('start', cb, preloadNode, httpEchoServer), + post: (cb) => batch('stop', cb, preloadNode, httpEchoServer) }, browser: { pre: (cb) => batch('start', cb, ipfsdServer, preloadNode, httpEchoServer), From 79ae97f1d8833a40934823141943f40b280a1521 Mon Sep 17 00:00:00 2001 From: Marcin Rataj Date: Wed, 4 Sep 2019 12:27:08 +0200 Subject: [PATCH 15/23] refactor: addressing review License: MIT Signed-off-by: Marcin Rataj --- src/core/runtime/dns-browser.js | 2 +- src/core/runtime/preload-browser.js | 5 +---- 2 files changed, 2 insertions(+), 5 deletions(-) diff --git a/src/core/runtime/dns-browser.js b/src/core/runtime/dns-browser.js index ab96e88867..91b275adf8 100644 --- a/src/core/runtime/dns-browser.js +++ b/src/core/runtime/dns-browser.js @@ -24,7 +24,7 @@ const api = ky.create({ hooks: { afterResponse: [ async response => { - const query = new URLSearchParams(new URL(response.url).search).toString() + const query = new URL(response.url).search.slice(1) const json = await response.json() cache.set(query, json, ttl) } diff --git a/src/core/runtime/preload-browser.js b/src/core/runtime/preload-browser.js index ca7eb188f8..a2c6d6cb2a 100644 --- a/src/core/runtime/preload-browser.js +++ b/src/core/runtime/preload-browser.js @@ -18,10 +18,7 @@ module.exports = function preload (url, callback) { const controller = new AbortController() const signal = controller.signal - _httpQueue.add(async () => { - await ky.get(url, { signal }) - setImmediate(callback) - }).catch((err) => setImmediate(() => callback(err))) + _httpQueue.add(() => ky.get(url, { signal })).then(() => callback(), callback) return { cancel: () => controller.abort() From 0f6b54363880cc6ddcccfb5e8cc3893f57a15c9c Mon Sep 17 00:00:00 2001 From: Marcin Rataj Date: Wed, 4 Sep 2019 12:42:34 +0200 Subject: [PATCH 16/23] refactor: ky v0.13.0 https://github.com/sindresorhus/ky/releases/tag/v0.13.0 License: MIT Signed-off-by: Marcin Rataj --- src/core/runtime/dns-browser.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/core/runtime/dns-browser.js b/src/core/runtime/dns-browser.js index 91b275adf8..b5ea4f97cf 100644 --- a/src/core/runtime/dns-browser.js +++ b/src/core/runtime/dns-browser.js @@ -23,7 +23,7 @@ const api = ky.create({ prefixUrl: 'https://ipfs.io/api/v0/', hooks: { afterResponse: [ - async response => { + async (input, options, response) => { const query = new URL(response.url).search.slice(1) const json = await response.json() cache.set(query, json, ttl) From e29e42bcc510fda11f2ae06c35ee0612f9fcf613 Mon Sep 17 00:00:00 2001 From: Marcin Rataj Date: Wed, 4 Sep 2019 13:23:10 +0200 Subject: [PATCH 17/23] fix: call callback via setImmediate https://github.com/ipfs/js-ipfs/pull/2304#discussion_r320700893 License: MIT Signed-off-by: Marcin Rataj --- src/core/runtime/preload-browser.js | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/src/core/runtime/preload-browser.js b/src/core/runtime/preload-browser.js index a2c6d6cb2a..7ecb1677eb 100644 --- a/src/core/runtime/preload-browser.js +++ b/src/core/runtime/preload-browser.js @@ -17,8 +17,9 @@ module.exports = function preload (url, callback) { const controller = new AbortController() const signal = controller.signal + const cb = () => setImmediate(callback) // https://github.com/ipfs/js-ipfs/pull/2304#discussion_r320700893 - _httpQueue.add(() => ky.get(url, { signal })).then(() => callback(), callback) + _httpQueue.add(() => ky.get(url, { signal })).then(cb, cb) return { cancel: () => controller.abort() From c239abd3c1324059b521b2186d5d21cb82529ea9 Mon Sep 17 00:00:00 2001 From: Marcin Rataj Date: Thu, 5 Sep 2019 13:08:34 +0200 Subject: [PATCH 18/23] fix(ci): skip tests without implementation Some ipfs.add() tests need to be skipped until https://github.com/ipfs/js-ipfs/pull/2379 is merged License: MIT Signed-off-by: Marcin Rataj --- test/core/interface.spec.js | 13 +++++++++++-- test/http-api/interface.js | 13 ++++++++++++- 2 files changed, 23 insertions(+), 3 deletions(-) diff --git a/test/core/interface.spec.js b/test/core/interface.spec.js index 66cc6966d1..cc6f8ae16e 100644 --- a/test/core/interface.spec.js +++ b/test/core/interface.spec.js @@ -58,13 +58,22 @@ describe('interface-ipfs-core tests', function () { }) tests.filesRegular(defaultCommonFactory, { - skip: isNode ? null : [{ + skip: [ + { + name: 'should add a string', + reason: 'TODO: unskip when https://github.com/ipfs/js-ipfs/pull/2379 is merged' + }, + { + name: 'should add a TypedArray', + reason: 'TODO: unskip when https://github.com/ipfs/js-ipfs/pull/2379 is merged' + } + ].concat(isNode ? [] : [{ name: 'addFromStream', reason: 'Not designed to run in the browser' }, { name: 'addFromFs', reason: 'Not designed to run in the browser' - }] + }]) }) tests.filesMFS(defaultCommonFactory) diff --git a/test/http-api/interface.js b/test/http-api/interface.js index f0b2cbc1f8..fb9983ab38 100644 --- a/test/http-api/interface.js +++ b/test/http-api/interface.js @@ -66,7 +66,18 @@ describe('interface-ipfs-core over ipfs-http-client tests', () => { } }) - tests.filesRegular(defaultCommonFactory) + tests.filesRegular(defaultCommonFactory, { + skip: [ + { + name: 'should add a string', + reason: 'TODO: unskip when https://github.com/ipfs/js-ipfs/pull/2379 is merged' + }, + { + name: 'should add a TypedArray', + reason: 'TODO: unskip when https://github.com/ipfs/js-ipfs/pull/2379 is merged' + } + ] + }) tests.filesMFS(defaultCommonFactory) From e7534ff624cf7171d1e1b352dc2eb6bf01b3acc7 Mon Sep 17 00:00:00 2001 From: Marcin Rataj Date: Thu, 5 Sep 2019 17:45:10 +0200 Subject: [PATCH 19/23] refactor: ipfsdServer start/stop Wait for ipfsd server to start/stop before calling callback License: MIT Signed-off-by: Marcin Rataj --- .aegir.js | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/.aegir.js b/.aegir.js index b04d05facc..8bafe7f504 100644 --- a/.aegir.js +++ b/.aegir.js @@ -11,8 +11,7 @@ const httpEchoServer = EchoHttpServer.createServer() // used by addFromURL const batch = (call, done, ...srvs) => parallel(srvs.map(srv => cb => { if (srv === ipfsdServer) { - srv[call]() - cb() + (srv[call]()).then(() => cb()) } else { srv[call](cb) } From 62dbdc3eed7b5149716e7c5b11529a65307dbe43 Mon Sep 17 00:00:00 2001 From: Marcin Rataj Date: Thu, 5 Sep 2019 19:03:41 +0200 Subject: [PATCH 20/23] refactor: ipfs-utils/src/env/isTest Checks if runtime is test without the need of polyfilling process.env as suggested in https://github.com/ipfs/js-ipfs/pull/2304#discussion_r321227711 License: MIT Signed-off-by: Marcin Rataj --- package.json | 2 +- src/cli/commands/daemon.js | 3 ++- src/core/config.js | 3 ++- src/core/index.js | 3 ++- 4 files changed, 7 insertions(+), 4 deletions(-) diff --git a/package.json b/package.json index fe20b09841..53c410a517 100644 --- a/package.json +++ b/package.json @@ -103,7 +103,7 @@ "ipfs-unixfs": "~0.1.16", "ipfs-unixfs-exporter": "~0.37.7", "ipfs-unixfs-importer": "~0.39.11", - "ipfs-utils": "~0.0.4", + "ipfs-utils": "https://github.com/ipfs/js-ipfs-utils/tarball/136b0e6e99b7df735e7cc8238825d51efaffb443/js-ipfs-utils.tar.gz", "ipld": "~0.24.1", "ipld-bitcoin": "~0.3.0", "ipld-dag-cbor": "~0.15.0", diff --git a/src/cli/commands/daemon.js b/src/cli/commands/daemon.js index aac35d39c2..d7a7ec92a5 100644 --- a/src/cli/commands/daemon.js +++ b/src/cli/commands/daemon.js @@ -3,6 +3,7 @@ const os = require('os') const toUri = require('multiaddr-to-uri') const { ipfsPathHelp } = require('../utils') +const { isTest } = require('ipfs-utils/src/env') module.exports = { command: 'daemon', @@ -32,7 +33,7 @@ module.exports = { }) .option('enable-preload', { type: 'boolean', - default: process.env.NODE_ENV !== 'test' // preload by default, unless in test env + default: !isTest // preload by default, unless in test env }) }, diff --git a/src/core/config.js b/src/core/config.js index 367535c562..4a104b3247 100644 --- a/src/core/config.js +++ b/src/core/config.js @@ -3,6 +3,7 @@ const Multiaddr = require('multiaddr') const mafmt = require('mafmt') const { struct, superstruct } = require('superstruct') +const { isTest } = require('ipfs-utils/src/env') const { optional, union } = struct const s = superstruct({ @@ -32,7 +33,7 @@ const configSchema = s({ addresses: optional(s(['multiaddr'])), interval: 'number?' }, { // defaults - enabled: process.env.NODE_ENV !== 'test', // preload by default, unless in test env + enabled: !isTest, // preload by default, unless in test env interval: 30 * 1000 }), pubsub: optional(s({ diff --git a/src/core/index.js b/src/core/index.js index bdfc249f21..18f45295b2 100644 --- a/src/core/index.js +++ b/src/core/index.js @@ -28,6 +28,7 @@ const defaultRepo = require('./runtime/repo-nodejs') const preload = require('./preload') const mfsPreload = require('./mfs-preload') const ipldOptions = require('./runtime/ipld-nodejs') +const { isTest } = require('ipfs-utils/src/env') /** * @typedef { import("./ipns/index") } IPNS @@ -51,7 +52,7 @@ class IPFS extends EventEmitter { enabled: false }, preload: { - enabled: process.env.NODE_ENV !== 'test', // preload by default, unless in test env + enabled: !isTest, // preload by default, unless in test env addresses: [ '/dnsaddr/node0.preload.ipfs.io/https', '/dnsaddr/node1.preload.ipfs.io/https' From ad65329253da333885b86c7927aa8f0a1e628551 Mon Sep 17 00:00:00 2001 From: Marcin Rataj Date: Fri, 6 Sep 2019 17:05:03 +0200 Subject: [PATCH 21/23] chore: ipfs-utils v0.2.0 we need this version for isTest check License: MIT Signed-off-by: Marcin Rataj --- package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/package.json b/package.json index 733c576db3..a2602b73f0 100644 --- a/package.json +++ b/package.json @@ -103,7 +103,7 @@ "ipfs-unixfs": "~0.1.16", "ipfs-unixfs-exporter": "~0.37.7", "ipfs-unixfs-importer": "~0.39.11", - "ipfs-utils": "https://github.com/ipfs/js-ipfs-utils/tarball/136b0e6e99b7df735e7cc8238825d51efaffb443/js-ipfs-utils.tar.gz", + "ipfs-utils": "~0.2.0", "ipld": "~0.24.1", "ipld-bitcoin": "~0.3.0", "ipld-dag-cbor": "~0.15.0", From ddd49ce41f6f9abd48b1e54bc4c7dd3e58fd0ee9 Mon Sep 17 00:00:00 2001 From: Marcin Rataj Date: Mon, 9 Sep 2019 12:30:03 +0200 Subject: [PATCH 22/23] chore: unskip .add tests License: MIT Signed-off-by: Marcin Rataj --- test/core/interface.spec.js | 13 ++----------- test/http-api/interface.js | 13 +------------ 2 files changed, 3 insertions(+), 23 deletions(-) diff --git a/test/core/interface.spec.js b/test/core/interface.spec.js index 3559662601..10e5c225bd 100644 --- a/test/core/interface.spec.js +++ b/test/core/interface.spec.js @@ -58,22 +58,13 @@ describe('interface-ipfs-core tests', function () { }) tests.filesRegular(defaultCommonFactory, { - skip: [ - { - name: 'should add a string', - reason: 'TODO: unskip when https://github.com/ipfs/js-ipfs/pull/2379 is merged' - }, - { - name: 'should add a TypedArray', - reason: 'TODO: unskip when https://github.com/ipfs/js-ipfs/pull/2379 is merged' - } - ].concat(isNode ? [] : [{ + skip: isNode ? null : [{ name: 'addFromStream', reason: 'Not designed to run in the browser' }, { name: 'addFromFs', reason: 'Not designed to run in the browser' - }]) + }] }) tests.filesMFS(defaultCommonFactory) diff --git a/test/http-api/interface.js b/test/http-api/interface.js index cb0188595a..8c4811822c 100644 --- a/test/http-api/interface.js +++ b/test/http-api/interface.js @@ -66,18 +66,7 @@ describe('interface-ipfs-core over ipfs-http-client tests', () => { } }) - tests.filesRegular(defaultCommonFactory, { - skip: [ - { - name: 'should add a string', - reason: 'TODO: unskip when https://github.com/ipfs/js-ipfs/pull/2379 is merged' - }, - { - name: 'should add a TypedArray', - reason: 'TODO: unskip when https://github.com/ipfs/js-ipfs/pull/2379 is merged' - } - ] - }) + tests.filesRegular(defaultCommonFactory) tests.filesMFS(defaultCommonFactory) From 2e82b2a9b468db99937256f3d77261eaef9a8c0f Mon Sep 17 00:00:00 2001 From: Marcin Rataj Date: Tue, 10 Sep 2019 12:51:29 +0200 Subject: [PATCH 23/23] style: remove underscore License: MIT Signed-off-by: Marcin Rataj --- src/core/runtime/dns-browser.js | 4 ++-- src/core/runtime/preload-browser.js | 4 ++-- 2 files changed, 4 insertions(+), 4 deletions(-) diff --git a/src/core/runtime/dns-browser.js b/src/core/runtime/dns-browser.js index b5ea4f97cf..dd2bdba338 100644 --- a/src/core/runtime/dns-browser.js +++ b/src/core/runtime/dns-browser.js @@ -15,7 +15,7 @@ const ttl = 60 * 1000 // browsers limit concurrent connections per host, // we don't want preload calls to exhaust the limit (~6) -const _httpQueue = new PQueue({ concurrency: 4 }) +const httpQueue = new PQueue({ concurrency: 4 }) // Delegated HTTP resolver sending DNSLink queries to ipfs.io // TODO: replace hardcoded host with configurable DNS over HTTPS: https://github.com/ipfs/js-ipfs/issues/2212 @@ -54,7 +54,7 @@ module.exports = (fqdn, opts = {}, cb) => { } // fallback to delegated DNS resolver - const response = await _httpQueue.add(() => api.get('dns', { searchParams }).json()) + const response = await httpQueue.add(() => api.get('dns', { searchParams }).json()) return ipfsPath(response) } diff --git a/src/core/runtime/preload-browser.js b/src/core/runtime/preload-browser.js index 7ecb1677eb..81d2ad786e 100644 --- a/src/core/runtime/preload-browser.js +++ b/src/core/runtime/preload-browser.js @@ -10,7 +10,7 @@ log.error = debug('ipfs:preload:error') // browsers limit concurrent connections per host, // we don't want preload calls to exhaust the limit (~6) -const _httpQueue = new PQueue({ concurrency: 4 }) +const httpQueue = new PQueue({ concurrency: 4 }) module.exports = function preload (url, callback) { log(url) @@ -19,7 +19,7 @@ module.exports = function preload (url, callback) { const signal = controller.signal const cb = () => setImmediate(callback) // https://github.com/ipfs/js-ipfs/pull/2304#discussion_r320700893 - _httpQueue.add(() => ky.get(url, { signal })).then(cb, cb) + httpQueue.add(() => ky.get(url, { signal })).then(cb, cb) return { cancel: () => controller.abort()