diff --git a/src/node/utils/Minify.js b/src/node/utils/Minify.js
index 58dba2097a1..2e8a2d960fc 100644
--- a/src/node/utils/Minify.js
+++ b/src/node/utils/Minify.js
@@ -166,8 +166,8 @@ const minify = async (req, res) => {
filename = path.join('../node_modules/', library, libraryPath);
}
}
- const [, spec] = /^plugins\/ep_etherpad-lite\/(tests\/frontend\/specs\/.*)/.exec(filename) || [];
- if (spec != null) filename = `../${spec}`;
+ const [, testf] = /^plugins\/ep_etherpad-lite\/(tests\/frontend\/.*)/.exec(filename) || [];
+ if (testf != null) filename = `../${testf}`;
const contentType = mime.lookup(filename);
diff --git a/src/tests/frontend/easysync-helper.js b/src/tests/frontend/easysync-helper.js
new file mode 100644
index 00000000000..6b4bfc95982
--- /dev/null
+++ b/src/tests/frontend/easysync-helper.js
@@ -0,0 +1,222 @@
+'use strict';
+
+const Changeset = require('../../static/js/Changeset');
+const AttributePool = require('../../static/js/AttributePool');
+
+const randInt = (maxValue) => Math.floor(Math.random() * maxValue);
+
+const poolOrArray = (attribs) => {
+ if (attribs.getAttrib) {
+ return attribs; // it's already an attrib pool
+ } else {
+ // assume it's an array of attrib strings to be split and added
+ const p = new AttributePool();
+ attribs.forEach((kv) => {
+ p.putAttrib(kv.split(','));
+ });
+ return p;
+ }
+};
+exports.poolOrArray = poolOrArray;
+
+const randomInlineString = (len) => {
+ const assem = Changeset.stringAssembler();
+ for (let i = 0; i < len; i++) {
+ assem.append(String.fromCharCode(randInt(26) + 97));
+ }
+ return assem.toString();
+};
+
+const randomMultiline = (approxMaxLines, approxMaxCols) => {
+ const numParts = randInt(approxMaxLines * 2) + 1;
+ const txt = Changeset.stringAssembler();
+ txt.append(randInt(2) ? '\n' : '');
+ for (let i = 0; i < numParts; i++) {
+ if ((i % 2) === 0) {
+ if (randInt(10)) {
+ txt.append(randomInlineString(randInt(approxMaxCols) + 1));
+ } else {
+ txt.append('\n');
+ }
+ } else {
+ txt.append('\n');
+ }
+ }
+ return txt.toString();
+};
+exports.randomMultiline = randomMultiline;
+
+const randomStringOperation = (numCharsLeft) => {
+ let result;
+ switch (randInt(9)) {
+ case 0:
+ {
+ // insert char
+ result = {
+ insert: randomInlineString(1),
+ };
+ break;
+ }
+ case 1:
+ {
+ // delete char
+ result = {
+ remove: 1,
+ };
+ break;
+ }
+ case 2:
+ {
+ // skip char
+ result = {
+ skip: 1,
+ };
+ break;
+ }
+ case 3:
+ {
+ // insert small
+ result = {
+ insert: randomInlineString(randInt(4) + 1),
+ };
+ break;
+ }
+ case 4:
+ {
+ // delete small
+ result = {
+ remove: randInt(4) + 1,
+ };
+ break;
+ }
+ case 5:
+ {
+ // skip small
+ result = {
+ skip: randInt(4) + 1,
+ };
+ break;
+ }
+ case 6:
+ {
+ // insert multiline;
+ result = {
+ insert: randomMultiline(5, 20),
+ };
+ break;
+ }
+ case 7:
+ {
+ // delete multiline
+ result = {
+ remove: Math.round(numCharsLeft * Math.random() * Math.random()),
+ };
+ break;
+ }
+ case 8:
+ {
+ // skip multiline
+ result = {
+ skip: Math.round(numCharsLeft * Math.random() * Math.random()),
+ };
+ break;
+ }
+ case 9:
+ {
+ // delete to end
+ result = {
+ remove: numCharsLeft,
+ };
+ break;
+ }
+ case 10:
+ {
+ // skip to end
+ result = {
+ skip: numCharsLeft,
+ };
+ break;
+ }
+ }
+ const maxOrig = numCharsLeft - 1;
+ if ('remove' in result) {
+ result.remove = Math.min(result.remove, maxOrig);
+ } else if ('skip' in result) {
+ result.skip = Math.min(result.skip, maxOrig);
+ }
+ return result;
+};
+
+const randomTwoPropAttribs = (opcode) => {
+ // assumes attrib pool like ['apple,','apple,true','banana,','banana,true']
+ if (opcode === '-' || randInt(3)) {
+ return '';
+ } else if (randInt(3)) { // eslint-disable-line no-dupe-else-if
+ if (opcode === '+' || randInt(2)) {
+ return `*${Changeset.numToString(randInt(2) * 2 + 1)}`;
+ } else {
+ return `*${Changeset.numToString(randInt(2) * 2)}`;
+ }
+ } else if (opcode === '+' || randInt(4) === 0) {
+ return '*1*3';
+ } else {
+ return ['*0*2', '*0*3', '*1*2'][randInt(3)];
+ }
+};
+
+const randomTestChangeset = (origText, withAttribs) => {
+ const charBank = Changeset.stringAssembler();
+ let textLeft = origText; // always keep final newline
+ const outTextAssem = Changeset.stringAssembler();
+ const opAssem = Changeset.smartOpAssembler();
+ const oldLen = origText.length;
+
+ const nextOp = new Changeset.Op();
+
+ const appendMultilineOp = (opcode, txt) => {
+ nextOp.opcode = opcode;
+ if (withAttribs) {
+ nextOp.attribs = randomTwoPropAttribs(opcode);
+ }
+ txt.replace(/\n|[^\n]+/g, (t) => {
+ if (t === '\n') {
+ nextOp.chars = 1;
+ nextOp.lines = 1;
+ opAssem.append(nextOp);
+ } else {
+ nextOp.chars = t.length;
+ nextOp.lines = 0;
+ opAssem.append(nextOp);
+ }
+ return '';
+ });
+ };
+
+ const doOp = () => {
+ const o = randomStringOperation(textLeft.length);
+ if (o.insert) {
+ const txt = o.insert;
+ charBank.append(txt);
+ outTextAssem.append(txt);
+ appendMultilineOp('+', txt);
+ } else if (o.skip) {
+ const txt = textLeft.substring(0, o.skip);
+ textLeft = textLeft.substring(o.skip);
+ outTextAssem.append(txt);
+ appendMultilineOp('=', txt);
+ } else if (o.remove) {
+ const txt = textLeft.substring(0, o.remove);
+ textLeft = textLeft.substring(o.remove);
+ appendMultilineOp('-', txt);
+ }
+ };
+
+ while (textLeft.length > 1) doOp();
+ for (let i = 0; i < 5; i++) doOp(); // do some more (only insertions will happen)
+ const outText = `${outTextAssem.toString()}\n`;
+ opAssem.endDocument();
+ const cs = Changeset.pack(oldLen, outText.length, opAssem.toString(), charBank.toString());
+ Changeset.checkRep(cs);
+ return [cs, outText];
+};
+exports.randomTestChangeset = randomTestChangeset;
diff --git a/src/tests/frontend/specs/easysync-assembler.js b/src/tests/frontend/specs/easysync-assembler.js
new file mode 100644
index 00000000000..d9ce04ae2ac
--- /dev/null
+++ b/src/tests/frontend/specs/easysync-assembler.js
@@ -0,0 +1,63 @@
+'use strict';
+
+const Changeset = require('../../../static/js/Changeset');
+
+describe('easysync-assembler', function () {
+ it('opAssembler', 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);
+ });
+
+ it('smartOpAssembler', 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.smartOpAssembler();
+ for (const op of Changeset.deserializeOps(x)) assem.append(op);
+ assem.endDocument();
+ expect(assem.toString()).to.equal(x);
+ });
+
+ describe('append atext to assembler', function () {
+ const testAppendATextToAssembler = (testId, atext, correctOps) => {
+ it(`testAppendATextToAssembler#${testId}`, async function () {
+ const assem = Changeset.smartOpAssembler();
+ for (const op of Changeset.opsFromAText(atext)) assem.append(op);
+ expect(assem.toString()).to.equal(correctOps);
+ });
+ };
+
+ testAppendATextToAssembler(1, {
+ text: '\n',
+ attribs: '|1+1',
+ }, '');
+ testAppendATextToAssembler(2, {
+ text: '\n\n',
+ attribs: '|2+2',
+ }, '|1+1');
+ testAppendATextToAssembler(3, {
+ text: '\n\n',
+ attribs: '*x|2+2',
+ }, '*x|1+1');
+ testAppendATextToAssembler(4, {
+ text: '\n\n',
+ attribs: '*x|1+1|1+1',
+ }, '*x|1+1');
+ testAppendATextToAssembler(5, {
+ text: 'foo\n',
+ attribs: '|1+4',
+ }, '+3');
+ testAppendATextToAssembler(6, {
+ text: '\nfoo\n',
+ attribs: '|2+5',
+ }, '|1+1+3');
+ testAppendATextToAssembler(7, {
+ text: '\nfoo\n',
+ attribs: '*x|2+5',
+ }, '*x|1+1*x+3');
+ testAppendATextToAssembler(8, {
+ text: '\n\n\nfoo\n',
+ attribs: '|2+2*x|2+5',
+ }, '|2+2*x|1+1*x+3');
+ });
+});
diff --git a/src/tests/frontend/specs/easysync-compose.js b/src/tests/frontend/specs/easysync-compose.js
new file mode 100644
index 00000000000..69757763c6c
--- /dev/null
+++ b/src/tests/frontend/specs/easysync-compose.js
@@ -0,0 +1,53 @@
+'use strict';
+
+const Changeset = require('../../../static/js/Changeset');
+const AttributePool = require('../../../static/js/AttributePool');
+const {randomMultiline, randomTestChangeset} = require('../easysync-helper.js');
+
+describe('easysync-compose', function () {
+ describe('compose', function () {
+ const testCompose = (randomSeed) => {
+ it(`testCompose#${randomSeed}`, async function () {
+ const p = new AttributePool();
+
+ const startText = `${randomMultiline(10, 20)}\n`;
+
+ const x1 = randomTestChangeset(startText);
+ const change1 = x1[0];
+ const text1 = x1[1];
+
+ const x2 = randomTestChangeset(text1);
+ const change2 = x2[0];
+ const text2 = x2[1];
+
+ const x3 = randomTestChangeset(text2);
+ const change3 = x3[0];
+ const text3 = x3[1];
+
+ const change12 = Changeset.checkRep(Changeset.compose(change1, change2, p));
+ const change23 = Changeset.checkRep(Changeset.compose(change2, change3, p));
+ const change123 = Changeset.checkRep(Changeset.compose(change12, change3, p));
+ const change123a = Changeset.checkRep(Changeset.compose(change1, change23, p));
+ expect(change123a).to.equal(change123);
+
+ expect(Changeset.applyToText(change12, startText)).to.equal(text2);
+ expect(Changeset.applyToText(change23, text1)).to.equal(text3);
+ expect(Changeset.applyToText(change123, startText)).to.equal(text3);
+ });
+ };
+
+ for (let i = 0; i < 30; i++) testCompose(i);
+ });
+
+ describe('compose attributes', function () {
+ it('simpleComposeAttributesTest', async function () {
+ const p = new AttributePool();
+ p.putAttrib(['bold', '']);
+ p.putAttrib(['bold', 'true']);
+ const cs1 = Changeset.checkRep('Z:2>1*1+1*1=1$x');
+ const cs2 = Changeset.checkRep('Z:3>0*0|1=3$');
+ const cs12 = Changeset.checkRep(Changeset.compose(cs1, cs2, p));
+ expect(cs12).to.equal('Z:2>1+1*0|1=2$x');
+ });
+ });
+});
diff --git a/src/tests/frontend/specs/easysync-follow.js b/src/tests/frontend/specs/easysync-follow.js
new file mode 100644
index 00000000000..9ec5a7e8301
--- /dev/null
+++ b/src/tests/frontend/specs/easysync-follow.js
@@ -0,0 +1,82 @@
+'use strict';
+
+const Changeset = require('../../../static/js/Changeset');
+const AttributePool = require('../../../static/js/AttributePool');
+const {randomMultiline, randomTestChangeset} = require('../easysync-helper.js');
+
+describe('easysync-follow', function () {
+ describe('follow & compose', function () {
+ const testFollow = (randomSeed) => {
+ it(`testFollow#${randomSeed}`, async function () {
+ const p = new AttributePool();
+
+ const startText = `${randomMultiline(10, 20)}\n`;
+
+ const cs1 = randomTestChangeset(startText)[0];
+ const cs2 = randomTestChangeset(startText)[0];
+
+ const afb = Changeset.checkRep(Changeset.follow(cs1, cs2, false, p));
+ const bfa = Changeset.checkRep(Changeset.follow(cs2, cs1, true, p));
+
+ const merge1 = Changeset.checkRep(Changeset.compose(cs1, afb));
+ const merge2 = Changeset.checkRep(Changeset.compose(cs2, bfa));
+
+ expect(merge2).to.equal(merge1);
+ });
+ };
+
+ for (let i = 0; i < 30; i++) testFollow(i);
+ });
+
+ describe('followAttributes & composeAttributes', function () {
+ const p = new AttributePool();
+ p.putAttrib(['x', '']);
+ p.putAttrib(['x', 'abc']);
+ p.putAttrib(['x', 'def']);
+ p.putAttrib(['y', '']);
+ p.putAttrib(['y', 'abc']);
+ p.putAttrib(['y', 'def']);
+ let n = 0;
+
+ const testFollow = (a, b, afb, bfa, merge) => {
+ it(`manual #${++n}`, async function () {
+ expect(Changeset.exportedForTestingOnly.followAttributes(a, b, p)).to.equal(afb);
+ expect(Changeset.exportedForTestingOnly.followAttributes(b, a, p)).to.equal(bfa);
+ expect(Changeset.composeAttributes(a, afb, true, p)).to.equal(merge);
+ expect(Changeset.composeAttributes(b, bfa, true, p)).to.equal(merge);
+ });
+ };
+
+ testFollow('', '', '', '', '');
+ testFollow('*0', '', '', '*0', '*0');
+ testFollow('*0', '*0', '', '', '*0');
+ testFollow('*0', '*1', '', '*0', '*0');
+ testFollow('*1', '*2', '', '*1', '*1');
+ testFollow('*0*1', '', '', '*0*1', '*0*1');
+ testFollow('*0*4', '*2*3', '*3', '*0', '*0*3');
+ testFollow('*0*4', '*2', '', '*0*4', '*0*4');
+ });
+
+ describe('chracterRangeFollow', function () {
+ const testCharacterRangeFollow = (testId, cs, oldRange, insertionsAfter, correctNewRange) => {
+ it(`testCharacterRangeFollow#${testId}`, async function () {
+ cs = Changeset.checkRep(cs);
+ expect(Changeset.characterRangeFollow(cs, oldRange[0], oldRange[1], insertionsAfter))
+ .to.eql(correctNewRange);
+ });
+ };
+
+ testCharacterRangeFollow(1, 'Z:z>9*0=1=4-3+9=1|1-4-4+1*0+a$123456789abcdefghijk',
+ [7, 10], false, [14, 15]);
+ testCharacterRangeFollow(2, 'Z:bc<6|x=b4|2-6$', [400, 407], false, [400, 401]);
+ testCharacterRangeFollow(3, 'Z:4>0-3+3$abc', [0, 3], false, [3, 3]);
+ testCharacterRangeFollow(4, 'Z:4>0-3+3$abc', [0, 3], true, [0, 0]);
+ testCharacterRangeFollow(5, 'Z:5>1+1=1-3+3$abcd', [1, 4], false, [5, 5]);
+ testCharacterRangeFollow(6, 'Z:5>1+1=1-3+3$abcd', [1, 4], true, [2, 2]);
+ testCharacterRangeFollow(7, 'Z:5>1+1=1-3+3$abcd', [0, 6], false, [1, 7]);
+ testCharacterRangeFollow(8, 'Z:5>1+1=1-3+3$abcd', [0, 3], false, [1, 2]);
+ testCharacterRangeFollow(9, 'Z:5>1+1=1-3+3$abcd', [2, 5], false, [5, 6]);
+ testCharacterRangeFollow(10, 'Z:2>1+1$a', [0, 0], false, [1, 1]);
+ testCharacterRangeFollow(11, 'Z:2>1+1$a', [0, 0], true, [0, 0]);
+ });
+});
diff --git a/src/tests/frontend/specs/easysync-inverseRandom.js b/src/tests/frontend/specs/easysync-inverseRandom.js
new file mode 100644
index 00000000000..41ef86d5779
--- /dev/null
+++ b/src/tests/frontend/specs/easysync-inverseRandom.js
@@ -0,0 +1,53 @@
+'use strict';
+
+const Changeset = require('../../../static/js/Changeset');
+const {randomMultiline, randomTestChangeset, poolOrArray} = require('../easysync-helper.js');
+
+describe('easysync-inverseRandom', function () {
+ describe('inverse random', function () {
+ const testInverseRandom = (randomSeed) => {
+ it(`testInverseRandom#${randomSeed}`, async function () {
+ const p = poolOrArray(['apple,', 'apple,true', 'banana,', 'banana,true']);
+
+ const startText = `${randomMultiline(10, 20)}\n`;
+ const alines =
+ Changeset.splitAttributionLines(Changeset.makeAttribution(startText), startText);
+ const lines = startText.slice(0, -1).split('\n').map((s) => `${s}\n`);
+
+ const stylifier = randomTestChangeset(startText, true)[0];
+
+ Changeset.mutateAttributionLines(stylifier, alines, p);
+ Changeset.mutateTextLines(stylifier, lines);
+
+ const changeset = randomTestChangeset(lines.join(''), true)[0];
+ const inverseChangeset = Changeset.inverse(changeset, lines, alines, p);
+
+ const origLines = lines.slice();
+ const origALines = alines.slice();
+
+ Changeset.mutateTextLines(changeset, lines);
+ Changeset.mutateAttributionLines(changeset, alines, p);
+ Changeset.mutateTextLines(inverseChangeset, lines);
+ Changeset.mutateAttributionLines(inverseChangeset, alines, p);
+ expect(lines).to.eql(origLines);
+ expect(alines).to.eql(origALines);
+ });
+ };
+
+ for (let i = 0; i < 30; i++) testInverseRandom(i);
+ });
+
+ describe('inverse', function () {
+ const testInverse = (testId, cs, lines, alines, pool, correctOutput) => {
+ it(`testInverse#${testId}`, async function () {
+ pool = poolOrArray(pool);
+ const str = Changeset.inverse(Changeset.checkRep(cs), lines, alines, pool);
+ expect(str).to.equal(correctOutput);
+ });
+ };
+
+ // take "FFFFTTTTT" and apply "-FT--FFTT", the inverse of which is "--F--TT--"
+ testInverse(1, 'Z:9>0=1*0=1*1=1=2*0=2*1|1=2$', null,
+ ['+4*1+5'], ['bold,', 'bold,true'], 'Z:9>0=2*0=1=2*1=2$');
+ });
+});
diff --git a/src/tests/frontend/specs/easysync-mutations.js b/src/tests/frontend/specs/easysync-mutations.js
new file mode 100644
index 00000000000..7cf43c8b7e5
--- /dev/null
+++ b/src/tests/frontend/specs/easysync-mutations.js
@@ -0,0 +1,304 @@
+'use strict';
+
+const Changeset = require('../../../static/js/Changeset');
+const AttributePool = require('../../../static/js/AttributePool');
+const {poolOrArray} = require('../easysync-helper.js');
+
+describe('easysync-mutations', function () {
+ const applyMutations = (mu, arrayOfArrays) => {
+ arrayOfArrays.forEach((a) => {
+ const result = mu[a[0]](...a.slice(1));
+ if (a[0] === 'remove' && a[3]) {
+ expect(result).to.equal(a[3]);
+ }
+ });
+ };
+
+ const mutationsToChangeset = (oldLen, arrayOfArrays) => {
+ const assem = Changeset.smartOpAssembler();
+ const op = new Changeset.Op();
+ const bank = Changeset.stringAssembler();
+ let oldPos = 0;
+ let newLen = 0;
+ arrayOfArrays.forEach((a) => {
+ if (a[0] === 'skip') {
+ op.opcode = '=';
+ op.chars = a[1];
+ op.lines = (a[2] || 0);
+ assem.append(op);
+ oldPos += op.chars;
+ newLen += op.chars;
+ } else if (a[0] === 'remove') {
+ op.opcode = '-';
+ op.chars = a[1];
+ op.lines = (a[2] || 0);
+ assem.append(op);
+ oldPos += op.chars;
+ } else if (a[0] === 'insert') {
+ op.opcode = '+';
+ bank.append(a[1]);
+ op.chars = a[1].length;
+ op.lines = (a[2] || 0);
+ assem.append(op);
+ newLen += op.chars;
+ }
+ });
+ newLen += oldLen - oldPos;
+ assem.endDocument();
+ return Changeset.pack(oldLen, newLen, assem.toString(), bank.toString());
+ };
+
+ const runMutationTest = (testId, origLines, muts, correct) => {
+ it(`runMutationTest#${testId}`, async function () {
+ let lines = origLines.slice();
+ const mu = new Changeset.exportedForTestingOnly.TextLinesMutator(lines);
+ applyMutations(mu, muts);
+ mu.close();
+ expect(lines).to.eql(correct);
+
+ const inText = origLines.join('');
+ const cs = mutationsToChangeset(inText.length, muts);
+ lines = origLines.slice();
+ Changeset.mutateTextLines(cs, lines);
+ expect(lines).to.eql(correct);
+
+ const correctText = correct.join('');
+ const outText = Changeset.applyToText(cs, inText);
+ expect(outText).to.equal(correctText);
+ });
+ };
+
+ runMutationTest(1, ['apple\n', 'banana\n', 'cabbage\n', 'duffle\n', 'eggplant\n'], [
+ ['remove', 1, 0, 'a'],
+ ['insert', 'tu'],
+ ['remove', 1, 0, 'p'],
+ ['skip', 4, 1],
+ ['skip', 7, 1],
+ ['insert', 'cream\npie\n', 2],
+ ['skip', 2],
+ ['insert', 'bot'],
+ ['insert', '\n', 1],
+ ['insert', 'bu'],
+ ['skip', 3],
+ ['remove', 3, 1, 'ge\n'],
+ ['remove', 6, 0, 'duffle'],
+ ], ['tuple\n', 'banana\n', 'cream\n', 'pie\n', 'cabot\n', 'bubba\n', 'eggplant\n']);
+
+ runMutationTest(2, ['apple\n', 'banana\n', 'cabbage\n', 'duffle\n', 'eggplant\n'], [
+ ['remove', 1, 0, 'a'],
+ ['remove', 1, 0, 'p'],
+ ['insert', 'tu'],
+ ['skip', 11, 2],
+ ['insert', 'cream\npie\n', 2],
+ ['skip', 2],
+ ['insert', 'bot'],
+ ['insert', '\n', 1],
+ ['insert', 'bu'],
+ ['skip', 3],
+ ['remove', 3, 1, 'ge\n'],
+ ['remove', 6, 0, 'duffle'],
+ ], ['tuple\n', 'banana\n', 'cream\n', 'pie\n', 'cabot\n', 'bubba\n', 'eggplant\n']);
+
+ runMutationTest(3, ['apple\n', 'banana\n', 'cabbage\n', 'duffle\n', 'eggplant\n'], [
+ ['remove', 6, 1, 'apple\n'],
+ ['skip', 15, 2],
+ ['skip', 6],
+ ['remove', 1, 1, '\n'],
+ ['remove', 8, 0, 'eggplant'],
+ ['skip', 1, 1],
+ ], ['banana\n', 'cabbage\n', 'duffle\n']);
+
+ runMutationTest(4, ['15\n'], [
+ ['skip', 1],
+ ['insert', '\n2\n3\n4\n', 4],
+ ['skip', 2, 1],
+ ], ['1\n', '2\n', '3\n', '4\n', '5\n']);
+
+ runMutationTest(5, ['1\n', '2\n', '3\n', '4\n', '5\n'], [
+ ['skip', 1],
+ ['remove', 7, 4, '\n2\n3\n4\n'],
+ ['skip', 2, 1],
+ ], ['15\n']);
+
+ runMutationTest(6, ['123\n', 'abc\n', 'def\n', 'ghi\n', 'xyz\n'], [
+ ['insert', '0'],
+ ['skip', 4, 1],
+ ['skip', 4, 1],
+ ['remove', 8, 2, 'def\nghi\n'],
+ ['skip', 4, 1],
+ ], ['0123\n', 'abc\n', 'xyz\n']);
+
+ runMutationTest(7, ['apple\n', 'banana\n', 'cabbage\n', 'duffle\n', 'eggplant\n'], [
+ ['remove', 6, 1, 'apple\n'],
+ ['skip', 15, 2, true],
+ ['skip', 6, 0, true],
+ ['remove', 1, 1, '\n'],
+ ['remove', 8, 0, 'eggplant'],
+ ['skip', 1, 1, true],
+ ], ['banana\n', 'cabbage\n', 'duffle\n']);
+
+ it('mutatorHasMore', async function () {
+ const lines = ['1\n', '2\n', '3\n', '4\n'];
+ let mu;
+
+ mu = new Changeset.exportedForTestingOnly.TextLinesMutator(lines);
+ expect(mu.hasMore()).to.be(true);
+ mu.skip(8, 4);
+ expect(mu.hasMore()).to.be(false);
+ mu.close();
+ expect(mu.hasMore()).to.be(false);
+
+ // still 1,2,3,4
+ mu = new Changeset.exportedForTestingOnly.TextLinesMutator(lines);
+ expect(mu.hasMore()).to.be(true);
+ mu.remove(2, 1);
+ expect(mu.hasMore()).to.be(true);
+ mu.skip(2, 1);
+ expect(mu.hasMore()).to.be(true);
+ mu.skip(2, 1);
+ expect(mu.hasMore()).to.be(true);
+ mu.skip(2, 1);
+ expect(mu.hasMore()).to.be(false);
+ mu.insert('5\n', 1);
+ expect(mu.hasMore()).to.be(false);
+ mu.close();
+ expect(mu.hasMore()).to.be(false);
+
+ // 2,3,4,5 now
+ mu = new Changeset.exportedForTestingOnly.TextLinesMutator(lines);
+ expect(mu.hasMore()).to.be(true);
+ mu.remove(6, 3);
+ expect(mu.hasMore()).to.be(true);
+ mu.remove(2, 1);
+ expect(mu.hasMore()).to.be(false);
+ mu.insert('hello\n', 1);
+ expect(mu.hasMore()).to.be(false);
+ mu.close();
+ expect(mu.hasMore()).to.be(false);
+ });
+
+ describe('mutateTextLines', function () {
+ const testMutateTextLines = (testId, cs, lines, correctLines) => {
+ it(`testMutateTextLines#${testId}`, async function () {
+ const a = lines.slice();
+ Changeset.mutateTextLines(cs, a);
+ expect(a).to.eql(correctLines);
+ });
+ };
+
+ testMutateTextLines(1, 'Z:4<1|1-2-1|1+1+1$\nc', ['a\n', 'b\n'], ['\n', 'c\n']);
+ testMutateTextLines(2, 'Z:4>0|1-2-1|2+3$\nc\n', ['a\n', 'b\n'], ['\n', 'c\n', '\n']);
+ });
+
+ describe('mutate attributions', function () {
+ const testPoolWithChars = (() => {
+ const p = new AttributePool();
+ p.putAttrib(['char', 'newline']);
+ for (let i = 1; i < 36; i++) {
+ p.putAttrib(['char', Changeset.numToString(i)]);
+ }
+ p.putAttrib(['char', '']);
+ return p;
+ })();
+
+ const runMutateAttributionTest = (testId, attribs, cs, alines, outCorrect) => {
+ it(`runMutateAttributionTest#${testId}`, async function () {
+ const p = poolOrArray(attribs);
+ const alines2 = Array.prototype.slice.call(alines);
+ Changeset.mutateAttributionLines(Changeset.checkRep(cs), alines2, p);
+ expect(alines2).to.eql(outCorrect);
+
+ const removeQuestionMarks = (a) => a.replace(/\?/g, '');
+ const inMerged = Changeset.joinAttributionLines(alines.map(removeQuestionMarks));
+ const correctMerged = Changeset.joinAttributionLines(outCorrect.map(removeQuestionMarks));
+ const mergedResult = Changeset.applyToAttribution(cs, inMerged, p);
+ expect(mergedResult).to.equal(correctMerged);
+ });
+ };
+
+ // turn 123\n 456\n 789\n into 123\n 456\n 789\n
+ runMutateAttributionTest(1,
+ ['bold,true'], 'Z:c>0|1=4=1*0=1$', ['|1+4', '|1+4', '|1+4'], ['|1+4', '+1*0+1|1+2', '|1+4']
+ );
+
+ // make a document bold
+ runMutateAttributionTest(2,
+ ['bold,true'], 'Z:c>0*0|3=c$', ['|1+4', '|1+4', '|1+4'], ['*0|1+4', '*0|1+4', '*0|1+4']);
+
+ // clear bold on document
+ runMutateAttributionTest(3,
+ ['bold,', 'bold,true'], 'Z:c>0*0|3=c$',
+ ['*1+1+1*1+1|1+1', '+1*1+1|1+2', '*1+1+1*1+1|1+1'], ['|1+4', '|1+4', '|1+4']);
+
+ // add a character on line 3 of a document with 5 blank lines, and make sure
+ // the optimization that skips purely-kept lines is working; if any attribution string
+ // with a '?' is parsed it will cause an error.
+ runMutateAttributionTest(4,
+ ['foo,bar', 'line,1', 'line,2', 'line,3', 'line,4', 'line,5'],
+ 'Z:5>1|2=2+1$x', ['?*1|1+1', '?*2|1+1', '*3|1+1', '?*4|1+1', '?*5|1+1'],
+ ['?*1|1+1', '?*2|1+1', '+1*3|1+1', '?*4|1+1', '?*5|1+1']);
+
+ // based on runMutationTest#1
+ runMutateAttributionTest(5, testPoolWithChars,
+ 'Z:11>7-2*t+1*u+1|2=b|2+a=2*b+1*o+1*t+1*0|1+1*b+1*u+1=3|1-3-6$tucream\npie\nbot\nbu',
+ [
+ '*a+1*p+2*l+1*e+1*0|1+1',
+ '*b+1*a+1*n+1*a+1*n+1*a+1*0|1+1',
+ '*c+1*a+1*b+2*a+1*g+1*e+1*0|1+1',
+ '*d+1*u+1*f+2*l+1*e+1*0|1+1',
+ '*e+1*g+2*p+1*l+1*a+1*n+1*t+1*0|1+1',
+ ],
+ [
+ '*t+1*u+1*p+1*l+1*e+1*0|1+1',
+ '*b+1*a+1*n+1*a+1*n+1*a+1*0|1+1',
+ '|1+6',
+ '|1+4',
+ '*c+1*a+1*b+1*o+1*t+1*0|1+1',
+ '*b+1*u+1*b+2*a+1*0|1+1',
+ '*e+1*g+2*p+1*l+1*a+1*n+1*t+1*0|1+1',
+ ]);
+
+ // based on runMutationTest#3
+ runMutateAttributionTest(6, testPoolWithChars,
+ 'Z:117=1|4+7$\n2\n3\n4\n',
+ ['*1+1*5|1+2'], ['*1+1|1+1', '|1+2', '|1+2', '|1+2', '*5|1+2']);
+
+ // based on runMutationTest#5
+ runMutateAttributionTest(8, testPoolWithChars, 'Z:a<7=1|4-7$',
+ ['*1|1+2', '*2|1+2', '*3|1+2', '*4|1+2', '*5|1+2'], ['*1+1*5|1+2']);
+
+ // based on runMutationTest#6
+ runMutateAttributionTest(9, testPoolWithChars, 'Z:k<7*0+1*10|2=8|2-8$0',
+ [
+ '*1+1*2+1*3+1|1+1',
+ '*a+1*b+1*c+1|1+1',
+ '*d+1*e+1*f+1|1+1',
+ '*g+1*h+1*i+1|1+1',
+ '?*x+1*y+1*z+1|1+1',
+ ],
+ ['*0+1|1+4', '|1+4', '?*x+1*y+1*z+1|1+1']);
+
+ runMutateAttributionTest(10, testPoolWithChars, 'Z:6>4=1+1=1+1|1=1+1=1*0+1$abcd',
+ ['|1+3', '|1+3'], ['|1+5', '+2*0+1|1+2']);
+
+
+ runMutateAttributionTest(11, testPoolWithChars, 'Z:s>1|1=4=6|1+1$\n',
+ ['*0|1+4', '*0|1+8', '*0+5|1+1', '*0|1+1', '*0|1+5', '*0|1+1', '*0|1+1', '*0|1+1', '|1+1'],
+ [
+ '*0|1+4',
+ '*0+6|1+1',
+ '*0|1+2',
+ '*0+5|1+1',
+ '*0|1+1',
+ '*0|1+5',
+ '*0|1+1',
+ '*0|1+1',
+ '*0|1+1',
+ '|1+1',
+ ]);
+ });
+});
diff --git a/src/tests/frontend/specs/easysync-other.js b/src/tests/frontend/specs/easysync-other.js
new file mode 100644
index 00000000000..26376713a1e
--- /dev/null
+++ b/src/tests/frontend/specs/easysync-other.js
@@ -0,0 +1,141 @@
+'use strict';
+
+const Changeset = require('../../../static/js/Changeset');
+const AttributePool = require('../../../static/js/AttributePool');
+const {randomMultiline, poolOrArray} = require('../easysync-helper.js');
+
+describe('easysync-other', function () {
+ describe('filter attribute numbers', function () {
+ const testFilterAttribNumbers = (testId, cs, filter, correctOutput) => {
+ it(`testFilterAttribNumbers#${testId}`, async function () {
+ const str = Changeset.filterAttribNumbers(cs, filter);
+ expect(str).to.equal(correctOutput);
+ });
+ };
+
+ testFilterAttribNumbers(1, '*0*1+1+2+3*1+4*2+5*0*2*1*b*c+6',
+ (n) => (n % 2) === 0, '*0+1+2+3+4*2+5*0*2*c+6');
+ testFilterAttribNumbers(2, '*0*1+1+2+3*1+4*2+5*0*2*1*b*c+6',
+ (n) => (n % 2) === 1, '*1+1+2+3*1+4+5*1*b+6');
+ });
+
+ describe('make attribs string', function () {
+ const testMakeAttribsString = (testId, pool, opcode, attribs, correctString) => {
+ it(`testMakeAttribsString#${testId}`, async function () {
+ const p = poolOrArray(pool);
+ const str = Changeset.makeAttribsString(opcode, attribs, p);
+ expect(str).to.equal(correctString);
+ });
+ };
+
+ testMakeAttribsString(1, ['bold,'], '+', [
+ ['bold', ''],
+ ], '');
+ testMakeAttribsString(2, ['abc,def', 'bold,'], '=', [
+ ['bold', ''],
+ ], '*1');
+ testMakeAttribsString(3, ['abc,def', 'bold,true'], '+', [
+ ['abc', 'def'],
+ ['bold', 'true'],
+ ], '*0*1');
+ testMakeAttribsString(4, ['abc,def', 'bold,true'], '+', [
+ ['bold', 'true'],
+ ['abc', 'def'],
+ ], '*0*1');
+ });
+
+ describe('other', function () {
+ it('testMoveOpsToNewPool', async function () {
+ const pool1 = new AttributePool();
+ const pool2 = new AttributePool();
+
+ pool1.putAttrib(['baz', 'qux']);
+ pool1.putAttrib(['foo', 'bar']);
+
+ pool2.putAttrib(['foo', 'bar']);
+
+ expect(Changeset.moveOpsToNewPool('Z:1>2*1+1*0+1$ab', pool1, pool2))
+ .to.equal('Z:1>2*0+1*1+1$ab');
+ expect(Changeset.moveOpsToNewPool('*1+1*0+1', pool1, pool2)).to.equal('*0+1*1+1');
+ });
+
+ it('testMakeSplice', async function () {
+ const t = 'a\nb\nc\n';
+ const t2 = Changeset.applyToText(Changeset.makeSplice(t, 5, 0, 'def'), t);
+ expect(t2).to.equal('a\nb\ncdef\n');
+ });
+
+ it('testToSplices', async function () {
+ const cs = Changeset.checkRep('Z:z>9*0=1=4-3+9=1|1-4-4+1*0+a$123456789abcdefghijk');
+ const correctSplices = [
+ [5, 8, '123456789'],
+ [9, 17, 'abcdefghijk'],
+ ];
+ expect(Changeset.exportedForTestingOnly.toSplices(cs)).to.eql(correctSplices);
+ });
+
+ it('opAttributeValue', async function () {
+ const p = new AttributePool();
+ p.putAttrib(['name', 'david']);
+ p.putAttrib(['color', 'green']);
+
+ const stringOp = (str) => Changeset.deserializeOps(str).next().value;
+
+ expect(Changeset.opAttributeValue(stringOp('*0*1+1'), 'name', p)).to.equal('david');
+ expect(Changeset.opAttributeValue(stringOp('*0+1'), 'name', p)).to.equal('david');
+ expect(Changeset.opAttributeValue(stringOp('*1+1'), 'name', p)).to.equal('');
+ expect(Changeset.opAttributeValue(stringOp('+1'), 'name', p)).to.equal('');
+ expect(Changeset.opAttributeValue(stringOp('*0*1+1'), 'color', p)).to.equal('green');
+ expect(Changeset.opAttributeValue(stringOp('*1+1'), 'color', p)).to.equal('green');
+ expect(Changeset.opAttributeValue(stringOp('*0+1'), 'color', p)).to.equal('');
+ expect(Changeset.opAttributeValue(stringOp('+1'), 'color', p)).to.equal('');
+ });
+
+ describe('applyToAttribution', function () {
+ const runApplyToAttributionTest = (testId, attribs, cs, inAttr, outCorrect) => {
+ it(`applyToAttribution#${testId}`, async function () {
+ const p = poolOrArray(attribs);
+ const result = Changeset.applyToAttribution(Changeset.checkRep(cs), inAttr, p);
+ expect(result).to.equal(outCorrect);
+ });
+ };
+
+ // turn cactus\n into actusabcd\n
+ runApplyToAttributionTest(1,
+ ['bold,', 'bold,true'], 'Z:7>3-1*0=1*1=1=3+4$abcd', '+1*1+1|1+5', '+1*1+1|1+8');
+
+ // turn "david\ngreenspan\n" into "david\ngreen\n"
+ runApplyToAttributionTest(2,
+ ['bold,', 'bold,true'], 'Z:g<4*1|1=6*1=5-4$', '|2+g', '*1|1+6*1+5|1+1');
+ });
+
+ describe('split/join attribution lines', function () {
+ const testSplitJoinAttributionLines = (randomSeed) => {
+ const stringToOps = (str) => {
+ const assem = Changeset.mergingOpAssembler();
+ const o = new Changeset.Op('+');
+ o.chars = 1;
+ for (let i = 0; i < str.length; i++) {
+ const c = str.charAt(i);
+ o.lines = (c === '\n' ? 1 : 0);
+ o.attribs = (c === 'a' || c === 'b' ? `*${c}` : '');
+ assem.append(o);
+ }
+ return assem.toString();
+ };
+
+ it(`testSplitJoinAttributionLines#${randomSeed}`, async function () {
+ const doc = `${randomMultiline(10, 20)}\n`;
+
+ const theJoined = stringToOps(doc);
+ const theSplit = doc.match(/[^\n]*\n/g).map(stringToOps);
+
+ expect(Changeset.splitAttributionLines(theJoined, doc)).to.eql(theSplit);
+ expect(Changeset.joinAttributionLines(theSplit)).to.equal(theJoined);
+ });
+ };
+
+ for (let i = 0; i < 10; i++) testSplitJoinAttributionLines(i);
+ });
+ });
+});
diff --git a/src/tests/frontend/specs/easysync-subAttribution.js b/src/tests/frontend/specs/easysync-subAttribution.js
new file mode 100644
index 00000000000..0cb4e8f7c33
--- /dev/null
+++ b/src/tests/frontend/specs/easysync-subAttribution.js
@@ -0,0 +1,55 @@
+'use strict';
+
+const Changeset = require('../../../static/js/Changeset');
+
+describe('easysync-subAttribution', function () {
+ const testSubattribution = (testId, astr, start, end, correctOutput) => {
+ it(`subattribution#${testId}`, async function () {
+ const str = Changeset.subattribution(astr, start, end);
+ expect(str).to.equal(correctOutput);
+ });
+ };
+
+ testSubattribution(1, '+1', 0, 0, '');
+ testSubattribution(2, '+1', 0, 1, '+1');
+ testSubattribution(3, '+1', 0, undefined, '+1');
+ testSubattribution(4, '|1+1', 0, 0, '');
+ testSubattribution(5, '|1+1', 0, 1, '|1+1');
+ testSubattribution(6, '|1+1', 0, undefined, '|1+1');
+ testSubattribution(7, '*0+1', 0, 0, '');
+ testSubattribution(8, '*0+1', 0, 1, '*0+1');
+ testSubattribution(9, '*0+1', 0, undefined, '*0+1');
+ testSubattribution(10, '*0|1+1', 0, 0, '');
+ testSubattribution(11, '*0|1+1', 0, 1, '*0|1+1');
+ testSubattribution(12, '*0|1+1', 0, undefined, '*0|1+1');
+ testSubattribution(13, '*0+2+1*1+3', 0, 1, '*0+1');
+ testSubattribution(14, '*0+2+1*1+3', 0, 2, '*0+2');
+ testSubattribution(15, '*0+2+1*1+3', 0, 3, '*0+2+1');
+ testSubattribution(16, '*0+2+1*1+3', 0, 4, '*0+2+1*1+1');
+ testSubattribution(17, '*0+2+1*1+3', 0, 5, '*0+2+1*1+2');
+ testSubattribution(18, '*0+2+1*1+3', 0, 6, '*0+2+1*1+3');
+ testSubattribution(19, '*0+2+1*1+3', 0, 7, '*0+2+1*1+3');
+ testSubattribution(20, '*0+2+1*1+3', 0, undefined, '*0+2+1*1+3');
+ testSubattribution(21, '*0+2+1*1+3', 1, undefined, '*0+1+1*1+3');
+ testSubattribution(22, '*0+2+1*1+3', 2, undefined, '+1*1+3');
+ testSubattribution(23, '*0+2+1*1+3', 3, undefined, '*1+3');
+ testSubattribution(24, '*0+2+1*1+3', 4, undefined, '*1+2');
+ testSubattribution(25, '*0+2+1*1+3', 5, undefined, '*1+1');
+ testSubattribution(26, '*0+2+1*1+3', 6, undefined, '');
+ testSubattribution(27, '*0+2+1*1|1+3', 0, 1, '*0+1');
+ testSubattribution(28, '*0+2+1*1|1+3', 0, 2, '*0+2');
+ testSubattribution(29, '*0+2+1*1|1+3', 0, 3, '*0+2+1');
+ testSubattribution(30, '*0+2+1*1|1+3', 0, 4, '*0+2+1*1+1');
+ testSubattribution(31, '*0+2+1*1|1+3', 0, 5, '*0+2+1*1+2');
+ testSubattribution(32, '*0+2+1*1|1+3', 0, 6, '*0+2+1*1|1+3');
+ testSubattribution(33, '*0+2+1*1|1+3', 0, 7, '*0+2+1*1|1+3');
+ testSubattribution(34, '*0+2+1*1|1+3', 0, undefined, '*0+2+1*1|1+3');
+ testSubattribution(35, '*0+2+1*1|1+3', 1, undefined, '*0+1+1*1|1+3');
+ testSubattribution(36, '*0+2+1*1|1+3', 2, undefined, '+1*1|1+3');
+ testSubattribution(37, '*0+2+1*1|1+3', 3, undefined, '*1|1+3');
+ testSubattribution(38, '*0+2+1*1|1+3', 4, undefined, '*1|1+2');
+ testSubattribution(39, '*0+2+1*1|1+3', 5, undefined, '*1|1+1');
+ testSubattribution(40, '*0+2+1*1|1+3', 1, 5, '*0+1+1*1+2');
+ testSubattribution(41, '*0+2+1*1|1+3', 2, 6, '+1*1|1+3');
+ testSubattribution(42, '*0+2+1*1+3', 2, 6, '+1*1+3');
+});
diff --git a/src/tests/frontend/specs/easysync.js b/src/tests/frontend/specs/easysync.js
deleted file mode 100644
index 5d066dfd97b..00000000000
--- a/src/tests/frontend/specs/easysync.js
+++ /dev/null
@@ -1,922 +0,0 @@
-'use strict';
-/**
- * I found this tests in the old Etherpad and used it to test if the Changeset library can be run on
- * node.js. It has no use for ep-lite, but I thought I keep it cause it may help someone to
- * understand the Changeset library
- * https://github.com/ether/pad/blob/master/infrastructure/ace/www/easysync2_tests.js
- */
-
-/*
- * Copyright 2009 Google Inc., 2011 Peter 'Pita' Martischka (Primary Technology Ltd)
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS-IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-
-const Changeset = require('../../../static/js/Changeset');
-const AttributePool = require('../../../static/js/AttributePool');
-
-const randInt = (maxValue) => Math.floor(Math.random() * maxValue);
-
-describe('easysync', function () {
- it('throughIterator', 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);
- });
-
- it('throughSmartAssembler', 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.smartOpAssembler();
- for (const op of Changeset.deserializeOps(x)) assem.append(op);
- assem.endDocument();
- expect(assem.toString()).to.equal(x);
- });
-
- const applyMutations = (mu, arrayOfArrays) => {
- arrayOfArrays.forEach((a) => {
- const result = mu[a[0]](...a.slice(1));
- if (a[0] === 'remove' && a[3]) {
- expect(result).to.equal(a[3]);
- }
- });
- };
-
- const mutationsToChangeset = (oldLen, arrayOfArrays) => {
- const assem = Changeset.smartOpAssembler();
- const op = new Changeset.Op();
- const bank = Changeset.stringAssembler();
- let oldPos = 0;
- let newLen = 0;
- arrayOfArrays.forEach((a) => {
- if (a[0] === 'skip') {
- op.opcode = '=';
- op.chars = a[1];
- op.lines = (a[2] || 0);
- assem.append(op);
- oldPos += op.chars;
- newLen += op.chars;
- } else if (a[0] === 'remove') {
- op.opcode = '-';
- op.chars = a[1];
- op.lines = (a[2] || 0);
- assem.append(op);
- oldPos += op.chars;
- } else if (a[0] === 'insert') {
- op.opcode = '+';
- bank.append(a[1]);
- op.chars = a[1].length;
- op.lines = (a[2] || 0);
- assem.append(op);
- newLen += op.chars;
- }
- });
- newLen += oldLen - oldPos;
- assem.endDocument();
- return Changeset.pack(oldLen, newLen, assem.toString(), bank.toString());
- };
-
- const runMutationTest = (testId, origLines, muts, correct) => {
- it(`runMutationTest#${testId}`, async function () {
- let lines = origLines.slice();
- const mu = new Changeset.exportedForTestingOnly.TextLinesMutator(lines);
- applyMutations(mu, muts);
- mu.close();
- expect(lines).to.eql(correct);
-
- const inText = origLines.join('');
- const cs = mutationsToChangeset(inText.length, muts);
- lines = origLines.slice();
- Changeset.mutateTextLines(cs, lines);
- expect(lines).to.eql(correct);
-
- const correctText = correct.join('');
- const outText = Changeset.applyToText(cs, inText);
- expect(outText).to.equal(correctText);
- });
- };
-
- runMutationTest(1, ['apple\n', 'banana\n', 'cabbage\n', 'duffle\n', 'eggplant\n'], [
- ['remove', 1, 0, 'a'],
- ['insert', 'tu'],
- ['remove', 1, 0, 'p'],
- ['skip', 4, 1],
- ['skip', 7, 1],
- ['insert', 'cream\npie\n', 2],
- ['skip', 2],
- ['insert', 'bot'],
- ['insert', '\n', 1],
- ['insert', 'bu'],
- ['skip', 3],
- ['remove', 3, 1, 'ge\n'],
- ['remove', 6, 0, 'duffle'],
- ], ['tuple\n', 'banana\n', 'cream\n', 'pie\n', 'cabot\n', 'bubba\n', 'eggplant\n']);
-
- runMutationTest(2, ['apple\n', 'banana\n', 'cabbage\n', 'duffle\n', 'eggplant\n'], [
- ['remove', 1, 0, 'a'],
- ['remove', 1, 0, 'p'],
- ['insert', 'tu'],
- ['skip', 11, 2],
- ['insert', 'cream\npie\n', 2],
- ['skip', 2],
- ['insert', 'bot'],
- ['insert', '\n', 1],
- ['insert', 'bu'],
- ['skip', 3],
- ['remove', 3, 1, 'ge\n'],
- ['remove', 6, 0, 'duffle'],
- ], ['tuple\n', 'banana\n', 'cream\n', 'pie\n', 'cabot\n', 'bubba\n', 'eggplant\n']);
-
- runMutationTest(3, ['apple\n', 'banana\n', 'cabbage\n', 'duffle\n', 'eggplant\n'], [
- ['remove', 6, 1, 'apple\n'],
- ['skip', 15, 2],
- ['skip', 6],
- ['remove', 1, 1, '\n'],
- ['remove', 8, 0, 'eggplant'],
- ['skip', 1, 1],
- ], ['banana\n', 'cabbage\n', 'duffle\n']);
-
- runMutationTest(4, ['15\n'], [
- ['skip', 1],
- ['insert', '\n2\n3\n4\n', 4],
- ['skip', 2, 1],
- ], ['1\n', '2\n', '3\n', '4\n', '5\n']);
-
- runMutationTest(5, ['1\n', '2\n', '3\n', '4\n', '5\n'], [
- ['skip', 1],
- ['remove', 7, 4, '\n2\n3\n4\n'],
- ['skip', 2, 1],
- ], ['15\n']);
-
- runMutationTest(6, ['123\n', 'abc\n', 'def\n', 'ghi\n', 'xyz\n'], [
- ['insert', '0'],
- ['skip', 4, 1],
- ['skip', 4, 1],
- ['remove', 8, 2, 'def\nghi\n'],
- ['skip', 4, 1],
- ], ['0123\n', 'abc\n', 'xyz\n']);
-
- runMutationTest(7, ['apple\n', 'banana\n', 'cabbage\n', 'duffle\n', 'eggplant\n'], [
- ['remove', 6, 1, 'apple\n'],
- ['skip', 15, 2, true],
- ['skip', 6, 0, true],
- ['remove', 1, 1, '\n'],
- ['remove', 8, 0, 'eggplant'],
- ['skip', 1, 1, true],
- ], ['banana\n', 'cabbage\n', 'duffle\n']);
-
- const poolOrArray = (attribs) => {
- if (attribs.getAttrib) {
- return attribs; // it's already an attrib pool
- } else {
- // assume it's an array of attrib strings to be split and added
- const p = new AttributePool();
- attribs.forEach((kv) => {
- p.putAttrib(kv.split(','));
- });
- return p;
- }
- };
-
- const runApplyToAttributionTest = (testId, attribs, cs, inAttr, outCorrect) => {
- it(`applyToAttribution#${testId}`, async function () {
- const p = poolOrArray(attribs);
- const result = Changeset.applyToAttribution(Changeset.checkRep(cs), inAttr, p);
- expect(result).to.equal(outCorrect);
- });
- };
-
- // turn cactus\n into actusabcd\n
- runApplyToAttributionTest(1,
- ['bold,', 'bold,true'], 'Z:7>3-1*0=1*1=1=3+4$abcd', '+1*1+1|1+5', '+1*1+1|1+8');
-
- // turn "david\ngreenspan\n" into "david\ngreen\n"
- runApplyToAttributionTest(2,
- ['bold,', 'bold,true'], 'Z:g<4*1|1=6*1=5-4$', '|2+g', '*1|1+6*1+5|1+1');
-
- it('mutatorHasMore', async function () {
- const lines = ['1\n', '2\n', '3\n', '4\n'];
- let mu;
-
- mu = new Changeset.exportedForTestingOnly.TextLinesMutator(lines);
- expect(mu.hasMore()).to.be(true);
- mu.skip(8, 4);
- expect(mu.hasMore()).to.be(false);
- mu.close();
- expect(mu.hasMore()).to.be(false);
-
- // still 1,2,3,4
- mu = new Changeset.exportedForTestingOnly.TextLinesMutator(lines);
- expect(mu.hasMore()).to.be(true);
- mu.remove(2, 1);
- expect(mu.hasMore()).to.be(true);
- mu.skip(2, 1);
- expect(mu.hasMore()).to.be(true);
- mu.skip(2, 1);
- expect(mu.hasMore()).to.be(true);
- mu.skip(2, 1);
- expect(mu.hasMore()).to.be(false);
- mu.insert('5\n', 1);
- expect(mu.hasMore()).to.be(false);
- mu.close();
- expect(mu.hasMore()).to.be(false);
-
- // 2,3,4,5 now
- mu = new Changeset.exportedForTestingOnly.TextLinesMutator(lines);
- expect(mu.hasMore()).to.be(true);
- mu.remove(6, 3);
- expect(mu.hasMore()).to.be(true);
- mu.remove(2, 1);
- expect(mu.hasMore()).to.be(false);
- mu.insert('hello\n', 1);
- expect(mu.hasMore()).to.be(false);
- mu.close();
- expect(mu.hasMore()).to.be(false);
- });
-
- const runMutateAttributionTest = (testId, attribs, cs, alines, outCorrect) => {
- it(`runMutateAttributionTest#${testId}`, async function () {
- const p = poolOrArray(attribs);
- const alines2 = Array.prototype.slice.call(alines);
- Changeset.mutateAttributionLines(Changeset.checkRep(cs), alines2, p);
- expect(alines2).to.eql(outCorrect);
-
- const removeQuestionMarks = (a) => a.replace(/\?/g, '');
- const inMerged = Changeset.joinAttributionLines(alines.map(removeQuestionMarks));
- const correctMerged = Changeset.joinAttributionLines(outCorrect.map(removeQuestionMarks));
- const mergedResult = Changeset.applyToAttribution(cs, inMerged, p);
- expect(mergedResult).to.equal(correctMerged);
- });
- };
-
- // turn 123\n 456\n 789\n into 123\n 456\n 789\n
- runMutateAttributionTest(1,
- ['bold,true'], 'Z:c>0|1=4=1*0=1$', ['|1+4', '|1+4', '|1+4'], ['|1+4', '+1*0+1|1+2', '|1+4']);
-
- // make a document bold
- runMutateAttributionTest(2,
- ['bold,true'], 'Z:c>0*0|3=c$', ['|1+4', '|1+4', '|1+4'], ['*0|1+4', '*0|1+4', '*0|1+4']);
-
- // clear bold on document
- runMutateAttributionTest(3,
- ['bold,', 'bold,true'], 'Z:c>0*0|3=c$',
- ['*1+1+1*1+1|1+1', '+1*1+1|1+2', '*1+1+1*1+1|1+1'], ['|1+4', '|1+4', '|1+4']);
-
- // add a character on line 3 of a document with 5 blank lines, and make sure
- // the optimization that skips purely-kept lines is working; if any attribution string
- // with a '?' is parsed it will cause an error.
- runMutateAttributionTest(4,
- ['foo,bar', 'line,1', 'line,2', 'line,3', 'line,4', 'line,5'],
- 'Z:5>1|2=2+1$x', ['?*1|1+1', '?*2|1+1', '*3|1+1', '?*4|1+1', '?*5|1+1'],
- ['?*1|1+1', '?*2|1+1', '+1*3|1+1', '?*4|1+1', '?*5|1+1']);
-
- const testPoolWithChars = (() => {
- const p = new AttributePool();
- p.putAttrib(['char', 'newline']);
- for (let i = 1; i < 36; i++) {
- p.putAttrib(['char', Changeset.numToString(i)]);
- }
- p.putAttrib(['char', '']);
- return p;
- })();
-
- // based on runMutationTest#1
- runMutateAttributionTest(5, testPoolWithChars,
- 'Z:11>7-2*t+1*u+1|2=b|2+a=2*b+1*o+1*t+1*0|1+1*b+1*u+1=3|1-3-6$tucream\npie\nbot\nbu',
- [
- '*a+1*p+2*l+1*e+1*0|1+1',
- '*b+1*a+1*n+1*a+1*n+1*a+1*0|1+1',
- '*c+1*a+1*b+2*a+1*g+1*e+1*0|1+1',
- '*d+1*u+1*f+2*l+1*e+1*0|1+1',
- '*e+1*g+2*p+1*l+1*a+1*n+1*t+1*0|1+1',
- ],
- [
- '*t+1*u+1*p+1*l+1*e+1*0|1+1',
- '*b+1*a+1*n+1*a+1*n+1*a+1*0|1+1',
- '|1+6',
- '|1+4',
- '*c+1*a+1*b+1*o+1*t+1*0|1+1',
- '*b+1*u+1*b+2*a+1*0|1+1',
- '*e+1*g+2*p+1*l+1*a+1*n+1*t+1*0|1+1',
- ]);
-
- // based on runMutationTest#3
- runMutateAttributionTest(6, testPoolWithChars,
- 'Z:117=1|4+7$\n2\n3\n4\n',
- ['*1+1*5|1+2'], ['*1+1|1+1', '|1+2', '|1+2', '|1+2', '*5|1+2']);
-
- // based on runMutationTest#5
- runMutateAttributionTest(8, testPoolWithChars, 'Z:a<7=1|4-7$',
- ['*1|1+2', '*2|1+2', '*3|1+2', '*4|1+2', '*5|1+2'], ['*1+1*5|1+2']);
-
- // based on runMutationTest#6
- runMutateAttributionTest(9, testPoolWithChars, 'Z:k<7*0+1*10|2=8|2-8$0',
- [
- '*1+1*2+1*3+1|1+1',
- '*a+1*b+1*c+1|1+1',
- '*d+1*e+1*f+1|1+1',
- '*g+1*h+1*i+1|1+1',
- '?*x+1*y+1*z+1|1+1',
- ],
- ['*0+1|1+4', '|1+4', '?*x+1*y+1*z+1|1+1']);
-
- runMutateAttributionTest(10, testPoolWithChars, 'Z:6>4=1+1=1+1|1=1+1=1*0+1$abcd',
- ['|1+3', '|1+3'], ['|1+5', '+2*0+1|1+2']);
-
-
- runMutateAttributionTest(11, testPoolWithChars, 'Z:s>1|1=4=6|1+1$\n',
- ['*0|1+4', '*0|1+8', '*0+5|1+1', '*0|1+1', '*0|1+5', '*0|1+1', '*0|1+1', '*0|1+1', '|1+1'],
- [
- '*0|1+4',
- '*0+6|1+1',
- '*0|1+2',
- '*0+5|1+1',
- '*0|1+1',
- '*0|1+5',
- '*0|1+1',
- '*0|1+1',
- '*0|1+1',
- '|1+1',
- ]);
-
- const randomInlineString = (len) => {
- const assem = Changeset.stringAssembler();
- for (let i = 0; i < len; i++) {
- assem.append(String.fromCharCode(randInt(26) + 97));
- }
- return assem.toString();
- };
-
- const randomMultiline = (approxMaxLines, approxMaxCols) => {
- const numParts = randInt(approxMaxLines * 2) + 1;
- const txt = Changeset.stringAssembler();
- txt.append(randInt(2) ? '\n' : '');
- for (let i = 0; i < numParts; i++) {
- if ((i % 2) === 0) {
- if (randInt(10)) {
- txt.append(randomInlineString(randInt(approxMaxCols) + 1));
- } else {
- txt.append('\n');
- }
- } else {
- txt.append('\n');
- }
- }
- return txt.toString();
- };
-
- const randomStringOperation = (numCharsLeft) => {
- let result;
- switch (randInt(9)) {
- case 0:
- {
- // insert char
- result = {
- insert: randomInlineString(1),
- };
- break;
- }
- case 1:
- {
- // delete char
- result = {
- remove: 1,
- };
- break;
- }
- case 2:
- {
- // skip char
- result = {
- skip: 1,
- };
- break;
- }
- case 3:
- {
- // insert small
- result = {
- insert: randomInlineString(randInt(4) + 1),
- };
- break;
- }
- case 4:
- {
- // delete small
- result = {
- remove: randInt(4) + 1,
- };
- break;
- }
- case 5:
- {
- // skip small
- result = {
- skip: randInt(4) + 1,
- };
- break;
- }
- case 6:
- {
- // insert multiline;
- result = {
- insert: randomMultiline(5, 20),
- };
- break;
- }
- case 7:
- {
- // delete multiline
- result = {
- remove: Math.round(numCharsLeft * Math.random() * Math.random()),
- };
- break;
- }
- case 8:
- {
- // skip multiline
- result = {
- skip: Math.round(numCharsLeft * Math.random() * Math.random()),
- };
- break;
- }
- case 9:
- {
- // delete to end
- result = {
- remove: numCharsLeft,
- };
- break;
- }
- case 10:
- {
- // skip to end
- result = {
- skip: numCharsLeft,
- };
- break;
- }
- }
- const maxOrig = numCharsLeft - 1;
- if ('remove' in result) {
- result.remove = Math.min(result.remove, maxOrig);
- } else if ('skip' in result) {
- result.skip = Math.min(result.skip, maxOrig);
- }
- return result;
- };
-
- const randomTwoPropAttribs = (opcode) => {
- // assumes attrib pool like ['apple,','apple,true','banana,','banana,true']
- if (opcode === '-' || randInt(3)) {
- return '';
- } else if (randInt(3)) { // eslint-disable-line no-dupe-else-if
- if (opcode === '+' || randInt(2)) {
- return `*${Changeset.numToString(randInt(2) * 2 + 1)}`;
- } else {
- return `*${Changeset.numToString(randInt(2) * 2)}`;
- }
- } else if (opcode === '+' || randInt(4) === 0) {
- return '*1*3';
- } else {
- return ['*0*2', '*0*3', '*1*2'][randInt(3)];
- }
- };
-
- const randomTestChangeset = (origText, withAttribs) => {
- const charBank = Changeset.stringAssembler();
- let textLeft = origText; // always keep final newline
- const outTextAssem = Changeset.stringAssembler();
- const opAssem = Changeset.smartOpAssembler();
- const oldLen = origText.length;
-
- const nextOp = new Changeset.Op();
-
- const appendMultilineOp = (opcode, txt) => {
- nextOp.opcode = opcode;
- if (withAttribs) {
- nextOp.attribs = randomTwoPropAttribs(opcode);
- }
- txt.replace(/\n|[^\n]+/g, (t) => {
- if (t === '\n') {
- nextOp.chars = 1;
- nextOp.lines = 1;
- opAssem.append(nextOp);
- } else {
- nextOp.chars = t.length;
- nextOp.lines = 0;
- opAssem.append(nextOp);
- }
- return '';
- });
- };
-
- const doOp = () => {
- const o = randomStringOperation(textLeft.length);
- if (o.insert) {
- const txt = o.insert;
- charBank.append(txt);
- outTextAssem.append(txt);
- appendMultilineOp('+', txt);
- } else if (o.skip) {
- const txt = textLeft.substring(0, o.skip);
- textLeft = textLeft.substring(o.skip);
- outTextAssem.append(txt);
- appendMultilineOp('=', txt);
- } else if (o.remove) {
- const txt = textLeft.substring(0, o.remove);
- textLeft = textLeft.substring(o.remove);
- appendMultilineOp('-', txt);
- }
- };
-
- while (textLeft.length > 1) doOp();
- for (let i = 0; i < 5; i++) doOp(); // do some more (only insertions will happen)
- const outText = `${outTextAssem.toString()}\n`;
- opAssem.endDocument();
- const cs = Changeset.pack(oldLen, outText.length, opAssem.toString(), charBank.toString());
- Changeset.checkRep(cs);
- return [cs, outText];
- };
-
- const testCompose = (randomSeed) => {
- it(`testCompose#${randomSeed}`, async function () {
- const p = new AttributePool();
-
- const startText = `${randomMultiline(10, 20)}\n`;
-
- const x1 = randomTestChangeset(startText);
- const change1 = x1[0];
- const text1 = x1[1];
-
- const x2 = randomTestChangeset(text1);
- const change2 = x2[0];
- const text2 = x2[1];
-
- const x3 = randomTestChangeset(text2);
- const change3 = x3[0];
- const text3 = x3[1];
-
- const change12 = Changeset.checkRep(Changeset.compose(change1, change2, p));
- const change23 = Changeset.checkRep(Changeset.compose(change2, change3, p));
- const change123 = Changeset.checkRep(Changeset.compose(change12, change3, p));
- const change123a = Changeset.checkRep(Changeset.compose(change1, change23, p));
- expect(change123a).to.equal(change123);
-
- expect(Changeset.applyToText(change12, startText)).to.equal(text2);
- expect(Changeset.applyToText(change23, text1)).to.equal(text3);
- expect(Changeset.applyToText(change123, startText)).to.equal(text3);
- });
- };
-
- for (let i = 0; i < 30; i++) testCompose(i);
-
- it('simpleComposeAttributesTest', async function () {
- const p = new AttributePool();
- p.putAttrib(['bold', '']);
- p.putAttrib(['bold', 'true']);
- const cs1 = Changeset.checkRep('Z:2>1*1+1*1=1$x');
- const cs2 = Changeset.checkRep('Z:3>0*0|1=3$');
- const cs12 = Changeset.checkRep(Changeset.compose(cs1, cs2, p));
- expect(cs12).to.equal('Z:2>1+1*0|1=2$x');
- });
-
- (() => {
- const p = new AttributePool();
- p.putAttrib(['x', '']);
- p.putAttrib(['x', 'abc']);
- p.putAttrib(['x', 'def']);
- p.putAttrib(['y', '']);
- p.putAttrib(['y', 'abc']);
- p.putAttrib(['y', 'def']);
- let n = 0;
-
- const testFollow = (a, b, afb, bfa, merge) => {
- it(`testFollow manual #${++n}`, async function () {
- expect(Changeset.exportedForTestingOnly.followAttributes(a, b, p)).to.equal(afb);
- expect(Changeset.exportedForTestingOnly.followAttributes(b, a, p)).to.equal(bfa);
- expect(Changeset.composeAttributes(a, afb, true, p)).to.equal(merge);
- expect(Changeset.composeAttributes(b, bfa, true, p)).to.equal(merge);
- });
- };
-
- testFollow('', '', '', '', '');
- testFollow('*0', '', '', '*0', '*0');
- testFollow('*0', '*0', '', '', '*0');
- testFollow('*0', '*1', '', '*0', '*0');
- testFollow('*1', '*2', '', '*1', '*1');
- testFollow('*0*1', '', '', '*0*1', '*0*1');
- testFollow('*0*4', '*2*3', '*3', '*0', '*0*3');
- testFollow('*0*4', '*2', '', '*0*4', '*0*4');
- })();
-
- const testFollow = (randomSeed) => {
- it(`testFollow#${randomSeed}`, async function () {
- const p = new AttributePool();
-
- const startText = `${randomMultiline(10, 20)}\n`;
-
- const cs1 = randomTestChangeset(startText)[0];
- const cs2 = randomTestChangeset(startText)[0];
-
- const afb = Changeset.checkRep(Changeset.follow(cs1, cs2, false, p));
- const bfa = Changeset.checkRep(Changeset.follow(cs2, cs1, true, p));
-
- const merge1 = Changeset.checkRep(Changeset.compose(cs1, afb));
- const merge2 = Changeset.checkRep(Changeset.compose(cs2, bfa));
-
- expect(merge2).to.equal(merge1);
- });
- };
-
- for (let i = 0; i < 30; i++) testFollow(i);
-
- const testSplitJoinAttributionLines = (randomSeed) => {
- const stringToOps = (str) => {
- const assem = Changeset.mergingOpAssembler();
- const o = new Changeset.Op('+');
- o.chars = 1;
- for (let i = 0; i < str.length; i++) {
- const c = str.charAt(i);
- o.lines = (c === '\n' ? 1 : 0);
- o.attribs = (c === 'a' || c === 'b' ? `*${c}` : '');
- assem.append(o);
- }
- return assem.toString();
- };
-
- it(`testSplitJoinAttributionLines#${randomSeed}`, async function () {
- const doc = `${randomMultiline(10, 20)}\n`;
-
- const theJoined = stringToOps(doc);
- const theSplit = doc.match(/[^\n]*\n/g).map(stringToOps);
-
- expect(Changeset.splitAttributionLines(theJoined, doc)).to.eql(theSplit);
- expect(Changeset.joinAttributionLines(theSplit)).to.equal(theJoined);
- });
- };
-
- for (let i = 0; i < 10; i++) testSplitJoinAttributionLines(i);
-
- it('testMoveOpsToNewPool', async function () {
- const pool1 = new AttributePool();
- const pool2 = new AttributePool();
-
- pool1.putAttrib(['baz', 'qux']);
- pool1.putAttrib(['foo', 'bar']);
-
- pool2.putAttrib(['foo', 'bar']);
-
- expect(Changeset.moveOpsToNewPool('Z:1>2*1+1*0+1$ab', pool1, pool2))
- .to.equal('Z:1>2*0+1*1+1$ab');
- expect(Changeset.moveOpsToNewPool('*1+1*0+1', pool1, pool2)).to.equal('*0+1*1+1');
- });
-
- it('testMakeSplice', async function () {
- const t = 'a\nb\nc\n';
- const t2 = Changeset.applyToText(Changeset.makeSplice(t, 5, 0, 'def'), t);
- expect(t2).to.equal('a\nb\ncdef\n');
- });
-
- it('testToSplices', async function () {
- const cs = Changeset.checkRep('Z:z>9*0=1=4-3+9=1|1-4-4+1*0+a$123456789abcdefghijk');
- const correctSplices = [
- [5, 8, '123456789'],
- [9, 17, 'abcdefghijk'],
- ];
- expect(Changeset.exportedForTestingOnly.toSplices(cs)).to.eql(correctSplices);
- });
-
- const testCharacterRangeFollow = (testId, cs, oldRange, insertionsAfter, correctNewRange) => {
- it(`testCharacterRangeFollow#${testId}`, async function () {
- cs = Changeset.checkRep(cs);
- expect(Changeset.characterRangeFollow(cs, oldRange[0], oldRange[1], insertionsAfter))
- .to.eql(correctNewRange);
- });
- };
-
- testCharacterRangeFollow(1, 'Z:z>9*0=1=4-3+9=1|1-4-4+1*0+a$123456789abcdefghijk',
- [7, 10], false, [14, 15]);
- testCharacterRangeFollow(2, 'Z:bc<6|x=b4|2-6$', [400, 407], false, [400, 401]);
- testCharacterRangeFollow(3, 'Z:4>0-3+3$abc', [0, 3], false, [3, 3]);
- testCharacterRangeFollow(4, 'Z:4>0-3+3$abc', [0, 3], true, [0, 0]);
- testCharacterRangeFollow(5, 'Z:5>1+1=1-3+3$abcd', [1, 4], false, [5, 5]);
- testCharacterRangeFollow(6, 'Z:5>1+1=1-3+3$abcd', [1, 4], true, [2, 2]);
- testCharacterRangeFollow(7, 'Z:5>1+1=1-3+3$abcd', [0, 6], false, [1, 7]);
- testCharacterRangeFollow(8, 'Z:5>1+1=1-3+3$abcd', [0, 3], false, [1, 2]);
- testCharacterRangeFollow(9, 'Z:5>1+1=1-3+3$abcd', [2, 5], false, [5, 6]);
- testCharacterRangeFollow(10, 'Z:2>1+1$a', [0, 0], false, [1, 1]);
- testCharacterRangeFollow(11, 'Z:2>1+1$a', [0, 0], true, [0, 0]);
-
- it('testOpAttributeValue', async function () {
- const p = new AttributePool();
- p.putAttrib(['name', 'david']);
- p.putAttrib(['color', 'green']);
-
- const stringOp = (str) => Changeset.deserializeOps(str).next().value;
-
- expect(Changeset.opAttributeValue(stringOp('*0*1+1'), 'name', p)).to.equal('david');
- expect(Changeset.opAttributeValue(stringOp('*0+1'), 'name', p)).to.equal('david');
- expect(Changeset.opAttributeValue(stringOp('*1+1'), 'name', p)).to.equal('');
- expect(Changeset.opAttributeValue(stringOp('+1'), 'name', p)).to.equal('');
- expect(Changeset.opAttributeValue(stringOp('*0*1+1'), 'color', p)).to.equal('green');
- expect(Changeset.opAttributeValue(stringOp('*1+1'), 'color', p)).to.equal('green');
- expect(Changeset.opAttributeValue(stringOp('*0+1'), 'color', p)).to.equal('');
- expect(Changeset.opAttributeValue(stringOp('+1'), 'color', p)).to.equal('');
- });
-
- const testAppendATextToAssembler = (testId, atext, correctOps) => {
- it(`testAppendATextToAssembler#${testId}`, async function () {
- const assem = Changeset.smartOpAssembler();
- for (const op of Changeset.opsFromAText(atext)) assem.append(op);
- expect(assem.toString()).to.equal(correctOps);
- });
- };
-
- testAppendATextToAssembler(1, {
- text: '\n',
- attribs: '|1+1',
- }, '');
- testAppendATextToAssembler(2, {
- text: '\n\n',
- attribs: '|2+2',
- }, '|1+1');
- testAppendATextToAssembler(3, {
- text: '\n\n',
- attribs: '*x|2+2',
- }, '*x|1+1');
- testAppendATextToAssembler(4, {
- text: '\n\n',
- attribs: '*x|1+1|1+1',
- }, '*x|1+1');
- testAppendATextToAssembler(5, {
- text: 'foo\n',
- attribs: '|1+4',
- }, '+3');
- testAppendATextToAssembler(6, {
- text: '\nfoo\n',
- attribs: '|2+5',
- }, '|1+1+3');
- testAppendATextToAssembler(7, {
- text: '\nfoo\n',
- attribs: '*x|2+5',
- }, '*x|1+1*x+3');
- testAppendATextToAssembler(8, {
- text: '\n\n\nfoo\n',
- attribs: '|2+2*x|2+5',
- }, '|2+2*x|1+1*x+3');
-
- const testMakeAttribsString = (testId, pool, opcode, attribs, correctString) => {
- it(`testMakeAttribsString#${testId}`, async function () {
- const p = poolOrArray(pool);
- const str = Changeset.makeAttribsString(opcode, attribs, p);
- expect(str).to.equal(correctString);
- });
- };
-
- testMakeAttribsString(1, ['bold,'], '+', [
- ['bold', ''],
- ], '');
- testMakeAttribsString(2, ['abc,def', 'bold,'], '=', [
- ['bold', ''],
- ], '*1');
- testMakeAttribsString(3, ['abc,def', 'bold,true'], '+', [
- ['abc', 'def'],
- ['bold', 'true'],
- ], '*0*1');
- testMakeAttribsString(4, ['abc,def', 'bold,true'], '+', [
- ['bold', 'true'],
- ['abc', 'def'],
- ], '*0*1');
-
- const testSubattribution = (testId, astr, start, end, correctOutput) => {
- it(`testSubattribution#${testId}`, async function () {
- const str = Changeset.subattribution(astr, start, end);
- expect(str).to.equal(correctOutput);
- });
- };
-
- testSubattribution(1, '+1', 0, 0, '');
- testSubattribution(2, '+1', 0, 1, '+1');
- testSubattribution(3, '+1', 0, undefined, '+1');
- testSubattribution(4, '|1+1', 0, 0, '');
- testSubattribution(5, '|1+1', 0, 1, '|1+1');
- testSubattribution(6, '|1+1', 0, undefined, '|1+1');
- testSubattribution(7, '*0+1', 0, 0, '');
- testSubattribution(8, '*0+1', 0, 1, '*0+1');
- testSubattribution(9, '*0+1', 0, undefined, '*0+1');
- testSubattribution(10, '*0|1+1', 0, 0, '');
- testSubattribution(11, '*0|1+1', 0, 1, '*0|1+1');
- testSubattribution(12, '*0|1+1', 0, undefined, '*0|1+1');
- testSubattribution(13, '*0+2+1*1+3', 0, 1, '*0+1');
- testSubattribution(14, '*0+2+1*1+3', 0, 2, '*0+2');
- testSubattribution(15, '*0+2+1*1+3', 0, 3, '*0+2+1');
- testSubattribution(16, '*0+2+1*1+3', 0, 4, '*0+2+1*1+1');
- testSubattribution(17, '*0+2+1*1+3', 0, 5, '*0+2+1*1+2');
- testSubattribution(18, '*0+2+1*1+3', 0, 6, '*0+2+1*1+3');
- testSubattribution(19, '*0+2+1*1+3', 0, 7, '*0+2+1*1+3');
- testSubattribution(20, '*0+2+1*1+3', 0, undefined, '*0+2+1*1+3');
- testSubattribution(21, '*0+2+1*1+3', 1, undefined, '*0+1+1*1+3');
- testSubattribution(22, '*0+2+1*1+3', 2, undefined, '+1*1+3');
- testSubattribution(23, '*0+2+1*1+3', 3, undefined, '*1+3');
- testSubattribution(24, '*0+2+1*1+3', 4, undefined, '*1+2');
- testSubattribution(25, '*0+2+1*1+3', 5, undefined, '*1+1');
- testSubattribution(26, '*0+2+1*1+3', 6, undefined, '');
- testSubattribution(27, '*0+2+1*1|1+3', 0, 1, '*0+1');
- testSubattribution(28, '*0+2+1*1|1+3', 0, 2, '*0+2');
- testSubattribution(29, '*0+2+1*1|1+3', 0, 3, '*0+2+1');
- testSubattribution(30, '*0+2+1*1|1+3', 0, 4, '*0+2+1*1+1');
- testSubattribution(31, '*0+2+1*1|1+3', 0, 5, '*0+2+1*1+2');
- testSubattribution(32, '*0+2+1*1|1+3', 0, 6, '*0+2+1*1|1+3');
- testSubattribution(33, '*0+2+1*1|1+3', 0, 7, '*0+2+1*1|1+3');
- testSubattribution(34, '*0+2+1*1|1+3', 0, undefined, '*0+2+1*1|1+3');
- testSubattribution(35, '*0+2+1*1|1+3', 1, undefined, '*0+1+1*1|1+3');
- testSubattribution(36, '*0+2+1*1|1+3', 2, undefined, '+1*1|1+3');
- testSubattribution(37, '*0+2+1*1|1+3', 3, undefined, '*1|1+3');
- testSubattribution(38, '*0+2+1*1|1+3', 4, undefined, '*1|1+2');
- testSubattribution(39, '*0+2+1*1|1+3', 5, undefined, '*1|1+1');
- testSubattribution(40, '*0+2+1*1|1+3', 1, 5, '*0+1+1*1+2');
- testSubattribution(41, '*0+2+1*1|1+3', 2, 6, '+1*1|1+3');
- testSubattribution(42, '*0+2+1*1+3', 2, 6, '+1*1+3');
-
- const testFilterAttribNumbers = (testId, cs, filter, correctOutput) => {
- it(`testFilterAttribNumbers#${testId}`, async function () {
- const str = Changeset.filterAttribNumbers(cs, filter);
- expect(str).to.equal(correctOutput);
- });
- };
-
- testFilterAttribNumbers(1, '*0*1+1+2+3*1+4*2+5*0*2*1*b*c+6',
- (n) => (n % 2) === 0, '*0+1+2+3+4*2+5*0*2*c+6');
- testFilterAttribNumbers(2, '*0*1+1+2+3*1+4*2+5*0*2*1*b*c+6',
- (n) => (n % 2) === 1, '*1+1+2+3*1+4+5*1*b+6');
-
- const testInverse = (testId, cs, lines, alines, pool, correctOutput) => {
- it(`testInverse#${testId}`, async function () {
- pool = poolOrArray(pool);
- const str = Changeset.inverse(Changeset.checkRep(cs), lines, alines, pool);
- expect(str).to.equal(correctOutput);
- });
- };
-
- // take "FFFFTTTTT" and apply "-FT--FFTT", the inverse of which is "--F--TT--"
- testInverse(1, 'Z:9>0=1*0=1*1=1=2*0=2*1|1=2$', null,
- ['+4*1+5'], ['bold,', 'bold,true'], 'Z:9>0=2*0=1=2*1=2$');
-
- const testMutateTextLines = (testId, cs, lines, correctLines) => {
- it(`testMutateTextLines#${testId}`, async function () {
- const a = lines.slice();
- Changeset.mutateTextLines(cs, a);
- expect(a).to.eql(correctLines);
- });
- };
-
- testMutateTextLines(1, 'Z:4<1|1-2-1|1+1+1$\nc', ['a\n', 'b\n'], ['\n', 'c\n']);
- testMutateTextLines(2, 'Z:4>0|1-2-1|2+3$\nc\n', ['a\n', 'b\n'], ['\n', 'c\n', '\n']);
-
- const testInverseRandom = (randomSeed) => {
- it(`testInverseRandom#${randomSeed}`, async function () {
- const p = poolOrArray(['apple,', 'apple,true', 'banana,', 'banana,true']);
-
- const startText = `${randomMultiline(10, 20)}\n`;
- const alines =
- Changeset.splitAttributionLines(Changeset.makeAttribution(startText), startText);
- const lines = startText.slice(0, -1).split('\n').map((s) => `${s}\n`);
-
- const stylifier = randomTestChangeset(startText, true)[0];
-
- Changeset.mutateAttributionLines(stylifier, alines, p);
- Changeset.mutateTextLines(stylifier, lines);
-
- const changeset = randomTestChangeset(lines.join(''), true)[0];
- const inverseChangeset = Changeset.inverse(changeset, lines, alines, p);
-
- const origLines = lines.slice();
- const origALines = alines.slice();
-
- Changeset.mutateTextLines(changeset, lines);
- Changeset.mutateAttributionLines(changeset, alines, p);
- Changeset.mutateTextLines(inverseChangeset, lines);
- Changeset.mutateAttributionLines(inverseChangeset, alines, p);
- expect(lines).to.eql(origLines);
- expect(alines).to.eql(origALines);
- });
- };
-
- for (let i = 0; i < 30; i++) testInverseRandom(i);
-});