mirror of
https://github.com/ether/etherpad-lite.git
synced 2025-02-08 11:12:01 +01:00
244 lines
6.5 KiB
JavaScript
244 lines
6.5 KiB
JavaScript
var Changeset = require("ep_etherpad-lite/static/js/Changeset");
|
|
var AttributePool = require("ep_etherpad-lite/static/js/AttributePool");
|
|
var helper = require("./helper.js")
|
|
var assertEqualStrings = helper.assertEqualStrings;
|
|
|
|
describe("Changeset.compose",function(){
|
|
it("composes all the changesets - BUT THIS only calls applyToText?!?",function(done){
|
|
for (var i = 0; i < 30; i++) testCompose(i);
|
|
done();
|
|
})
|
|
it("simpleComposeAttributesTest",function(done){
|
|
var p = new AttributePool();
|
|
p.putAttrib(['bold', '']);
|
|
p.putAttrib(['bold', 'true']);
|
|
var cs1 = Changeset.checkRep("Z:2>1*1+1*1=1$x");
|
|
var cs2 = Changeset.checkRep("Z:3>0*0|1=3$");
|
|
var cs12 = Changeset.checkRep(Changeset.compose(cs1, cs2, p));
|
|
assertEqualStrings("Z:2>1+1*0|1=2$x", cs12);
|
|
done();
|
|
})
|
|
|
|
})
|
|
function testCompose(randomSeed) {
|
|
var rand = new random();
|
|
|
|
var p = new AttributePool();
|
|
|
|
var startText = randomMultiline(10, 20, rand) + '\n';
|
|
|
|
var x1 = randomTestChangeset(startText, rand);
|
|
var change1 = x1[0];
|
|
var text1 = x1[1];
|
|
|
|
var x2 = randomTestChangeset(text1, rand);
|
|
var change2 = x2[0];
|
|
var text2 = x2[1];
|
|
|
|
var x3 = randomTestChangeset(text2, rand);
|
|
var change3 = x3[0];
|
|
var text3 = x3[1];
|
|
|
|
//print(literal(Changeset.toBaseTen(startText)));
|
|
//print(literal(Changeset.toBaseTen(change1)));
|
|
//print(literal(Changeset.toBaseTen(change2)));
|
|
var change12 = Changeset.checkRep(Changeset.compose(change1, change2, p));
|
|
var change23 = Changeset.checkRep(Changeset.compose(change2, change3, p));
|
|
var change123 = Changeset.checkRep(Changeset.compose(change12, change3, p));
|
|
var change123a = Changeset.checkRep(Changeset.compose(change1, change23, p));
|
|
assertEqualStrings(change123, change123a);
|
|
|
|
assertEqualStrings(text2, Changeset.applyToText(change12, startText));
|
|
assertEqualStrings(text3, Changeset.applyToText(change23, text1));
|
|
assertEqualStrings(text3, Changeset.applyToText(change123, startText));
|
|
}
|
|
|
|
function random() {
|
|
this.nextInt = function (maxValue) {
|
|
return Math.floor(Math.random() * maxValue);
|
|
}
|
|
|
|
// maxValue is not used
|
|
this.nextDouble = function (maxValue) {
|
|
return Math.random();
|
|
}
|
|
}
|
|
function randomMultiline(approxMaxLines, approxMaxCols, rand) {
|
|
var numParts = rand.nextInt(approxMaxLines * 2) + 1;
|
|
var txt = Changeset.stringAssembler();
|
|
txt.append(rand.nextInt(2) ? '\n' : '');
|
|
for (var i = 0; i < numParts; i++) {
|
|
if ((i % 2) == 0) {
|
|
if (rand.nextInt(10)) {
|
|
txt.append(randomInlineString(rand.nextInt(approxMaxCols) + 1, rand));
|
|
} else {
|
|
txt.append('\n');
|
|
}
|
|
} else {
|
|
txt.append('\n');
|
|
}
|
|
}
|
|
return txt.toString();
|
|
}
|
|
function randomInlineString(len, rand) {
|
|
var assem = Changeset.stringAssembler();
|
|
for (var i = 0; i < len; i++) {
|
|
assem.append(String.fromCharCode(rand.nextInt(26) + 97));
|
|
}
|
|
return assem.toString();
|
|
}
|
|
function randomTestChangeset(origText, rand, withAttribs) {
|
|
var charBank = Changeset.stringAssembler();
|
|
var textLeft = origText; // always keep final newline
|
|
var outTextAssem = Changeset.stringAssembler();
|
|
var opAssem = Changeset.smartOpAssembler();
|
|
var oldLen = origText.length;
|
|
|
|
var nextOp = Changeset.newOp();
|
|
|
|
function appendMultilineOp(opcode, txt) {
|
|
nextOp.opcode = opcode;
|
|
if (withAttribs) {
|
|
nextOp.attribs = randomTwoPropAttribs(opcode, rand);
|
|
}
|
|
txt.replace(/\n|[^\n]+/g, function (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 '';
|
|
});
|
|
}
|
|
|
|
function doOp() {
|
|
var o = randomStringOperation(textLeft.length, rand);
|
|
if (o.insert) {
|
|
var txt = o.insert;
|
|
charBank.append(txt);
|
|
outTextAssem.append(txt);
|
|
appendMultilineOp('+', txt);
|
|
} else if (o.skip) {
|
|
txt = textLeft.substring(0, o.skip);
|
|
textLeft = textLeft.substring(o.skip);
|
|
outTextAssem.append(txt);
|
|
appendMultilineOp('=', txt);
|
|
} else if (o.remove) {
|
|
txt = textLeft.substring(0, o.remove);
|
|
textLeft = textLeft.substring(o.remove);
|
|
appendMultilineOp('-', txt);
|
|
}
|
|
}
|
|
|
|
while (textLeft.length > 1) doOp();
|
|
for (var i = 0; i < 5; i++) doOp(); // do some more (only insertions will happen)
|
|
var outText = outTextAssem.toString() + '\n';
|
|
opAssem.endDocument();
|
|
var cs = Changeset.pack(oldLen, outText.length, opAssem.toString(), charBank.toString());
|
|
Changeset.checkRep(cs);
|
|
return [cs, outText];
|
|
}
|
|
function randomStringOperation(numCharsLeft, rand) {
|
|
var result;
|
|
switch (rand.nextInt(9)) {
|
|
case 0:
|
|
{
|
|
// insert char
|
|
result = {
|
|
insert: randomInlineString(1, rand)
|
|
};
|
|
break;
|
|
}
|
|
case 1:
|
|
{
|
|
// delete char
|
|
result = {
|
|
remove: 1
|
|
};
|
|
break;
|
|
}
|
|
case 2:
|
|
{
|
|
// skip char
|
|
result = {
|
|
skip: 1
|
|
};
|
|
break;
|
|
}
|
|
case 3:
|
|
{
|
|
// insert small
|
|
result = {
|
|
insert: randomInlineString(rand.nextInt(4) + 1, rand)
|
|
};
|
|
break;
|
|
}
|
|
case 4:
|
|
{
|
|
// delete small
|
|
result = {
|
|
remove: rand.nextInt(4) + 1
|
|
};
|
|
break;
|
|
}
|
|
case 5:
|
|
{
|
|
// skip small
|
|
result = {
|
|
skip: rand.nextInt(4) + 1
|
|
};
|
|
break;
|
|
}
|
|
case 6:
|
|
{
|
|
// insert multiline;
|
|
result = {
|
|
insert: randomMultiline(5, 20, rand)
|
|
};
|
|
break;
|
|
}
|
|
case 7:
|
|
{
|
|
// delete multiline
|
|
result = {
|
|
remove: Math.round(numCharsLeft * rand.nextDouble() * rand.nextDouble())
|
|
};
|
|
break;
|
|
}
|
|
case 8:
|
|
{
|
|
// skip multiline
|
|
result = {
|
|
skip: Math.round(numCharsLeft * rand.nextDouble() * rand.nextDouble())
|
|
};
|
|
break;
|
|
}
|
|
case 9:
|
|
{
|
|
// delete to end
|
|
result = {
|
|
remove: numCharsLeft
|
|
};
|
|
break;
|
|
}
|
|
case 10:
|
|
{
|
|
// skip to end
|
|
result = {
|
|
skip: numCharsLeft
|
|
};
|
|
break;
|
|
}
|
|
}
|
|
var 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;
|
|
}
|