From ed66d9e0c09ca9c2559470d09aab5e4a04df3fa1 Mon Sep 17 00:00:00 2001 From: Irakli Gozalishvili Date: Fri, 5 Jun 2020 12:40:29 -0700 Subject: [PATCH 1/8] feat(package): backwards compatible pure data model API --- package.json | 3 +- src/dag-link/dagLink.js | 38 +--- src/dag-node/addLink.js | 4 +- src/dag-node/dagNode.js | 46 ++--- src/dag-node/rmLink.js | 20 +- src/dag-node/sortLinks.js | 13 +- src/serialize.js | 24 ++- test/dag-node-test.js | 4 +- test/resolver.spec.js | 379 +++++++++++++++++++++----------------- test/util.spec.js | 24 ++- 10 files changed, 304 insertions(+), 251 deletions(-) diff --git a/package.json b/package.json index f2fab2d..22a0688 100644 --- a/package.json +++ b/package.json @@ -76,6 +76,7 @@ "aegir": "^22.0.0", "chai": "^4.2.0", "dirty-chai": "^2.0.1", + "chai-subset": "1.6.0", "fs-extra": "^9.0.0", "ipfs-block-service": "~0.17.0", "ipfs-repo": "^3.0.0", @@ -84,4 +85,4 @@ "multibase": "^0.7.0", "multihashes": "~0.4.19" } -} +} \ No newline at end of file diff --git a/src/dag-link/dagLink.js b/src/dag-link/dagLink.js index dc55565..b661845 100644 --- a/src/dag-link/dagLink.js +++ b/src/dag-link/dagLink.js @@ -15,14 +15,16 @@ class DAGLink { // note - links should include size, but this assert is disabled // for now to maintain consistency with go-ipfs pinset - this._name = name || '' - this._nameBuf = null - this._size = size - this._cid = new CID(cid) + Object.defineProperties(this, { + Name: { value: name || '', writable: false, enumerable: true }, + Tsize: { value: size, writable: false, enumerable: true }, + Hash: { value: new CID(cid), writable: false, enumerable: true }, + _nameBuf: { value: null, writable: true, enumerable: false } + }) } toString () { - return `DAGLink <${this._cid.toBaseEncodedString()} - name: "${this.Name}", size: ${this.Tsize}>` + return `DAGLink <${this.Hash.toBaseEncodedString()} - name: "${this.Name}", size: ${this.Tsize}>` } toJSON () { @@ -37,10 +39,6 @@ class DAGLink { return Object.assign({}, this._json) } - get Name () { - return this._name - } - // Memoize the Buffer representation of name // We need this to sort the links, otherwise // we will reallocate new buffers every time @@ -49,29 +47,9 @@ class DAGLink { return this._nameBuf } - this._nameBuf = Buffer.from(this._name) + this._nameBuf = Buffer.from(this.Name) return this._nameBuf } - - set Name (name) { - throw new Error("Can't set property: 'name' is immutable") - } - - get Tsize () { - return this._size - } - - set Tsize (size) { - throw new Error("Can't set property: 'size' is immutable") - } - - get Hash () { - return this._cid - } - - set Hash (cid) { - throw new Error("Can't set property: 'cid' is immutable") - } } exports = module.exports = withIs(DAGLink, { className: 'DAGLink', symbolName: '@ipld/js-ipld-dag-pb/daglink' }) diff --git a/src/dag-node/addLink.js b/src/dag-node/addLink.js index 1f1eca2..24dbc34 100644 --- a/src/dag-node/addLink.js +++ b/src/dag-node/addLink.js @@ -26,8 +26,8 @@ const asDAGLink = (link) => { const addLink = (node, link) => { const dagLink = asDAGLink(link) - node._links.push(dagLink) - node._links = sortLinks(node._links) + node.Links.push(dagLink) + sortLinks.inplace(node.Links) } module.exports = addLink diff --git a/src/dag-node/dagNode.js b/src/dag-node/dagNode.js index 1184580..948dffd 100644 --- a/src/dag-node/dagNode.js +++ b/src/dag-node/dagNode.js @@ -25,24 +25,26 @@ class DAGNode { throw new Error('Passed \'serializedSize\' must be a number!') } - links = links.map((link) => { + links = links.map(link => { return DAGLink.isDAGLink(link) ? link : DAGLink.util.createDagLinkFromB58EncodedHash(link) }) - links = sortLinks(links) + sortLinks.inplace(links) - this._data = data - this._links = links - this._serializedSize = serializedSize - this._size = null + Object.defineProperties(this, { + Data: { value: data, writable: false, enumerable: true }, + Links: { value: links, writable: false, enumerable: true }, + _serializedSize: { value: serializedSize, writable: true, enumerable: false }, + _size: { value: null, writable: true, enumerable: false } + }) } toJSON () { if (!this._json) { this._json = Object.freeze({ data: this.Data, - links: this._links.map((l) => l.toJSON()), + links: this.Links.map((l) => l.toJSON()), size: this.size }) } @@ -75,10 +77,7 @@ class DAGNode { } serialize () { - return serializeDAGNode({ - Data: this._data, - Links: this._links - }) + return serializeDAGNode(this) } get size () { @@ -86,7 +85,7 @@ class DAGNode { if (this._serializedSize === null) { this._serializedSize = this.serialize().length } - this._size = this._links.reduce((sum, l) => sum + l.Tsize, this._serializedSize) + this._size = this.Links.reduce((sum, l) => sum + l.Tsize, this._serializedSize) } return this._size @@ -95,29 +94,6 @@ class DAGNode { set size (size) { throw new Error("Can't set property: 'size' is immutable") } - - // Getters for backwards compatible path resolving - get Data () { - return this._data - } - - set Data (_) { - throw new Error("Can't set property: 'Data' is immutable") - } - - get Links () { - return this._links.map((link) => { - return { - Name: link.Name, - Tsize: link.Tsize, - Hash: link.Hash - } - }) - } - - set Links (_) { - throw new Error("Can't set property: 'Links' is immutable") - } } exports = module.exports = withIs(DAGNode, { className: 'DAGNode', symbolName: '@ipld/js-ipld-dag-pb/dagnode' }) diff --git a/src/dag-node/rmLink.js b/src/dag-node/rmLink.js index 4ccd82d..fc1caa3 100644 --- a/src/dag-node/rmLink.js +++ b/src/dag-node/rmLink.js @@ -4,12 +4,26 @@ const CID = require('cids') const { Buffer } = require('buffer') const rmLink = (dagNode, nameOrCid) => { + let predicate = null + // It's a name if (typeof nameOrCid === 'string') { - dagNode._links = dagNode._links.filter((link) => link.Name !== nameOrCid) + predicate = link => link.Name === nameOrCid } else if (Buffer.isBuffer(nameOrCid) || CID.isCID(nameOrCid)) { - dagNode._links = dagNode._links.filter( - (link) => !link.Hash.equals(nameOrCid)) + predicate = link => link.Hash.equals(nameOrCid) + } + + if (predicate) { + const links = dagNode.Links + let index = 0 + while (index < links.length) { + const link = links[index] + if (predicate(link)) { + links.splice(index, 1) + } else { + index++ + } + } } else { throw new Error('second arg needs to be a name or CID') } diff --git a/src/dag-node/sortLinks.js b/src/dag-node/sortLinks.js index 399b3fa..753b538 100644 --- a/src/dag-node/sortLinks.js +++ b/src/dag-node/sortLinks.js @@ -8,7 +8,7 @@ const linkSort = (a, b) => { } /** - * + * Returns new sorted links array. * @param {Array} links * @returns {Array} */ @@ -16,4 +16,15 @@ const sortLinks = (links) => { return sort(links, linkSort) } +/** + * Sorts links in place (mutating given array) + * @param {Array} links + * @returns {void} + */ +const sortLinksInPlace = (links) => { + sort.inplace(links, linkSort) +} + +sortLinks.inplace = sortLinksInPlace + module.exports = sortLinks diff --git a/src/serialize.js b/src/serialize.js index 68d101d..d5fefb1 100644 --- a/src/serialize.js +++ b/src/serialize.js @@ -9,8 +9,8 @@ exports = module.exports const toProtoBuf = (node) => { const pbn = {} - if (node.Data && node.Data.length > 0) { - pbn.Data = node.Data + if (node.Data && node.Data.byteLength > 0) { + pbn.Data = asBuffer(node.Data) } else { // NOTE: this has to be null in order to match go-ipfs serialization // `null !== new Buffer(0)` @@ -20,7 +20,7 @@ const toProtoBuf = (node) => { if (node.Links && node.Links.length > 0) { pbn.Links = node.Links .map((link) => ({ - Hash: link.Hash.buffer, + Hash: asBuffer(link.Hash.buffer), Name: link.Name, Tsize: link.Tsize })) @@ -31,6 +31,24 @@ const toProtoBuf = (node) => { return pbn } +/** + * Takes bytes in various representations and returns `Buffer` + * view of the underyling data without copying. + * @param {Buffer|ArrayBuffer|ArrayBufferView} bytes + * @returns {Buffer} + */ +const asBuffer = (bytes) => { + if (Buffer.isBuffer(bytes)) { + return bytes + } else if (bytes instanceof ArrayBuffer) { + return Buffer.from(bytes, 0, bytes.byteLength) + } else if (ArrayBuffer.isView(bytes)) { + return Buffer.from(bytes.buffer, bytes.byteOffset, bytes.byteLength) + } else { + return bytes + } +} + /** * Serialize internal representation into a binary PB block. * diff --git a/test/dag-node-test.js b/test/dag-node-test.js index 1c9a0be..f219846 100644 --- a/test/dag-node-test.js +++ b/test/dag-node-test.js @@ -4,8 +4,10 @@ const chai = require('chai') const { Buffer } = require('buffer') const dirtyChai = require('dirty-chai') +const chaiSubset = require('chai-subset') const expect = chai.expect chai.use(dirtyChai) +chai.use(chaiSubset) const dagPB = require('../src') const DAGLink = dagPB.DAGLink @@ -80,7 +82,7 @@ module.exports = (repo) => { }) const node2 = new DAGNode(someData, l2) - expect(node2.Links).to.eql([l1[1], l1[0]]) + expect(node2.Links).to.containSubset([l1[1], l1[0]]) expect(node1.toJSON()).to.eql(node2.toJSON()) // check sorting diff --git a/test/resolver.spec.js b/test/resolver.spec.js index 4139ab9..2818347 100644 --- a/test/resolver.spec.js +++ b/test/resolver.spec.js @@ -5,8 +5,10 @@ const chai = require('chai') const { Buffer } = require('buffer') const dirtyChai = require('dirty-chai') +const chaiSubset = require('chai-subset') const expect = chai.expect chai.use(dirtyChai) +chai.use(chaiSubset) const CID = require('cids') const { DAGNode, resolver } = require('../src') @@ -28,186 +30,217 @@ describe('IPLD Format resolver (local)', () => { return utils.serialize(node) } - const emptyNodeBlob = create(Buffer.alloc(0), []) - const linksNodeBlob = create(Buffer.alloc(0), links) - const dataLinksNodeBlob = create(Buffer.from('aaah the data'), links) - - describe('empty node', () => { - describe('resolver.resolve', () => { - it('links path', () => { - const result = resolver.resolve(emptyNodeBlob, 'Links') - expect(result.value).to.eql([]) - expect(result.remainderPath).to.eql('') - }) - - it('data path', () => { - const result = resolver.resolve(emptyNodeBlob, 'Data') - expect(result.value).to.eql(Buffer.alloc(0)) - expect(result.remainderPath).to.eql('') - }) - - it('non existent path', () => { - expect(() => - resolver.resolve(emptyNodeBlob, 'pathThatDoesNotExist') - ).to.throw( - "Object has no property 'pathThatDoesNotExist'" - ) - }) + const createPlain = (data, links) => { + const node = { + Data: data, + Links: links + } + return utils.serialize(node) + } - it('empty path', () => { - const result = resolver.resolve(emptyNodeBlob, '') - expect(result.value.Data).to.eql(Buffer.alloc(0)) - expect(result.value.Links).to.eql([]) - expect(result.remainderPath).to.eql('') + const emptyNodeBlobs = [ + ['DAGNode', create(Buffer.alloc(0), [])], + ['{Data:Buffer}', createPlain(Buffer.alloc(0), [])], + ['{data:ArrayBuffer}', createPlain(new ArrayBuffer(), [])], + ['{data:Uint8Array}', createPlain(new Uint8Array(), [])] + ] + + const linksNodeBlobs = [ + ['DAGNode', create(Buffer.alloc(0), links)], + ['{Data:Buffer}', createPlain(Buffer.alloc(0), links)], + ['{data:ArrayBuffer}', createPlain(new ArrayBuffer(), links)], + ['{data:Uint8Array}', createPlain(new Uint8Array(), links)] + ] + + const dataLinksNodeBlobs = [ + ['DAGNode', create(Buffer.from('aaah the data'), links)], + ['{Data:Buffer}', createPlain(Buffer.from('aaah the data'), links)], + ['{data:ArrayBuffer}', createPlain(new TextEncoder().encode('aaah the data').buffer, links)], + ['{data:Uint8Array}', createPlain(new TextEncoder().encode('aaah the data'), links)] + ] + + for (const [kind, emptyNodeBlob] of emptyNodeBlobs) { + describe(`empty node (${kind})`, () => { + describe('resolver.resolve', () => { + it('links path', () => { + const result = resolver.resolve(emptyNodeBlob, 'Links') + expect(result.value).to.eql([]) + expect(result.remainderPath).to.eql('') + }) + + it('data path', () => { + const result = resolver.resolve(emptyNodeBlob, 'Data') + expect(result.value).to.eql(Buffer.alloc(0)) + expect(result.remainderPath).to.eql('') + }) + + it('non existent path', () => { + expect(() => + resolver.resolve(emptyNodeBlob, 'pathThatDoesNotExist') + ).to.throw( + "Object has no property 'pathThatDoesNotExist'" + ) + }) + + it('empty path', () => { + const result = resolver.resolve(emptyNodeBlob, '') + expect(result.value.Data).to.eql(Buffer.alloc(0)) + expect(result.value.Links).to.eql([]) + expect(result.remainderPath).to.eql('') + }) + }) + + it('resolver.tree', () => { + const tree = resolver.tree(emptyNodeBlob) + const paths = [...tree] + expect(paths).to.have.members([ + 'Links', + 'Data' + ]) }) }) + } - it('resolver.tree', () => { - const tree = resolver.tree(emptyNodeBlob) - const paths = [...tree] - expect(paths).to.have.members([ - 'Links', - 'Data' - ]) - }) - }) - - describe('links node', () => { - describe('resolver.resolve', () => { - it('links path', () => { - const result = resolver.resolve(linksNodeBlob, 'Links') - expect(result.value).to.eql(links) - expect(result.remainderPath).to.eql('') - }) - - it('links position path Hash', () => { - const result = resolver.resolve(linksNodeBlob, 'Links/1/Hash') - expect(result.value).to.eql(links[1].Hash) - expect(result.remainderPath).to.eql('') - }) - - it('links position path Name', () => { - const result = resolver.resolve(linksNodeBlob, 'Links/1/Name') - expect(result.value).to.eql(links[1].Name) - expect(result.remainderPath).to.eql('') - }) - - it('links position path Tsize', () => { - const result = resolver.resolve(linksNodeBlob, 'Links/1/Tsize') - expect(result.value).to.eql(links[1].Tsize) - expect(result.remainderPath).to.eql('') - }) - - it('links by name', () => { - const result = resolver.resolve(linksNodeBlob, 'named link') - expect(result.value.equals(links[1].Hash)).to.be.true() - expect(result.remainderPath).to.eql('') - }) - - it('missing link by name', () => { - expect(() => - resolver.resolve(linksNodeBlob, 'missing link') - ).to.throw( - "Object has no property 'missing link'" - ) - }) - - it('yield remainderPath if impossible to resolve through (a)', () => { - const result = resolver.resolve(linksNodeBlob, 'Links/1/Hash/Data') - expect(result.value.equals( - new CID('QmXg9Pp2ytZ14xgmQjYEiHjVjMFXzCVVEcRTWJBmLgR39V') - )).to.be.true() - expect(result.remainderPath).to.equal('Data') - }) - - it('yield remainderPath if impossible to resolve through (b)', () => { - const result = resolver.resolve(linksNodeBlob, 'Links/1/Hash/Links/0/Hash/Data') - expect(result.value.equals( - new CID('QmXg9Pp2ytZ14xgmQjYEiHjVjMFXzCVVEcRTWJBmLgR39V') - )).to.be.true() - expect(result.remainderPath).to.equal('Links/0/Hash/Data') - }) - - it('yield remainderPath if impossible to resolve through named link (a)', () => { - const result = resolver.resolve(linksNodeBlob, 'named link/Data') - expect(result.value.equals( - new CID('QmXg9Pp2ytZ14xgmQjYEiHjVjMFXzCVVEcRTWJBmLgR39V') - )).to.be.true() - expect(result.remainderPath).to.equal('Data') - }) - - it('yield remainderPath if impossible to resolve through named link (b)', () => { - const result = resolver.resolve(linksNodeBlob, 'named link/Links/0/Hash/Data') - expect(result.value.equals( - new CID('QmXg9Pp2ytZ14xgmQjYEiHjVjMFXzCVVEcRTWJBmLgR39V') - )).to.be.true() - expect(result.remainderPath).to.equal('Links/0/Hash/Data') + for (const [kind, linksNodeBlob] of linksNodeBlobs) { + describe(`links node ${kind}`, () => { + describe('resolver.resolve', () => { + it('links path', () => { + const result = resolver.resolve(linksNodeBlob, 'Links') + expect(result.value).to.containSubset(links) + expect(result.remainderPath).to.eql('') + }) + + it('links position path Hash', () => { + const result = resolver.resolve(linksNodeBlob, 'Links/1/Hash') + expect(result.value).to.eql(links[1].Hash) + expect(result.remainderPath).to.eql('') + }) + + it('links position path Name', () => { + const result = resolver.resolve(linksNodeBlob, 'Links/1/Name') + expect(result.value).to.eql(links[1].Name) + expect(result.remainderPath).to.eql('') + }) + + it('links position path Tsize', () => { + const result = resolver.resolve(linksNodeBlob, 'Links/1/Tsize') + expect(result.value).to.eql(links[1].Tsize) + expect(result.remainderPath).to.eql('') + }) + + it('links by name', () => { + const result = resolver.resolve(linksNodeBlob, 'named link') + expect(result.value.equals(links[1].Hash)).to.be.true() + expect(result.remainderPath).to.eql('') + }) + + it('missing link by name', () => { + expect(() => + resolver.resolve(linksNodeBlob, 'missing link') + ).to.throw( + "Object has no property 'missing link'" + ) + }) + + it('yield remainderPath if impossible to resolve through (a)', () => { + const result = resolver.resolve(linksNodeBlob, 'Links/1/Hash/Data') + expect(result.value.equals( + new CID('QmXg9Pp2ytZ14xgmQjYEiHjVjMFXzCVVEcRTWJBmLgR39V') + )).to.be.true() + expect(result.remainderPath).to.equal('Data') + }) + + it('yield remainderPath if impossible to resolve through (b)', () => { + const result = resolver.resolve(linksNodeBlob, 'Links/1/Hash/Links/0/Hash/Data') + expect(result.value.equals( + new CID('QmXg9Pp2ytZ14xgmQjYEiHjVjMFXzCVVEcRTWJBmLgR39V') + )).to.be.true() + expect(result.remainderPath).to.equal('Links/0/Hash/Data') + }) + + it('yield remainderPath if impossible to resolve through named link (a)', () => { + const result = resolver.resolve(linksNodeBlob, 'named link/Data') + expect(result.value.equals( + new CID('QmXg9Pp2ytZ14xgmQjYEiHjVjMFXzCVVEcRTWJBmLgR39V') + )).to.be.true() + expect(result.remainderPath).to.equal('Data') + }) + + it('yield remainderPath if impossible to resolve through named link (b)', () => { + const result = resolver.resolve(linksNodeBlob, 'named link/Links/0/Hash/Data') + expect(result.value.equals( + new CID('QmXg9Pp2ytZ14xgmQjYEiHjVjMFXzCVVEcRTWJBmLgR39V') + )).to.be.true() + expect(result.remainderPath).to.equal('Links/0/Hash/Data') + }) + }) + + it('resolver.tree', () => { + const tree = resolver.tree(linksNodeBlob) + const paths = [...tree] + expect(paths).to.have.members([ + 'Links', + 'Links/0', + 'Links/0/Name', + 'Links/0/Tsize', + 'Links/0/Hash', + 'Links/1', + 'Links/1/Name', + 'Links/1/Tsize', + 'Links/1/Hash', + 'Data' + ]) }) }) + } - it('resolver.tree', () => { - const tree = resolver.tree(linksNodeBlob) - const paths = [...tree] - expect(paths).to.have.members([ - 'Links', - 'Links/0', - 'Links/0/Name', - 'Links/0/Tsize', - 'Links/0/Hash', - 'Links/1', - 'Links/1/Name', - 'Links/1/Tsize', - 'Links/1/Hash', - 'Data' - ]) - }) - }) - - describe('links and data node', () => { - describe('resolver.resolve', () => { - it('links path', () => { - const result = resolver.resolve(dataLinksNodeBlob, 'Links') - expect(result.value).to.eql(links) - expect(result.remainderPath).to.eql('') - }) - - it('data path', () => { - const result = resolver.resolve(dataLinksNodeBlob, 'Data') - expect(result.value).to.eql(Buffer.from('aaah the data')) - expect(result.remainderPath).to.eql('') - }) - - it('non existent path', () => { - expect(() => - resolver.resolve(dataLinksNodeBlob, 'pathThatDoesNotExist') - ).to.throw( - "Object has no property 'pathThatDoesNotExist'" - ) - }) - - it('empty path', () => { - const result = resolver.resolve(dataLinksNodeBlob, '') - expect(result.value.Data).to.eql(Buffer.from('aaah the data')) - expect(result.value.Links).to.eql(links) - expect(result.remainderPath).to.eql('') + for (const [kind, dataLinksNodeBlob] of dataLinksNodeBlobs) { + describe(`links and data node (${kind})`, () => { + describe('resolver.resolve', () => { + it('links path', () => { + const result = resolver.resolve(dataLinksNodeBlob, 'Links') + expect(result.value).to.containSubset(links) + expect(result.remainderPath).to.eql('') + }) + + it('data path', () => { + const result = resolver.resolve(dataLinksNodeBlob, 'Data') + expect(result.value).to.eql(Buffer.from('aaah the data')) + expect(result.remainderPath).to.eql('') + }) + + it('non existent path', () => { + expect(() => + resolver.resolve(dataLinksNodeBlob, 'pathThatDoesNotExist') + ).to.throw( + "Object has no property 'pathThatDoesNotExist'" + ) + }) + + it('empty path', () => { + const result = resolver.resolve(dataLinksNodeBlob, '') + expect(result.value.Data).to.eql(Buffer.from('aaah the data')) + expect(result.value.Links).to.containSubset(links) + expect(result.remainderPath).to.eql('') + }) + }) + + it('resolver.tree', () => { + const tree = resolver.tree(dataLinksNodeBlob) + const paths = [...tree] + expect(paths).to.have.members([ + 'Links', + 'Links/0', + 'Links/0/Name', + 'Links/0/Tsize', + 'Links/0/Hash', + 'Links/1', + 'Links/1/Name', + 'Links/1/Tsize', + 'Links/1/Hash', + 'Data' + ]) }) }) - - it('resolver.tree', () => { - const tree = resolver.tree(dataLinksNodeBlob) - const paths = [...tree] - expect(paths).to.have.members([ - 'Links', - 'Links/0', - 'Links/0/Name', - 'Links/0/Tsize', - 'Links/0/Hash', - 'Links/1', - 'Links/1/Name', - 'Links/1/Tsize', - 'Links/1/Hash', - 'Data' - ]) - }) - }) + } }) diff --git a/test/util.spec.js b/test/util.spec.js index 1b91578..8460474 100644 --- a/test/util.spec.js +++ b/test/util.spec.js @@ -5,9 +5,11 @@ const CID = require('cids') const { Buffer } = require('buffer') const chai = require('chai') +const chaiSubset = require('chai-subset') const dirtyChai = require('dirty-chai') const expect = chai.expect chai.use(dirtyChai) +chai.use(chaiSubset) const { DAGLink @@ -33,6 +35,24 @@ describe('util', () => { expect(node.Data).to.deep.equal(data) }) + it('should serialize a node with ArrayBuffer data', () => { + const data = Uint8Array.from([0, 1, 2, 3]).buffer + const result = serialize({ Data: data }) + expect(result).to.be.an.instanceof(Uint8Array) + + const node = deserialize(result) + expect(node.Data).to.deep.equal(Buffer.from([0, 1, 2, 3])) + }) + + it('should serialize a node with Uint8Array data', () => { + const data = Uint8Array.from([0, 1, 2, 3]) + const result = serialize({ Data: data }) + expect(result).to.be.an.instanceof(Uint8Array) + + const node = deserialize(result) + expect(node.Data).to.deep.equal(Buffer.from([0, 1, 2, 3])) + }) + it('should serialize a node with links', () => { const links = [ new DAGLink('', 0, 'QmWDtUQj38YLW8v3q4A6LwPn4vYKEbuKWpgSm6bjKW6Xfe') @@ -41,7 +61,7 @@ describe('util', () => { expect(result).to.be.an.instanceof(Uint8Array) const node = deserialize(result) - expect(node.Links).to.deep.equal([{ + expect(node.Links).to.containSubset([{ Name: '', Tsize: 0, Hash: new CID('QmWDtUQj38YLW8v3q4A6LwPn4vYKEbuKWpgSm6bjKW6Xfe') @@ -58,7 +78,7 @@ describe('util', () => { expect(result).to.be.an.instanceof(Uint8Array) const node = deserialize(result) - expect(node.Links).to.deep.equal(links) + expect(node.Links).to.containSubset(links) }) it('should ignore invalid properties when serializing', () => { From 2eaff18be15a1b19e3b7c27d5204e94c66a557ca Mon Sep 17 00:00:00 2001 From: Irakli Gozalishvili Date: Fri, 5 Jun 2020 15:10:52 -0700 Subject: [PATCH 2/8] fix: lack of TextEncoder in older nodejs --- test/resolver.spec.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/test/resolver.spec.js b/test/resolver.spec.js index 2818347..91391c3 100644 --- a/test/resolver.spec.js +++ b/test/resolver.spec.js @@ -55,8 +55,8 @@ describe('IPLD Format resolver (local)', () => { const dataLinksNodeBlobs = [ ['DAGNode', create(Buffer.from('aaah the data'), links)], ['{Data:Buffer}', createPlain(Buffer.from('aaah the data'), links)], - ['{data:ArrayBuffer}', createPlain(new TextEncoder().encode('aaah the data').buffer, links)], - ['{data:Uint8Array}', createPlain(new TextEncoder().encode('aaah the data'), links)] + ['{data:ArrayBuffer}', createPlain(Uint8Array.from(Buffer.from('aaah the data')).buffer, links)], + ['{data:Uint8Array}', createPlain(Uint8Array.from(Buffer.from('aaah the data')), links)] ] for (const [kind, emptyNodeBlob] of emptyNodeBlobs) { From b25eb3c8ca3136c85321b5885897e7b6804fe269 Mon Sep 17 00:00:00 2001 From: Irakli Gozalishvili Date: Fri, 5 Jun 2020 15:28:25 -0700 Subject: [PATCH 3/8] fix: immutable CID test --- test/dag-link-test.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/test/dag-link-test.js b/test/dag-link-test.js index 2e23243..f020b61 100644 --- a/test/dag-link-test.js +++ b/test/dag-link-test.js @@ -63,7 +63,7 @@ module.exports = (repo) => { it('has an immutable CID', () => { const link = new DAGLink('hello', 3, 'QmXg9Pp2ytZ14xgmQjYEiHjVjMFXzCVVEcRTWJBmLgR39U') - expect(() => { link.Hash = 'foo' }).to.throw(/property/) + expect(() => { link.Hash = 'foo' }).to.throw(/read.only/) }) }) } From 221b52c2baa5f99452d95f5b430478cc6c34cfc8 Mon Sep 17 00:00:00 2001 From: Irakli Gozalishvili Date: Mon, 8 Jun 2020 16:41:25 -0700 Subject: [PATCH 4/8] chore: use chai from aegir --- package.json | 3 --- test/dag-link-test.js | 4 +--- test/dag-node-test.js | 6 +----- test/mod.spec.js | 2 +- test/resolver.spec.js | 6 +----- test/util.spec.js | 6 +----- 6 files changed, 5 insertions(+), 22 deletions(-) diff --git a/package.json b/package.json index 22a0688..d5ba370 100644 --- a/package.json +++ b/package.json @@ -74,9 +74,6 @@ }, "devDependencies": { "aegir": "^22.0.0", - "chai": "^4.2.0", - "dirty-chai": "^2.0.1", - "chai-subset": "1.6.0", "fs-extra": "^9.0.0", "ipfs-block-service": "~0.17.0", "ipfs-repo": "^3.0.0", diff --git a/test/dag-link-test.js b/test/dag-link-test.js index f020b61..232fec3 100644 --- a/test/dag-link-test.js +++ b/test/dag-link-test.js @@ -1,11 +1,9 @@ /* eslint-env mocha */ 'use strict' -const chai = require('chai') +const chai = require('aegir/utils/chai') const { Buffer } = require('buffer') -const dirtyChai = require('dirty-chai') const expect = chai.expect -chai.use(dirtyChai) const CID = require('cids') const DAGLink = require('../src').DAGLink diff --git a/test/dag-node-test.js b/test/dag-node-test.js index f219846..b4c86fc 100644 --- a/test/dag-node-test.js +++ b/test/dag-node-test.js @@ -1,13 +1,9 @@ /* eslint-env mocha */ 'use strict' -const chai = require('chai') +const chai = require('aegir/utils/chai') const { Buffer } = require('buffer') -const dirtyChai = require('dirty-chai') -const chaiSubset = require('chai-subset') const expect = chai.expect -chai.use(dirtyChai) -chai.use(chaiSubset) const dagPB = require('../src') const DAGLink = dagPB.DAGLink diff --git a/test/mod.spec.js b/test/mod.spec.js index c565b71..da23524 100644 --- a/test/mod.spec.js +++ b/test/mod.spec.js @@ -1,7 +1,7 @@ /* eslint-env mocha */ 'use strict' -const chai = require('chai') +const chai = require('aegir/utils/chai') const expect = chai.expect const multicodec = require('multicodec') diff --git a/test/resolver.spec.js b/test/resolver.spec.js index 91391c3..f1743fe 100644 --- a/test/resolver.spec.js +++ b/test/resolver.spec.js @@ -2,13 +2,9 @@ 'use strict' -const chai = require('chai') +const chai = require('aegir/utils/chai') const { Buffer } = require('buffer') -const dirtyChai = require('dirty-chai') -const chaiSubset = require('chai-subset') const expect = chai.expect -chai.use(dirtyChai) -chai.use(chaiSubset) const CID = require('cids') const { DAGNode, resolver } = require('../src') diff --git a/test/util.spec.js b/test/util.spec.js index 8460474..fe6540e 100644 --- a/test/util.spec.js +++ b/test/util.spec.js @@ -4,12 +4,8 @@ const CID = require('cids') const { Buffer } = require('buffer') -const chai = require('chai') -const chaiSubset = require('chai-subset') -const dirtyChai = require('dirty-chai') +const chai = require('aegir/utils/chai') const expect = chai.expect -chai.use(dirtyChai) -chai.use(chaiSubset) const { DAGLink From fba99fa80a93af178210ea7a3aaf0775e28abc21 Mon Sep 17 00:00:00 2001 From: Irakli Gozalishvili Date: Mon, 8 Jun 2020 16:50:11 -0700 Subject: [PATCH 5/8] chore: migrate Uint8Array support into protons lib --- package.json | 2 +- src/serialize.js | 22 ++-------------------- test/resolver.spec.js | 3 --- test/util.spec.js | 9 --------- 4 files changed, 3 insertions(+), 33 deletions(-) diff --git a/package.json b/package.json index d5ba370..72515b7 100644 --- a/package.json +++ b/package.json @@ -69,7 +69,7 @@ "class-is": "^1.1.0", "multicodec": "^1.0.1", "multihashing-async": "~0.8.1", - "protons": "^1.0.2", + "protons": "git://github.com/gozala/protons#uint8array", "stable": "^0.1.8" }, "devDependencies": { diff --git a/src/serialize.js b/src/serialize.js index d5fefb1..fb4aec1 100644 --- a/src/serialize.js +++ b/src/serialize.js @@ -10,7 +10,7 @@ const toProtoBuf = (node) => { const pbn = {} if (node.Data && node.Data.byteLength > 0) { - pbn.Data = asBuffer(node.Data) + pbn.Data = node.Data } else { // NOTE: this has to be null in order to match go-ipfs serialization // `null !== new Buffer(0)` @@ -20,7 +20,7 @@ const toProtoBuf = (node) => { if (node.Links && node.Links.length > 0) { pbn.Links = node.Links .map((link) => ({ - Hash: asBuffer(link.Hash.buffer), + Hash: link.Hash.buffer, Name: link.Name, Tsize: link.Tsize })) @@ -31,24 +31,6 @@ const toProtoBuf = (node) => { return pbn } -/** - * Takes bytes in various representations and returns `Buffer` - * view of the underyling data without copying. - * @param {Buffer|ArrayBuffer|ArrayBufferView} bytes - * @returns {Buffer} - */ -const asBuffer = (bytes) => { - if (Buffer.isBuffer(bytes)) { - return bytes - } else if (bytes instanceof ArrayBuffer) { - return Buffer.from(bytes, 0, bytes.byteLength) - } else if (ArrayBuffer.isView(bytes)) { - return Buffer.from(bytes.buffer, bytes.byteOffset, bytes.byteLength) - } else { - return bytes - } -} - /** * Serialize internal representation into a binary PB block. * diff --git a/test/resolver.spec.js b/test/resolver.spec.js index f1743fe..3f9b8cd 100644 --- a/test/resolver.spec.js +++ b/test/resolver.spec.js @@ -37,21 +37,18 @@ describe('IPLD Format resolver (local)', () => { const emptyNodeBlobs = [ ['DAGNode', create(Buffer.alloc(0), [])], ['{Data:Buffer}', createPlain(Buffer.alloc(0), [])], - ['{data:ArrayBuffer}', createPlain(new ArrayBuffer(), [])], ['{data:Uint8Array}', createPlain(new Uint8Array(), [])] ] const linksNodeBlobs = [ ['DAGNode', create(Buffer.alloc(0), links)], ['{Data:Buffer}', createPlain(Buffer.alloc(0), links)], - ['{data:ArrayBuffer}', createPlain(new ArrayBuffer(), links)], ['{data:Uint8Array}', createPlain(new Uint8Array(), links)] ] const dataLinksNodeBlobs = [ ['DAGNode', create(Buffer.from('aaah the data'), links)], ['{Data:Buffer}', createPlain(Buffer.from('aaah the data'), links)], - ['{data:ArrayBuffer}', createPlain(Uint8Array.from(Buffer.from('aaah the data')).buffer, links)], ['{data:Uint8Array}', createPlain(Uint8Array.from(Buffer.from('aaah the data')), links)] ] diff --git a/test/util.spec.js b/test/util.spec.js index fe6540e..46ba682 100644 --- a/test/util.spec.js +++ b/test/util.spec.js @@ -31,15 +31,6 @@ describe('util', () => { expect(node.Data).to.deep.equal(data) }) - it('should serialize a node with ArrayBuffer data', () => { - const data = Uint8Array.from([0, 1, 2, 3]).buffer - const result = serialize({ Data: data }) - expect(result).to.be.an.instanceof(Uint8Array) - - const node = deserialize(result) - expect(node.Data).to.deep.equal(Buffer.from([0, 1, 2, 3])) - }) - it('should serialize a node with Uint8Array data', () => { const data = Uint8Array.from([0, 1, 2, 3]) const result = serialize({ Data: data }) From 750620f33aebd0bc1868d8719b8e8654dc39eca7 Mon Sep 17 00:00:00 2001 From: Irakli Gozalishvili Date: Mon, 8 Jun 2020 17:31:52 -0700 Subject: [PATCH 6/8] chore: remove immutable sortLinks --- src/dag-node/addLink.js | 2 +- src/dag-node/dagNode.js | 2 +- src/dag-node/sortLinks.js | 13 +------------ 3 files changed, 3 insertions(+), 14 deletions(-) diff --git a/src/dag-node/addLink.js b/src/dag-node/addLink.js index 24dbc34..032b216 100644 --- a/src/dag-node/addLink.js +++ b/src/dag-node/addLink.js @@ -27,7 +27,7 @@ const asDAGLink = (link) => { const addLink = (node, link) => { const dagLink = asDAGLink(link) node.Links.push(dagLink) - sortLinks.inplace(node.Links) + sortLinks(node.Links) } module.exports = addLink diff --git a/src/dag-node/dagNode.js b/src/dag-node/dagNode.js index 948dffd..841211e 100644 --- a/src/dag-node/dagNode.js +++ b/src/dag-node/dagNode.js @@ -30,7 +30,7 @@ class DAGNode { ? link : DAGLink.util.createDagLinkFromB58EncodedHash(link) }) - sortLinks.inplace(links) + sortLinks(links) Object.defineProperties(this, { Data: { value: data, writable: false, enumerable: true }, diff --git a/src/dag-node/sortLinks.js b/src/dag-node/sortLinks.js index 753b538..6f39161 100644 --- a/src/dag-node/sortLinks.js +++ b/src/dag-node/sortLinks.js @@ -7,24 +7,13 @@ const linkSort = (a, b) => { return Buffer.compare(a.nameAsBuffer, b.nameAsBuffer) } -/** - * Returns new sorted links array. - * @param {Array} links - * @returns {Array} - */ -const sortLinks = (links) => { - return sort(links, linkSort) -} - /** * Sorts links in place (mutating given array) * @param {Array} links * @returns {void} */ -const sortLinksInPlace = (links) => { +const sortLinks = (links) => { sort.inplace(links, linkSort) } -sortLinks.inplace = sortLinksInPlace - module.exports = sortLinks From a8f448890311beaf4b86c441b2ab5fffdda43c9c Mon Sep 17 00:00:00 2001 From: Irakli Gozalishvili Date: Wed, 24 Jun 2020 15:18:36 -0700 Subject: [PATCH 7/8] chore: update protons to release with Uint8Array --- package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/package.json b/package.json index 72515b7..8e03706 100644 --- a/package.json +++ b/package.json @@ -69,7 +69,7 @@ "class-is": "^1.1.0", "multicodec": "^1.0.1", "multihashing-async": "~0.8.1", - "protons": "git://github.com/gozala/protons#uint8array", + "protons": "^1.2.1", "stable": "^0.1.8" }, "devDependencies": { From 541faea061b23d009be43619885554757311402d Mon Sep 17 00:00:00 2001 From: Irakli Gozalishvili Date: Wed, 24 Jun 2020 15:22:06 -0700 Subject: [PATCH 8/8] chore: change formatting as per review feedback --- src/dag-node/dagNode.js | 2 +- src/dag-node/rmLink.js | 4 ++-- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/src/dag-node/dagNode.js b/src/dag-node/dagNode.js index 841211e..878fb2b 100644 --- a/src/dag-node/dagNode.js +++ b/src/dag-node/dagNode.js @@ -25,7 +25,7 @@ class DAGNode { throw new Error('Passed \'serializedSize\' must be a number!') } - links = links.map(link => { + links = links.map((link) => { return DAGLink.isDAGLink(link) ? link : DAGLink.util.createDagLinkFromB58EncodedHash(link) diff --git a/src/dag-node/rmLink.js b/src/dag-node/rmLink.js index fc1caa3..7822a2f 100644 --- a/src/dag-node/rmLink.js +++ b/src/dag-node/rmLink.js @@ -8,9 +8,9 @@ const rmLink = (dagNode, nameOrCid) => { // It's a name if (typeof nameOrCid === 'string') { - predicate = link => link.Name === nameOrCid + predicate = (link) => link.Name === nameOrCid } else if (Buffer.isBuffer(nameOrCid) || CID.isCID(nameOrCid)) { - predicate = link => link.Hash.equals(nameOrCid) + predicate = (link) => link.Hash.equals(nameOrCid) } if (predicate) {