From b5486b6753508d83e710ba2c66be4a8e8ffc4de7 Mon Sep 17 00:00:00 2001 From: Richard Hansen Date: Tue, 12 Oct 2021 03:01:15 -0400 Subject: [PATCH] Changeset: Migrate from `opAssembler()` to `serializeOps()` --- CHANGELOG.md | 1 + src/node/utils/padDiff.js | 26 +++++++++---------- src/static/js/Changeset.js | 22 ++++++++++------ .../frontend/specs/easysync-assembler.js | 6 ++--- 4 files changed, 29 insertions(+), 26 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 14653a33c..8a520b27d 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -36,6 +36,7 @@ * `opAttributeValue()` * `opIterator()`: Deprecated in favor of the new `deserializeOps()` generator function. + * `opAssembler()`: Deprecated in favor of the new `serializeOps()` function. * `appendATextToAssembler()`: Deprecated in favor of the new `opsFromAText()` generator function. * `newOp()`: Deprecated in favor of the new `Op` class. diff --git a/src/node/utils/padDiff.js b/src/node/utils/padDiff.js index 4ab276b4b..dd93e84ff 100644 --- a/src/node/utils/padDiff.js +++ b/src/node/utils/padDiff.js @@ -206,28 +206,26 @@ PadDiff.prototype._extendChangesetWithAuthor = (changeset, author, apool) => { // unpack const unpacked = Changeset.unpack(changeset); - const assem = Changeset.opAssembler(); - // create deleted attribs const authorAttrib = apool.putAttrib(['author', author || '']); const deletedAttrib = apool.putAttrib(['removed', true]); const attribs = `*${Changeset.numToString(authorAttrib)}*${Changeset.numToString(deletedAttrib)}`; - for (const operator of Changeset.deserializeOps(unpacked.ops)) { - if (operator.opcode === '-') { - // this is a delete operator, extend it with the author - operator.attribs = attribs; - } else if (operator.opcode === '=' && operator.attribs) { - // this is operator changes only attributes, let's mark which author did that - operator.attribs += `*${Changeset.numToString(authorAttrib)}`; + const serializedOps = Changeset.serializeOps((function* () { + for (const operator of Changeset.deserializeOps(unpacked.ops)) { + if (operator.opcode === '-') { + // this is a delete operator, extend it with the author + operator.attribs = attribs; + } else if (operator.opcode === '=' && operator.attribs) { + // this is operator changes only attributes, let's mark which author did that + operator.attribs += `*${Changeset.numToString(authorAttrib)}`; + } + yield operator; } - - // append the new operator to our assembler - assem.append(operator); - } + })()); // return the modified changeset - return Changeset.pack(unpacked.oldLen, unpacked.newLen, assem.toString(), unpacked.charBank); + return Changeset.pack(unpacked.oldLen, unpacked.newLen, serializedOps, unpacked.charBank); }; // this method is 80% like Changeset.inverse. I just changed so instead of reverting, diff --git a/src/static/js/Changeset.js b/src/static/js/Changeset.js index 6761efef2..3b09c7426 100644 --- a/src/static/js/Changeset.js +++ b/src/static/js/Changeset.js @@ -297,7 +297,7 @@ const copyOp = (op1, op2 = new Op()) => Object.assign(op2, op1); * @param {Iterable} ops - Iterable of operations to serialize. * @returns {string} A string containing the encoded op data (example: '|5=2p=v*4*5+1'). */ -const serializeOps = (ops) => { +exports.serializeOps = (ops) => { let res = ''; for (const op of ops) res += op.toString(); return res; @@ -305,6 +305,8 @@ const serializeOps = (ops) => { /** * Serializes a sequence of Ops. + * + * @deprecated Use `serializeOps` instead. */ class OpAssembler { constructor() { @@ -324,7 +326,7 @@ class OpAssembler { } toString() { - return serializeOps(this._ops); + return exports.serializeOps(this._ops); } } @@ -334,12 +336,11 @@ class OpAssembler { */ class MergingOpAssembler { constructor() { - this._assem = new OpAssembler(); this.clear(); } clear() { - this._assem.clear(); + this._assem = []; this._bufOp = new Op(); // If we get, for example, insertions [xxx\n,yyy], those don't merge, but if we get // [xxx\n,yyy,zzz\n], that merges to [xxx\nyyyzzz\n]. This variable stores the length of yyy and @@ -352,11 +353,11 @@ class MergingOpAssembler { if (isEndDocument && this._bufOp.opcode === '=' && !this._bufOp.attribs) { // final merged keep, leave it implicit } else { - this._assem.append(this._bufOp); + this._assem.push(copyOp(this._bufOp)); if (this._bufOpAdditionalCharsAfterNewline) { this._bufOp.chars = this._bufOpAdditionalCharsAfterNewline; this._bufOp.lines = 0; - this._assem.append(this._bufOp); + this._assem.push(copyOp(this._bufOp)); this._bufOpAdditionalCharsAfterNewline = 0; } } @@ -390,7 +391,7 @@ class MergingOpAssembler { toString() { this._flush(); - return this._assem.toString(); + return exports.serializeOps(this._assem); } } @@ -590,9 +591,14 @@ exports.smartOpAssembler = () => new SmartOpAssembler(); exports.mergingOpAssembler = () => new MergingOpAssembler(); /** + * @deprecated Use `serializeOps` instead. * @returns {OpAssembler} */ -exports.opAssembler = () => new OpAssembler(); +exports.opAssembler = () => { + padutils.warnDeprecated( + 'Changeset.opAssembler() is deprecated; use Changeset.serializeOps() instead'); + return new OpAssembler(); +}; /** * A custom made String Iterator diff --git a/src/tests/frontend/specs/easysync-assembler.js b/src/tests/frontend/specs/easysync-assembler.js index d9ce04ae2..b035b64e4 100644 --- a/src/tests/frontend/specs/easysync-assembler.js +++ b/src/tests/frontend/specs/easysync-assembler.js @@ -3,11 +3,9 @@ const Changeset = require('../../../static/js/Changeset'); describe('easysync-assembler', function () { - it('opAssembler', async function () { + it('deserialize and serialize', async function () { const x = '-c*3*4+6|3=az*asdf0*1*2*3+1=1-1+1*0+1=1-1+1|c=c-1'; - const assem = Changeset.opAssembler(); - for (const op of Changeset.deserializeOps(x)) assem.append(op); - expect(assem.toString()).to.equal(x); + expect(Changeset.serializeOps(Changeset.deserializeOps(x))).to.equal(x); }); it('smartOpAssembler', async function () {