From a8ca9d535cff396030ddc3a8e4e872b1ae48be87 Mon Sep 17 00:00:00 2001 From: achingbrain Date: Wed, 27 Nov 2019 18:35:59 +0000 Subject: [PATCH 1/9] feat: support UnixFSv1.5 metadata --- src/add/form-data.browser.js | 8 ++++++++ src/add/form-data.js | 8 ++++++++ src/add/index.js | 10 ++++++++-- 3 files changed, 24 insertions(+), 2 deletions(-) diff --git a/src/add/form-data.browser.js b/src/add/form-data.browser.js index 247396c42..e02572a0e 100644 --- a/src/add/form-data.browser.js +++ b/src/add/form-data.browser.js @@ -9,6 +9,14 @@ exports.toFormData = async input => { let i = 0 for await (const file of files) { + if (file.mtime) { + formData.append('mtime', file.mtime) + } + + if (file.mode) { + formData.append('mode', file.mode) + } + if (file.content) { // In the browser there's _currently_ no streaming upload, buffer up our // async iterator chunks and append a big Blob :( diff --git a/src/add/form-data.js b/src/add/form-data.js index 96d55f672..f98ef70a9 100644 --- a/src/add/form-data.js +++ b/src/add/form-data.js @@ -12,6 +12,14 @@ exports.toFormData = async input => { let i = 0 for await (const file of files) { + if (file.mtime) { + formData.append(`file-${i}-mtime`, file.mtime) + } + + if (file.mode) { + formData.append(`file-${i}-mode`, file.mode) + } + if (file.content) { // In Node.js, FormData can be passed a stream so no need to buffer formData.append( diff --git a/src/add/index.js b/src/add/index.js index b1a8bb2a1..4afe7be80 100644 --- a/src/add/index.js +++ b/src/add/index.js @@ -50,6 +50,12 @@ module.exports = configure(({ ky }) => { } }) -function toCoreInterface ({ name, hash, size }) { - return { path: name, hash, size: parseInt(size) } +function toCoreInterface ({ name, hash, size, mode, mtime }) { + return { + path: name, + hash, + size: parseInt(size), + mode: mode, + mtime: mtime + } } From 6a7e5c9004a1adc030874da1ffaa46a0ffbbc0a6 Mon Sep 17 00:00:00 2001 From: achingbrain Date: Thu, 28 Nov 2019 08:45:48 +0000 Subject: [PATCH 2/9] fix: expose new mfs functions --- src/files/chmod.js | 24 ++++++++++++++++++++++++ src/files/touch.js | 29 +++++++++++++++++++++++++++++ 2 files changed, 53 insertions(+) create mode 100644 src/files/chmod.js create mode 100644 src/files/touch.js diff --git a/src/files/chmod.js b/src/files/chmod.js new file mode 100644 index 000000000..486644a03 --- /dev/null +++ b/src/files/chmod.js @@ -0,0 +1,24 @@ +'use strict' + +const configure = require('../lib/configure') + +module.exports = configure(({ ky }) => { + return function chmod (path, mode, options) { + options = options || {} + + const searchParams = new URLSearchParams(options.searchParams) + searchParams.append('arg', path) + searchParams.append('mode', mode) + if (options.format) searchParams.set('format', options.format) + if (options.flush != null) searchParams.set('flush', options.flush) + if (options.hashAlg) searchParams.set('hash', options.hashAlg) + if (options.parents != null) searchParams.set('parents', options.parents) + + return ky.post('files/chmod', { + timeout: options.timeout, + signal: options.signal, + headers: options.headers, + searchParams + }).text() + } +}) diff --git a/src/files/touch.js b/src/files/touch.js new file mode 100644 index 000000000..8dba736da --- /dev/null +++ b/src/files/touch.js @@ -0,0 +1,29 @@ +'use strict' + +const configure = require('../lib/configure') + +module.exports = configure(({ ky }) => { + return function touch (path, mtime, options) { + options = options || {} + + if (isNaN(mtime)) { + options = mtime + mtime = null + } + + const searchParams = new URLSearchParams(options.searchParams) + searchParams.append('arg', path) + if (mtime) searchParams.set('mtime', mtime) + if (options.format) searchParams.set('format', options.format) + if (options.flush != null) searchParams.set('flush', options.flush) + if (options.hashAlg) searchParams.set('hash', options.hashAlg) + if (options.parents != null) searchParams.set('parents', options.parents) + + return ky.post('files/touch', { + timeout: options.timeout, + signal: options.signal, + headers: options.headers, + searchParams + }).text() + } +}) From 90758e142130f18e34089038b5f2c6dd040907f3 Mon Sep 17 00:00:00 2001 From: achingbrain Date: Wed, 4 Dec 2019 14:11:01 +0000 Subject: [PATCH 3/9] refactor: send mtime and mode as headers instead of message parts --- src/add/form-data.browser.js | 14 ++++++++++---- src/add/form-data.js | 9 ++++++--- 2 files changed, 16 insertions(+), 7 deletions(-) diff --git a/src/add/form-data.browser.js b/src/add/form-data.browser.js index e02572a0e..daddd1f74 100644 --- a/src/add/form-data.browser.js +++ b/src/add/form-data.browser.js @@ -9,12 +9,14 @@ exports.toFormData = async input => { let i = 0 for await (const file of files) { + const headers = {} + if (file.mtime) { - formData.append('mtime', file.mtime) + headers.mtime = file.mtime } if (file.mode) { - formData.append('mode', file.mode) + headers.mode = file.mode } if (file.content) { @@ -26,9 +28,13 @@ exports.toFormData = async input => { bufs.push(chunk) } - formData.append(`file-${i}`, new Blob(bufs, { type: 'application/octet-stream' }), encodeURIComponent(file.path)) + formData.append(`file-${i}`, new Blob(bufs, { type: 'application/octet-stream' }), encodeURIComponent(file.path), { + header: headers + }) } else { - formData.append(`dir-${i}`, new Blob([], { type: 'application/x-directory' }), encodeURIComponent(file.path)) + formData.append(`dir-${i}`, new Blob([], { type: 'application/x-directory' }), encodeURIComponent(file.path), { + header: headers + }) } i++ diff --git a/src/add/form-data.js b/src/add/form-data.js index f98ef70a9..488d88131 100644 --- a/src/add/form-data.js +++ b/src/add/form-data.js @@ -12,12 +12,14 @@ exports.toFormData = async input => { let i = 0 for await (const file of files) { + const headers = {} + if (file.mtime) { - formData.append(`file-${i}-mtime`, file.mtime) + headers.mtime = file.mtime } if (file.mode) { - formData.append(`file-${i}-mode`, file.mode) + headers.mode = file.mode.toString(8).padStart(4, '0') } if (file.content) { @@ -34,7 +36,8 @@ exports.toFormData = async input => { { filepath: encodeURIComponent(file.path), contentType: 'application/octet-stream', - knownLength: file.content.length // Send Content-Length header if known + knownLength: file.content.length, // Send Content-Length header if known + header: headers } ) } else { From bf755d27765a6c0b25e496899bf6b5aec0142e97 Mon Sep 17 00:00:00 2001 From: achingbrain Date: Wed, 4 Dec 2019 14:41:46 +0000 Subject: [PATCH 4/9] fix: include headers for directories --- src/add/form-data.js | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/src/add/form-data.js b/src/add/form-data.js index 488d88131..ec07dd608 100644 --- a/src/add/form-data.js +++ b/src/add/form-data.js @@ -43,7 +43,8 @@ exports.toFormData = async input => { } else { formData.append(`dir-${i}`, Buffer.alloc(0), { filepath: encodeURIComponent(file.path), - contentType: 'application/x-directory' + contentType: 'application/x-directory', + header: headers }) } From 65a6ec5290e0f8bd285041dbaddd162367e85ecf Mon Sep 17 00:00:00 2001 From: achingbrain Date: Wed, 4 Dec 2019 17:24:22 +0000 Subject: [PATCH 5/9] chore: update ipfs utils dep version --- package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/package.json b/package.json index 7d562b413..32de03ac2 100644 --- a/package.json +++ b/package.json @@ -55,7 +55,7 @@ "explain-error": "^1.0.4", "form-data": "^3.0.0", "ipfs-block": "~0.8.1", - "ipfs-utils": "^0.4.0", + "ipfs-utils": "ipfs/js-ipfs-utils#support-unixfs-metadata", "ipld-dag-cbor": "~0.15.0", "ipld-dag-pb": "^0.18.1", "ipld-raw": "^4.0.0", From efe07fd2bf9f8455d476e13c2901d9b74c81e752 Mon Sep 17 00:00:00 2001 From: achingbrain Date: Fri, 6 Dec 2019 15:23:12 +0000 Subject: [PATCH 6/9] chore: update ipfs-utils dep --- package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/package.json b/package.json index e787b2875..94367213e 100644 --- a/package.json +++ b/package.json @@ -55,7 +55,7 @@ "explain-error": "^1.0.4", "form-data": "^3.0.0", "ipfs-block": "~0.8.1", - "ipfs-utils": "ipfs/js-ipfs-utils#support-unixfs-metadata", + "ipfs-utils": "^0.5.0", "ipld-dag-cbor": "~0.15.0", "ipld-dag-pb": "^0.18.1", "ipld-raw": "^4.0.0", From e55f92dae7f2f837453a9addfafa1bc8ae9f6977 Mon Sep 17 00:00:00 2001 From: achingbrain Date: Tue, 10 Dec 2019 16:05:40 +0000 Subject: [PATCH 7/9] fix: stringify mode in browser --- package.json | 4 ++-- src/add/form-data.browser.js | 2 +- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/package.json b/package.json index 9cf72a541..5b0655817 100644 --- a/package.json +++ b/package.json @@ -84,8 +84,8 @@ "cross-env": "^6.0.0", "detect-node": "^2.0.4", "go-ipfs-dep": "^0.4.22", - "interface-ipfs-core": "^0.124.0", - "ipfsd-ctl": "^0.47.1", + "interface-ipfs-core": "^0.124.1", + "ipfsd-ctl": "^0.47.4", "ndjson": "^1.5.0", "nock": "^11.4.0", "pull-stream": "^3.6.14", diff --git a/src/add/form-data.browser.js b/src/add/form-data.browser.js index daddd1f74..93e1054dd 100644 --- a/src/add/form-data.browser.js +++ b/src/add/form-data.browser.js @@ -16,7 +16,7 @@ exports.toFormData = async input => { } if (file.mode) { - headers.mode = file.mode + headers.mode = file.mode.toString(8).padStart(4, '0') } if (file.content) { From d34944eff821e5983e9b3c25dc840268945f5ded Mon Sep 17 00:00:00 2001 From: achingbrain Date: Mon, 23 Dec 2019 16:28:00 +0000 Subject: [PATCH 8/9] test: add tests for unixfs metadata --- package.json | 4 +- src/add/form-data.browser.js | 12 ++- src/add/form-data.js | 12 ++- src/add/index.js | 12 ++- src/files/chmod.js | 3 +- src/files/index.js | 2 + src/files/ls.js | 7 +- src/files/mkdir.js | 8 ++ src/files/stat.js | 5 +- src/files/touch.js | 14 +-- src/files/write.js | 13 ++- src/lib/buffer-to-form-data.js | 20 +++- src/lib/mode-to-string.js | 13 +++ src/lib/mtime-to-object.js | 56 ++++++++++ src/lib/object-to-camel-with-mode.js | 15 +++ src/ls.js | 18 +++- test/interface.spec.js | 156 +++++++++++++++++++++++++++ 17 files changed, 341 insertions(+), 29 deletions(-) create mode 100644 src/lib/mode-to-string.js create mode 100644 src/lib/mtime-to-object.js create mode 100644 src/lib/object-to-camel-with-mode.js diff --git a/package.json b/package.json index 441d63236..5363b2739 100644 --- a/package.json +++ b/package.json @@ -54,7 +54,7 @@ "explain-error": "^1.0.4", "form-data": "^3.0.0", "ipfs-block": "~0.8.1", - "ipfs-utils": "^0.5.0", + "ipfs-utils": "ipfs/js-ipfs-utils#format-mtime-as-timespec", "ipld-dag-cbor": "~0.15.0", "ipld-dag-pb": "^0.18.1", "ipld-raw": "^4.0.0", @@ -84,7 +84,7 @@ "cross-env": "^6.0.0", "detect-node": "^2.0.4", "go-ipfs-dep": "^0.4.22", - "interface-ipfs-core": "~0.125.0", + "interface-ipfs-core": "ipfs/interface-js-ipfs-core#add-unixfs-metadata-tests-and-docs", "ipfsd-ctl": "^1.0.0", "ndjson": "^1.5.0", "nock": "^11.4.0", diff --git a/src/add/form-data.browser.js b/src/add/form-data.browser.js index 93e1054dd..484f67c6c 100644 --- a/src/add/form-data.browser.js +++ b/src/add/form-data.browser.js @@ -2,6 +2,7 @@ /* eslint-env browser */ const normaliseInput = require('ipfs-utils/src/files/normalise-input') +const mtimeToObject = require('../lib/mtime-to-object') exports.toFormData = async input => { const files = normaliseInput(input) @@ -11,11 +12,16 @@ exports.toFormData = async input => { for await (const file of files) { const headers = {} - if (file.mtime) { - headers.mtime = file.mtime + if (file.mtime !== undefined && file.mtime !== null) { + const mtime = mtimeToObject(file.mtime) + + if (mtime) { + headers.mtime = mtime.secs + headers['mtime-nsecs'] = mtime.nsecs + } } - if (file.mode) { + if (file.mode !== undefined && file.mode !== null) { headers.mode = file.mode.toString(8).padStart(4, '0') } diff --git a/src/add/form-data.js b/src/add/form-data.js index ec07dd608..1ce5050da 100644 --- a/src/add/form-data.js +++ b/src/add/form-data.js @@ -5,6 +5,7 @@ const { Buffer } = require('buffer') const toStream = require('it-to-stream') const normaliseInput = require('ipfs-utils/src/files/normalise-input') const { isElectronRenderer } = require('ipfs-utils/src/env') +const mtimeToObject = require('../lib/mtime-to-object') exports.toFormData = async input => { const files = normaliseInput(input) @@ -14,11 +15,16 @@ exports.toFormData = async input => { for await (const file of files) { const headers = {} - if (file.mtime) { - headers.mtime = file.mtime + if (file.mtime !== undefined && file.mtime !== null) { + const mtime = mtimeToObject(file.mtime) + + if (mtime) { + headers.mtime = mtime.secs + headers['mtime-nsecs'] = mtime.nsecs + } } - if (file.mode) { + if (file.mode !== undefined && file.mode !== null) { headers.mode = file.mode.toString(8).padStart(4, '0') } diff --git a/src/add/index.js b/src/add/index.js index 841ad0a9f..546e5428d 100644 --- a/src/add/index.js +++ b/src/add/index.js @@ -53,11 +53,15 @@ module.exports = configure(({ ky }) => { }) function toCoreInterface ({ name, hash, size, mode, mtime }) { - return { + const output = { path: name, hash, - size: parseInt(size), - mode: mode, - mtime: mtime + size: parseInt(size) } + + if (mode !== undefined) { + output.mode = parseInt(mode, 8) + } + + return output } diff --git a/src/files/chmod.js b/src/files/chmod.js index 486644a03..b4c0a11dd 100644 --- a/src/files/chmod.js +++ b/src/files/chmod.js @@ -1,6 +1,7 @@ 'use strict' const configure = require('../lib/configure') +const modeToString = require('../lib/mode-to-string') module.exports = configure(({ ky }) => { return function chmod (path, mode, options) { @@ -8,7 +9,7 @@ module.exports = configure(({ ky }) => { const searchParams = new URLSearchParams(options.searchParams) searchParams.append('arg', path) - searchParams.append('mode', mode) + searchParams.append('mode', modeToString(mode)) if (options.format) searchParams.set('format', options.format) if (options.flush != null) searchParams.set('flush', options.flush) if (options.hashAlg) searchParams.set('hash', options.hashAlg) diff --git a/src/files/index.js b/src/files/index.js index 25e79fcab..ee6b7d8d8 100644 --- a/src/files/index.js +++ b/src/files/index.js @@ -8,6 +8,7 @@ module.exports = config => { const read = require('./read')(config) return { + chmod: callbackify.variadic(require('./chmod')(config)), cp: callbackify.variadic(require('./cp')(config)), mkdir: callbackify.variadic(require('./mkdir')(config)), flush: callbackify.variadic(require('./flush')(config)), @@ -19,6 +20,7 @@ module.exports = config => { read: callbackify.variadic(concatify(read)), readReadableStream: streamify.readable(read), readPullStream: pullify.source(read), + touch: callbackify.variadic(require('./touch')(config)), write: callbackify.variadic(require('./write')(config)), mv: callbackify.variadic(require('./mv')(config)) } diff --git a/src/files/ls.js b/src/files/ls.js index 1baa3f656..771acd6cb 100644 --- a/src/files/ls.js +++ b/src/files/ls.js @@ -4,7 +4,7 @@ const CID = require('cids') const ndjson = require('iterable-ndjson') const toIterable = require('../lib/stream-to-iterable') const configure = require('../lib/configure') -const toCamel = require('../lib/object-to-camel') +const toCamelWithMode = require('../lib/object-to-camel-with-mode') module.exports = configure(({ ky }) => { return async function * ls (path, options) { @@ -32,11 +32,12 @@ module.exports = configure(({ ky }) => { // go-ipfs does not yet support the "stream" option if ('Entries' in result) { for (const entry of result.Entries || []) { - yield toCamel(entry) + yield toCamelWithMode(entry) } return } - yield toCamel(result) + + yield toCamelWithMode(result) } } }) diff --git a/src/files/mkdir.js b/src/files/mkdir.js index 0fc3c238d..b9e1e2d51 100644 --- a/src/files/mkdir.js +++ b/src/files/mkdir.js @@ -1,10 +1,13 @@ 'use strict' const configure = require('../lib/configure') +const modeToString = require('../lib/mode-to-string') +const mtimeToObject = require('../lib/mtime-to-object') module.exports = configure(({ ky }) => { return (path, options) => { options = options || {} + const mtime = mtimeToObject(options.mtime) const searchParams = new URLSearchParams(options.searchParams) searchParams.append('arg', path) @@ -13,6 +16,11 @@ module.exports = configure(({ ky }) => { if (options.flush != null) searchParams.set('flush', options.flush) if (options.hashAlg) searchParams.set('hash', options.hashAlg) if (options.parents != null) searchParams.set('parents', options.parents) + if (mtime) { + searchParams.set('mtime', mtime.secs) + searchParams.set('mtimeNsecs', mtime.nsecs) + } + if (options.mode != null) searchParams.set('mode', modeToString(options.mode)) return ky.post('files/mkdir', { timeout: options.timeout, diff --git a/src/files/stat.js b/src/files/stat.js index 98026283e..efaf4c504 100644 --- a/src/files/stat.js +++ b/src/files/stat.js @@ -1,7 +1,7 @@ 'use strict' const configure = require('../lib/configure') -const toCamel = require('../lib/object-to-camel') +const toCamelWithMode = require('../lib/object-to-camel-with-mode') module.exports = configure(({ ky }) => { return async (path, options) => { @@ -27,6 +27,7 @@ module.exports = configure(({ ky }) => { }).json() res.WithLocality = res.WithLocality || false - return toCamel(res) + + return toCamelWithMode(res) } }) diff --git a/src/files/touch.js b/src/files/touch.js index 8dba736da..b38aca905 100644 --- a/src/files/touch.js +++ b/src/files/touch.js @@ -1,19 +1,19 @@ 'use strict' const configure = require('../lib/configure') +const mtimeToObject = require('../lib/mtime-to-object') module.exports = configure(({ ky }) => { - return function touch (path, mtime, options) { + return function touch (path, options) { options = options || {} - - if (isNaN(mtime)) { - options = mtime - mtime = null - } + const mtime = mtimeToObject(options.mtime) const searchParams = new URLSearchParams(options.searchParams) searchParams.append('arg', path) - if (mtime) searchParams.set('mtime', mtime) + if (mtime) { + searchParams.set('mtime', mtime.secs) + searchParams.set('mtimeNsecs', mtime.nsecs) + } if (options.format) searchParams.set('format', options.format) if (options.flush != null) searchParams.set('flush', options.flush) if (options.hashAlg) searchParams.set('hash', options.hashAlg) diff --git a/src/files/write.js b/src/files/write.js index 77a772ea6..e513e579d 100644 --- a/src/files/write.js +++ b/src/files/write.js @@ -2,10 +2,13 @@ const configure = require('../lib/configure') const toFormData = require('../lib/buffer-to-form-data') +const modeToString = require('../lib/mode-to-string') +const mtimeToObject = require('../lib/mtime-to-object') module.exports = configure(({ ky }) => { return async (path, input, options) => { options = options || {} + const mtime = mtimeToObject(options.mtime) const searchParams = new URLSearchParams(options.searchParams) searchParams.set('arg', path) @@ -18,13 +21,21 @@ module.exports = configure(({ ky }) => { if (options.parents != null) searchParams.set('parents', options.parents) if (options.rawLeaves != null) searchParams.set('raw-leaves', options.rawLeaves) if (options.truncate != null) searchParams.set('truncate', options.truncate) + if (mtime) { + searchParams.set('mtime', options.mtime.secs) + searchParams.set('mtimeNsecs', options.mtime.nsecs) + } const res = await ky.post('files/write', { timeout: options.timeout, signal: options.signal, headers: options.headers, searchParams, - body: toFormData(input) // TODO: support inputs other than buffer as per spec + body: toFormData(input, { + mode: options.mode != null ? modeToString(options.mode) : undefined, + mtime: mtime ? mtime.secs : undefined, + mtimeNsecs: mtime ? mtime.nsecs : undefined + }) // TODO: support inputs other than buffer as per spec }) return res.text() diff --git a/src/lib/buffer-to-form-data.js b/src/lib/buffer-to-form-data.js index 41f03383e..1a4830361 100644 --- a/src/lib/buffer-to-form-data.js +++ b/src/lib/buffer-to-form-data.js @@ -3,9 +3,25 @@ const FormData = require('form-data') const { isElectronRenderer } = require('ipfs-utils/src/env') -module.exports = buf => { +module.exports = (buf, { mode, mtime, mtimeNsecs } = {}) => { + const headers = {} + + if (mode != null) { + headers.mode = mode + } + + if (mtime != null) { + headers.mtime = mtime + + if (mtimeNsecs != null) { + headers['mtime-nsecs'] = mtimeNsecs + } + } + const formData = new FormData() - formData.append('file', buf) + formData.append('file', buf, { + header: headers + }) return formData } diff --git a/src/lib/mode-to-string.js b/src/lib/mode-to-string.js new file mode 100644 index 000000000..ee2742b9a --- /dev/null +++ b/src/lib/mode-to-string.js @@ -0,0 +1,13 @@ +'use strict' + +module.exports = (mode) => { + if (mode === undefined || mode === null) { + return undefined + } + + if (typeof mode === 'string' || mode instanceof String) { + return mode + } + + return mode.toString(8).padStart(4, '0') +} diff --git a/src/lib/mtime-to-object.js b/src/lib/mtime-to-object.js new file mode 100644 index 000000000..179459cd6 --- /dev/null +++ b/src/lib/mtime-to-object.js @@ -0,0 +1,56 @@ +'use strict' + +module.exports = function parseMtime (mtime) { + if (mtime == null) { + return undefined + } + + // Javascript Date + if (mtime instanceof Date) { + const ms = mtime.getTime() + const secs = Math.floor(ms / 1000) + + return { + secs: secs, + nsecs: (ms - (secs * 1000)) * 1000 + } + } + + // { secs, nsecs } + if (Object.prototype.hasOwnProperty.call(mtime, 'secs')) { + return { + secs: mtime.secs, + nsecs: mtime.nsecs + } + } + + // UnixFS TimeSpec + if (Object.prototype.hasOwnProperty.call(mtime, 'EpochSeconds')) { + return { + secs: mtime.EpochSeconds, + nsecs: mtime.EpochNanoseconds + } + } + + // process.hrtime() + if (Array.isArray(mtime)) { + return { + secs: mtime[0], + nsecs: mtime[1] + } + } + /* + TODO: https://github.com/ipfs/aegir/issues/487 + + // process.hrtime.bigint() + if (typeof mtime === 'bigint') { + const secs = mtime / BigInt(1e9) + const nsecs = mtime - (secs * BigInt(1e9)) + + return { + secs: parseInt(secs), + nsecs: parseInt(nsecs) + } + } + */ +} diff --git a/src/lib/object-to-camel-with-mode.js b/src/lib/object-to-camel-with-mode.js new file mode 100644 index 000000000..d42941273 --- /dev/null +++ b/src/lib/object-to-camel-with-mode.js @@ -0,0 +1,15 @@ +'use strict' + +const toCamel = require('./object-to-camel') + +function toCamelWithMode (entry) { + const file = toCamel(entry) + + if (Object.prototype.hasOwnProperty.call(file, 'mode')) { + file.mode = parseInt(file.mode, 8) + } + + return file +} + +module.exports = toCamelWithMode diff --git a/src/ls.js b/src/ls.js index a9cd476f9..43e92a54a 100644 --- a/src/ls.js +++ b/src/ls.js @@ -48,7 +48,7 @@ module.exports = configure(({ ky }) => { } for (const link of result) { - yield { + const entry = { name: link.Name, path: path + '/' + link.Name, size: link.Size, @@ -56,6 +56,22 @@ module.exports = configure(({ ky }) => { type: typeOf(link), depth: link.Depth || 1 } + + if (link.Mode) { + entry.mode = parseInt(link.Mode, 8) + } + + if (link.Mtime !== undefined && link.Mtime !== null) { + entry.mtime = { + secs: link.Mtime + } + + if (link.MtimeNsecs !== undefined && link.MtimeNsecs !== null) { + entry.mtime.nsecs = link.MtimeNsecs + } + } + + yield entry } } }) diff --git a/test/interface.spec.js b/test/interface.spec.js index 5435518cf..e09cac5a6 100644 --- a/test/interface.spec.js +++ b/test/interface.spec.js @@ -114,6 +114,134 @@ describe('interface-ipfs-core tests', () => { { name: 'should ls from outside of mfs', reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'should change file mode', + reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'should change directory mode', + reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'should change file mode as string', + reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'should change file mode to 0', + reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'should update file mtime', + reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'should update directory mtime', + reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'should make directory and specify mode', + reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'should make directory and specify mtime', + reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'should write file and specify mode', + reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'should write file and specify mtime', + reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'should respect metadata when copying files', + reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'should respect metadata when copying directories', + reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'should respect metadata when copying from outside of mfs', + reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'ls directory with long option should include metadata', + reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'should have default mtime', + reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'should set mtime as Date', + reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'should set mtime as { nsecs, secs }', + reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'should set mtime as timespec', + reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'should set mtime as hrtime', + reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'should make directory and have default mode', + reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'should make directory and specify mode as string', + reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'should make directory and specify mode as number', + reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'should make directory and specify mtime as Date', + reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'should make directory and specify mtime as { nsecs, secs }', + reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'should make directory and specify mtime as timespec', + reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'should make directory and specify mtime as hrtime', + reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'should write file and specify mode as a string', + reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'should write file and specify mode as a number', + reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'should write file and specify mtime as Date', + reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'should write file and specify mtime as { nsecs, secs }', + reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'should write file and specify mtime as timespec', + reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'should write file and specify mtime as hrtime', + reason: 'TODO not implemented in go-ipfs yet' } ] }) @@ -125,6 +253,30 @@ describe('interface-ipfs-core tests', () => { name: 'addFromFs', reason: 'Not designed to run in the browser' }, + { + name: 'should add with mode as string', + reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'should add with mode as number', + reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'should add with mtime as Date', + reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'should add with mtime as { nsecs, secs }', + reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'should add with mtime as timespec', + reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'should add with mtime as hrtime', + reason: 'TODO not implemented in go-ipfs yet' + }, // .catPullStream { name: 'should export a chunk of a file', @@ -137,6 +289,10 @@ describe('interface-ipfs-core tests', () => { { name: 'should export a chunk of a file in a Readable Stream', reason: 'TODO not implemented in go-ipfs yet' + }, + { + name: 'should ls with metadata', + reason: 'TODO not implemented in go-ipfs yet' } ] }) From 0833e2e84e3642475ddddcf945c8c4e1ed64836c Mon Sep 17 00:00:00 2001 From: achingbrain Date: Thu, 9 Jan 2020 10:24:11 +0000 Subject: [PATCH 9/9] fix: fix up tests etc for optional mtime --- package.json | 4 ++-- src/files/ls.js | 6 +++--- src/files/mkdir.js | 5 ++++- src/files/stat.js | 4 ++-- src/files/write.js | 7 +++++-- src/lib/mtime-to-object.js | 6 +++--- src/lib/object-to-camel-with-metadata.js | 24 ++++++++++++++++++++++++ src/lib/object-to-camel-with-mode.js | 15 --------------- 8 files changed, 43 insertions(+), 28 deletions(-) create mode 100644 src/lib/object-to-camel-with-metadata.js delete mode 100644 src/lib/object-to-camel-with-mode.js diff --git a/package.json b/package.json index 5363b2739..261b888c9 100644 --- a/package.json +++ b/package.json @@ -54,7 +54,7 @@ "explain-error": "^1.0.4", "form-data": "^3.0.0", "ipfs-block": "~0.8.1", - "ipfs-utils": "ipfs/js-ipfs-utils#format-mtime-as-timespec", + "ipfs-utils": "^0.4.2", "ipld-dag-cbor": "~0.15.0", "ipld-dag-pb": "^0.18.1", "ipld-raw": "^4.0.0", @@ -84,7 +84,7 @@ "cross-env": "^6.0.0", "detect-node": "^2.0.4", "go-ipfs-dep": "^0.4.22", - "interface-ipfs-core": "ipfs/interface-js-ipfs-core#add-unixfs-metadata-tests-and-docs", + "interface-ipfs-core": "^0.126.0", "ipfsd-ctl": "^1.0.0", "ndjson": "^1.5.0", "nock": "^11.4.0", diff --git a/src/files/ls.js b/src/files/ls.js index 771acd6cb..51ee33912 100644 --- a/src/files/ls.js +++ b/src/files/ls.js @@ -4,7 +4,7 @@ const CID = require('cids') const ndjson = require('iterable-ndjson') const toIterable = require('../lib/stream-to-iterable') const configure = require('../lib/configure') -const toCamelWithMode = require('../lib/object-to-camel-with-mode') +const toCamelWithMetadata = require('../lib/object-to-camel-with-metadata') module.exports = configure(({ ky }) => { return async function * ls (path, options) { @@ -32,12 +32,12 @@ module.exports = configure(({ ky }) => { // go-ipfs does not yet support the "stream" option if ('Entries' in result) { for (const entry of result.Entries || []) { - yield toCamelWithMode(entry) + yield toCamelWithMetadata(entry) } return } - yield toCamelWithMode(result) + yield toCamelWithMetadata(result) } } }) diff --git a/src/files/mkdir.js b/src/files/mkdir.js index b9e1e2d51..3a50c7728 100644 --- a/src/files/mkdir.js +++ b/src/files/mkdir.js @@ -18,7 +18,10 @@ module.exports = configure(({ ky }) => { if (options.parents != null) searchParams.set('parents', options.parents) if (mtime) { searchParams.set('mtime', mtime.secs) - searchParams.set('mtimeNsecs', mtime.nsecs) + + if (mtime.nsecs != null) { + searchParams.set('mtimeNsecs', mtime.nsecs) + } } if (options.mode != null) searchParams.set('mode', modeToString(options.mode)) diff --git a/src/files/stat.js b/src/files/stat.js index efaf4c504..1b4af061b 100644 --- a/src/files/stat.js +++ b/src/files/stat.js @@ -1,7 +1,7 @@ 'use strict' const configure = require('../lib/configure') -const toCamelWithMode = require('../lib/object-to-camel-with-mode') +const toCamelWithMetadata = require('../lib/object-to-camel-with-metadata') module.exports = configure(({ ky }) => { return async (path, options) => { @@ -28,6 +28,6 @@ module.exports = configure(({ ky }) => { res.WithLocality = res.WithLocality || false - return toCamelWithMode(res) + return toCamelWithMetadata(res) } }) diff --git a/src/files/write.js b/src/files/write.js index e513e579d..e31f0a8a2 100644 --- a/src/files/write.js +++ b/src/files/write.js @@ -22,8 +22,11 @@ module.exports = configure(({ ky }) => { if (options.rawLeaves != null) searchParams.set('raw-leaves', options.rawLeaves) if (options.truncate != null) searchParams.set('truncate', options.truncate) if (mtime) { - searchParams.set('mtime', options.mtime.secs) - searchParams.set('mtimeNsecs', options.mtime.nsecs) + searchParams.set('mtime', mtime.secs) + + if (mtime.nsecs != null) { + searchParams.set('mtimeNsecs', mtime.nsecs) + } } const res = await ky.post('files/write', { diff --git a/src/lib/mtime-to-object.js b/src/lib/mtime-to-object.js index 179459cd6..be89148f6 100644 --- a/src/lib/mtime-to-object.js +++ b/src/lib/mtime-to-object.js @@ -25,10 +25,10 @@ module.exports = function parseMtime (mtime) { } // UnixFS TimeSpec - if (Object.prototype.hasOwnProperty.call(mtime, 'EpochSeconds')) { + if (Object.prototype.hasOwnProperty.call(mtime, 'Seconds')) { return { - secs: mtime.EpochSeconds, - nsecs: mtime.EpochNanoseconds + secs: mtime.Seconds, + nsecs: mtime.FractionalNanoseconds } } diff --git a/src/lib/object-to-camel-with-metadata.js b/src/lib/object-to-camel-with-metadata.js new file mode 100644 index 000000000..55f16d0bb --- /dev/null +++ b/src/lib/object-to-camel-with-metadata.js @@ -0,0 +1,24 @@ +'use strict' + +const toCamel = require('./object-to-camel') + +function toCamelWithMetadata (entry) { + const file = toCamel(entry) + + if (Object.prototype.hasOwnProperty.call(file, 'mode')) { + file.mode = parseInt(file.mode, 8) + } + + if (Object.prototype.hasOwnProperty.call(file, 'mtime')) { + file.mtime = { + secs: file.mtime, + nsecs: file.mtimeNsecs || 0 + } + + delete file.mtimeNsecs + } + + return file +} + +module.exports = toCamelWithMetadata diff --git a/src/lib/object-to-camel-with-mode.js b/src/lib/object-to-camel-with-mode.js deleted file mode 100644 index d42941273..000000000 --- a/src/lib/object-to-camel-with-mode.js +++ /dev/null @@ -1,15 +0,0 @@ -'use strict' - -const toCamel = require('./object-to-camel') - -function toCamelWithMode (entry) { - const file = toCamel(entry) - - if (Object.prototype.hasOwnProperty.call(file, 'mode')) { - file.mode = parseInt(file.mode, 8) - } - - return file -} - -module.exports = toCamelWithMode