2012-10-08 00:34:29 +02:00
|
|
|
describe("the test helper", function(){
|
|
|
|
describe("the newPad method", function(){
|
|
|
|
xit("doesn't leak memory if you creates iframes over and over again", function(done){
|
2012-10-08 13:37:24 +02:00
|
|
|
this.timeout(100000);
|
2012-10-08 00:34:29 +02:00
|
|
|
|
|
|
|
var times = 10;
|
|
|
|
|
|
|
|
var loadPad = function(){
|
2013-12-05 08:41:29 +01:00
|
|
|
helper.newPad(function(){
|
2012-10-08 00:34:29 +02:00
|
|
|
times--;
|
|
|
|
if(times > 0){
|
|
|
|
loadPad();
|
|
|
|
} else {
|
|
|
|
done();
|
|
|
|
}
|
2013-12-05 08:41:29 +01:00
|
|
|
})
|
2012-10-08 00:34:29 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
loadPad();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("gives me 3 jquery instances of chrome, outer and inner", function(done){
|
2020-03-22 21:30:00 +01:00
|
|
|
this.timeout(10000);
|
2012-10-08 00:34:29 +02:00
|
|
|
|
2013-12-05 08:41:29 +01:00
|
|
|
helper.newPad(function(){
|
|
|
|
//check if the jquery selectors have the desired elements
|
2012-10-08 00:34:29 +02:00
|
|
|
expect(helper.padChrome$("#editbar").length).to.be(1);
|
|
|
|
expect(helper.padOuter$("#outerdocbody").length).to.be(1);
|
|
|
|
expect(helper.padInner$("#innerdocbody").length).to.be(1);
|
|
|
|
|
|
|
|
//check if the document object was set correctly
|
|
|
|
expect(helper.padChrome$.window.document).to.be(helper.padChrome$.document);
|
|
|
|
expect(helper.padOuter$.window.document).to.be(helper.padOuter$.document);
|
|
|
|
expect(helper.padInner$.window.document).to.be(helper.padInner$.document);
|
|
|
|
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
});
|
2020-03-24 18:36:08 +01:00
|
|
|
// Make sure the cookies are cleared, and make sure that the cookie
|
|
|
|
// clearing has taken effect at this point in the code. It has been
|
|
|
|
// observed that the former can happen without the latter if there
|
|
|
|
// isn't a timeout (within `newPad`) after clearing the cookies.
|
|
|
|
// However this doesn't seem to always be easily replicated, so this
|
|
|
|
// timeout may or may end up in the code. None the less, we test here
|
|
|
|
// to catch it if the bug comes up again.
|
|
|
|
it("clears cookies", function(done) {
|
|
|
|
this.timeout(60000);
|
|
|
|
|
|
|
|
// set cookies far into the future to make sure they're not expired yet
|
|
|
|
window.document.cookie = 'token=foo;expires=Thu, 01 Jan 3030 00:00:00 GMT; path=/';
|
|
|
|
window.document.cookie = 'language=bar;expires=Thu, 01 Jan 3030 00:00:00 GMT; path=/';
|
|
|
|
|
2020-05-28 15:25:07 +02:00
|
|
|
expect(window.document.cookie).to.contain('token=foo');
|
|
|
|
expect(window.document.cookie).to.contain('language=bar');
|
2020-03-24 18:36:08 +01:00
|
|
|
|
|
|
|
helper.newPad(function(){
|
|
|
|
// helper function seems to have cleared cookies
|
|
|
|
// NOTE: this doesn't yet mean it's proven to have taken effect by this point in execution
|
|
|
|
var firstCookie = window.document.cookie
|
2020-05-28 15:25:07 +02:00
|
|
|
expect(firstCookie).to.not.contain('token=foo');
|
|
|
|
expect(firstCookie).to.not.contain('language=bar');
|
2020-03-24 18:36:08 +01:00
|
|
|
|
|
|
|
var chrome$ = helper.padChrome$;
|
|
|
|
|
|
|
|
// click on the settings button to make settings visible
|
|
|
|
var $userButton = chrome$(".buttonicon-showusers");
|
|
|
|
$userButton.click();
|
|
|
|
|
|
|
|
var $usernameInput = chrome$("#myusernameedit");
|
|
|
|
$usernameInput.click();
|
|
|
|
|
|
|
|
$usernameInput.val('John McLear');
|
|
|
|
$usernameInput.blur();
|
|
|
|
|
|
|
|
// Before refreshing, make sure the name is there
|
2020-05-28 15:25:07 +02:00
|
|
|
expect($usernameInput.val()).to.be('John McLear');
|
|
|
|
|
|
|
|
// Now that we have a chrome, we can set a pad cookie, so we can confirm it gets wiped as well
|
|
|
|
chrome$.document.cookie = 'prefsHtml=baz;expires=Thu, 01 Jan 3030 00:00:00 GMT';
|
|
|
|
expect(chrome$.document.cookie).to.contain('prefsHtml=baz');
|
|
|
|
|
|
|
|
// Cookies are weird. Because it's attached to chrome$ (as helper.setPadCookies does), AND we
|
|
|
|
// didn't put path=/, we shouldn't expect it to be visible on window.document.cookie. Let's just
|
|
|
|
// be sure.
|
|
|
|
expect(window.document.cookie).to.not.contain('prefsHtml=baz');
|
2020-03-24 18:36:08 +01:00
|
|
|
|
|
|
|
setTimeout(function(){ //give it a second to save the username on the server side
|
|
|
|
helper.newPad(function(){ // get a new pad, let it clear the cookies
|
|
|
|
var chrome$ = helper.padChrome$;
|
|
|
|
|
|
|
|
// helper function seems to have cleared cookies
|
|
|
|
// NOTE: this doesn't yet mean cookies were cleared effectively.
|
|
|
|
// We still need to test below that we're in a new session
|
2020-05-28 15:25:07 +02:00
|
|
|
expect(window.document.cookie).to.not.contain('token=foo');
|
|
|
|
expect(window.document.cookie).to.not.contain('language=bar');
|
|
|
|
expect(chrome$.document.cookie).to.contain('prefsHtml=baz');
|
|
|
|
expect(window.document.cookie).to.not.contain('prefsHtml=baz');
|
|
|
|
|
|
|
|
expect(window.document.cookie).to.not.be(firstCookie);
|
2020-03-24 18:36:08 +01:00
|
|
|
|
|
|
|
// click on the settings button to make settings visible
|
|
|
|
var $userButton = chrome$(".buttonicon-showusers");
|
|
|
|
$userButton.click();
|
|
|
|
|
|
|
|
// confirm that the session was actually cleared
|
|
|
|
var $usernameInput = chrome$("#myusernameedit");
|
2020-05-28 16:27:48 +02:00
|
|
|
expect($usernameInput.val()).to.be('');
|
2020-03-24 18:36:08 +01:00
|
|
|
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
}, 1000);
|
2020-05-28 15:25:07 +02:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it("sets pad prefs cookie", function(done) {
|
|
|
|
this.timeout(60000);
|
|
|
|
|
|
|
|
helper.newPad({
|
|
|
|
padPrefs: {foo:"bar"},
|
|
|
|
cb: function(){
|
|
|
|
var chrome$ = helper.padChrome$;
|
|
|
|
expect(chrome$.document.cookie).to.contain('prefsHttp=%7B%22');
|
|
|
|
expect(chrome$.document.cookie).to.contain('foo%22%3A%22bar');
|
|
|
|
done();
|
|
|
|
}
|
|
|
|
});
|
2020-03-24 18:36:08 +01:00
|
|
|
});
|
2012-10-08 00:34:29 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
describe("the waitFor method", function(){
|
2013-12-05 08:41:29 +01:00
|
|
|
it("takes a timeout and waits long enough", function(done){
|
|
|
|
this.timeout(2000);
|
2019-02-26 23:25:15 +01:00
|
|
|
var startTime = Date.now();
|
2012-10-08 00:34:29 +02:00
|
|
|
|
|
|
|
helper.waitFor(function(){
|
2013-12-05 08:41:29 +01:00
|
|
|
return false;
|
2012-10-08 00:34:29 +02:00
|
|
|
}, 1500).fail(function(){
|
2019-02-26 23:25:15 +01:00
|
|
|
var duration = Date.now() - startTime;
|
2012-10-08 00:34:29 +02:00
|
|
|
expect(duration).to.be.greaterThan(1400);
|
|
|
|
done();
|
|
|
|
});
|
2013-12-05 08:41:29 +01:00
|
|
|
});
|
2012-10-08 00:34:29 +02:00
|
|
|
|
2013-12-05 08:41:29 +01:00
|
|
|
it("takes an interval and checks on every interval", function(done){
|
2012-10-08 00:34:29 +02:00
|
|
|
this.timeout(4000);
|
|
|
|
var checks = 0;
|
2016-06-21 11:48:10 +02:00
|
|
|
|
2012-10-08 00:34:29 +02:00
|
|
|
helper.waitFor(function(){
|
|
|
|
checks++;
|
2013-12-05 08:41:29 +01:00
|
|
|
return false;
|
2012-10-08 00:34:29 +02:00
|
|
|
}, 2000, 100).fail(function(){
|
2012-11-03 23:20:27 +01:00
|
|
|
expect(checks).to.be.greaterThan(10);
|
|
|
|
expect(checks).to.be.lessThan(30);
|
2012-10-08 00:34:29 +02:00
|
|
|
done();
|
|
|
|
});
|
2013-12-05 08:41:29 +01:00
|
|
|
});
|
2012-10-08 00:34:29 +02:00
|
|
|
|
2020-10-14 03:57:19 +02:00
|
|
|
it('rejects if the predicate throws', async function() {
|
|
|
|
let err;
|
|
|
|
await helper.waitFor(() => { throw new Error('test exception'); })
|
|
|
|
.fail(() => {}) // Suppress the redundant uncatchable exception.
|
|
|
|
.catch((e) => { err = e; });
|
|
|
|
expect(err).to.be.an(Error);
|
|
|
|
expect(err.message).to.be('test exception');
|
|
|
|
});
|
|
|
|
|
2013-12-05 08:41:29 +01:00
|
|
|
describe("returns a deferred object", function(){
|
2012-10-08 00:34:29 +02:00
|
|
|
it("it calls done after success", function(done){
|
|
|
|
helper.waitFor(function(){
|
2013-12-05 08:41:29 +01:00
|
|
|
return true;
|
|
|
|
}).done(function(){
|
2012-10-08 00:34:29 +02:00
|
|
|
done();
|
2013-12-05 08:41:29 +01:00
|
|
|
});
|
2012-10-08 00:34:29 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
it("calls fail after failure", function(done){
|
2013-12-05 08:41:29 +01:00
|
|
|
helper.waitFor(function(){
|
|
|
|
return false;
|
|
|
|
},0).fail(function(){
|
2012-10-08 00:34:29 +02:00
|
|
|
done();
|
2013-12-05 08:41:29 +01:00
|
|
|
});
|
2012-10-08 00:34:29 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
xit("throws if you don't listen for fails", function(done){
|
2013-12-05 08:41:29 +01:00
|
|
|
var onerror = window.onerror;
|
|
|
|
window.onerror = function(){
|
2012-10-08 00:34:29 +02:00
|
|
|
window.onerror = onerror;
|
2013-12-05 08:41:29 +01:00
|
|
|
done();
|
|
|
|
}
|
2012-10-08 00:34:29 +02:00
|
|
|
|
2013-12-05 08:41:29 +01:00
|
|
|
helper.waitFor(function(){
|
|
|
|
return false;
|
|
|
|
},100);
|
2012-10-08 00:34:29 +02:00
|
|
|
});
|
2013-12-05 08:41:29 +01:00
|
|
|
});
|
2020-10-14 04:02:46 +02:00
|
|
|
|
|
|
|
describe('checks first then sleeps', function() {
|
|
|
|
it('resolves quickly if the predicate is immediately true', async function() {
|
|
|
|
const before = Date.now();
|
|
|
|
await helper.waitFor(() => true, 1000, 900);
|
|
|
|
expect(Date.now() - before).to.be.lessThan(800);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('polls exactly once if timeout < interval', async function() {
|
|
|
|
let calls = 0;
|
|
|
|
await helper.waitFor(() => { calls++; }, 1, 1000)
|
|
|
|
.fail(() => {}) // Suppress the redundant uncatchable exception.
|
|
|
|
.catch(() => {}); // Don't throw an exception -- we know it rejects.
|
|
|
|
expect(calls).to.be(1);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('resolves if condition is immediately true even if timeout is 0', async function() {
|
|
|
|
await helper.waitFor(() => true, 0);
|
|
|
|
});
|
|
|
|
});
|
2012-10-08 00:34:29 +02:00
|
|
|
});
|
2016-06-21 11:48:10 +02:00
|
|
|
|
2020-10-09 20:50:47 +02:00
|
|
|
describe('the waitForPromise method', function() {
|
2020-10-14 02:44:07 +02:00
|
|
|
it('returns a Promise', async function() {
|
|
|
|
expect(helper.waitForPromise(() => true)).to.be.a(Promise);
|
|
|
|
});
|
|
|
|
|
2020-10-09 20:50:47 +02:00
|
|
|
it('takes a timeout and waits long enough', async function() {
|
|
|
|
this.timeout(2000);
|
2020-10-14 02:44:07 +02:00
|
|
|
const startTime = Date.now();
|
|
|
|
let rejected;
|
|
|
|
await helper.waitForPromise(() => false, 1500)
|
|
|
|
.catch(() => { rejected = true; });
|
|
|
|
expect(rejected).to.be(true);
|
|
|
|
const duration = Date.now() - startTime;
|
|
|
|
expect(duration).to.be.greaterThan(1490);
|
2020-10-09 20:50:47 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
it('takes an interval and checks on every interval', async function() {
|
|
|
|
this.timeout(4000);
|
2020-10-14 02:44:07 +02:00
|
|
|
let checks = 0;
|
|
|
|
let rejected;
|
|
|
|
await helper.waitForPromise(() => { checks++; return false; }, 2000, 100)
|
|
|
|
.catch(() => { rejected = true; });
|
|
|
|
expect(rejected).to.be(true);
|
|
|
|
expect(checks).to.be.greaterThan(15);
|
|
|
|
expect(checks).to.be.lessThan(21);
|
2020-10-09 20:50:47 +02:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2016-06-21 11:48:10 +02:00
|
|
|
describe("the selectLines method", function(){
|
|
|
|
// function to support tests, use a single way to represent whitespaces
|
|
|
|
var cleanText = function(text){
|
|
|
|
return text
|
2017-02-20 10:02:31 +01:00
|
|
|
// IE replaces line breaks with a whitespace, so we need to unify its behavior
|
|
|
|
// for other browsers, to have all tests running for all browsers
|
2017-02-20 10:14:27 +01:00
|
|
|
.replace(/\n/gi, "")
|
2017-02-20 10:02:31 +01:00
|
|
|
.replace(/\s/gi, " ");
|
2016-06-21 11:48:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
before(function(done){
|
|
|
|
helper.newPad(function() {
|
|
|
|
// create some lines to be used on the tests
|
|
|
|
var $firstLine = helper.padInner$("div").first();
|
2017-02-20 10:02:31 +01:00
|
|
|
$firstLine.sendkeys("{selectall}some{enter}short{enter}lines{enter}to test{enter}{enter}");
|
2016-06-21 11:48:10 +02:00
|
|
|
|
|
|
|
// wait for lines to be split
|
|
|
|
helper.waitFor(function(){
|
2017-02-20 10:03:56 +01:00
|
|
|
var $fourthLine = helper.padInner$("div").eq(3);
|
2016-06-21 11:48:10 +02:00
|
|
|
return $fourthLine.text() === "to test";
|
|
|
|
}).done(done);
|
|
|
|
});
|
|
|
|
|
|
|
|
this.timeout(60000);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("changes editor selection to be between startOffset of $startLine and endOffset of $endLine", function(done){
|
|
|
|
var inner$ = helper.padInner$;
|
|
|
|
|
|
|
|
var startOffset = 2;
|
|
|
|
var endOffset = 4;
|
|
|
|
|
|
|
|
var $lines = inner$("div");
|
2017-02-20 10:03:56 +01:00
|
|
|
var $startLine = $lines.eq(1);
|
|
|
|
var $endLine = $lines.eq(3);
|
2016-06-21 11:48:10 +02:00
|
|
|
|
|
|
|
helper.selectLines($startLine, $endLine, startOffset, endOffset);
|
|
|
|
|
|
|
|
var selection = inner$.document.getSelection();
|
2020-03-22 21:32:39 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
* replace() is required here because Firefox keeps the line breaks.
|
|
|
|
*
|
|
|
|
* I'm not sure this is ideal behavior of getSelection() where the text
|
|
|
|
* is not consistent between browsers but that's the situation so that's
|
|
|
|
* how I'm covering it in this test.
|
|
|
|
*/
|
|
|
|
expect(cleanText(selection.toString().replace(/(\r\n|\n|\r)/gm,""))).to.be("ort lines to t");
|
2016-06-21 11:48:10 +02:00
|
|
|
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("ends selection at beginning of $endLine when it is an empty line", function(done){
|
|
|
|
var inner$ = helper.padInner$;
|
|
|
|
|
|
|
|
var startOffset = 2;
|
|
|
|
var endOffset = 1;
|
|
|
|
|
|
|
|
var $lines = inner$("div");
|
2017-02-20 10:03:56 +01:00
|
|
|
var $startLine = $lines.eq(1);
|
|
|
|
var $endLine = $lines.eq(4);
|
2016-06-21 11:48:10 +02:00
|
|
|
|
|
|
|
helper.selectLines($startLine, $endLine, startOffset, endOffset);
|
|
|
|
|
|
|
|
var selection = inner$.document.getSelection();
|
2020-03-22 21:32:39 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
* replace() is required here because Firefox keeps the line breaks.
|
|
|
|
*
|
|
|
|
* I'm not sure this is ideal behavior of getSelection() where the text
|
|
|
|
* is not consistent between browsers but that's the situation so that's
|
|
|
|
* how I'm covering it in this test.
|
|
|
|
*/
|
|
|
|
expect(cleanText(selection.toString().replace(/(\r\n|\n|\r)/gm,""))).to.be("ort lines to test");
|
2016-06-21 11:48:10 +02:00
|
|
|
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
|
2016-06-21 16:07:57 +02:00
|
|
|
it("ends selection at beginning of $endLine when its offset is zero", function(done){
|
|
|
|
var inner$ = helper.padInner$;
|
|
|
|
|
|
|
|
var startOffset = 2;
|
|
|
|
var endOffset = 0;
|
|
|
|
|
|
|
|
var $lines = inner$("div");
|
2017-02-20 10:03:56 +01:00
|
|
|
var $startLine = $lines.eq(1);
|
|
|
|
var $endLine = $lines.eq(3);
|
2016-06-21 16:07:57 +02:00
|
|
|
|
|
|
|
helper.selectLines($startLine, $endLine, startOffset, endOffset);
|
|
|
|
|
|
|
|
var selection = inner$.document.getSelection();
|
2020-03-22 21:32:39 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
* replace() is required here because Firefox keeps the line breaks.
|
|
|
|
*
|
|
|
|
* I'm not sure this is ideal behavior of getSelection() where the text
|
|
|
|
* is not consistent between browsers but that's the situation so that's
|
|
|
|
* how I'm covering it in this test.
|
|
|
|
*/
|
|
|
|
expect(cleanText(selection.toString().replace(/(\r\n|\n|\r)/gm,""))).to.be("ort lines ");
|
2016-06-21 16:07:57 +02:00
|
|
|
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
|
2016-06-21 11:48:10 +02:00
|
|
|
it("selects full line when offset is longer than line content", function(done){
|
|
|
|
var inner$ = helper.padInner$;
|
|
|
|
|
|
|
|
var startOffset = 2;
|
|
|
|
var endOffset = 50;
|
|
|
|
|
|
|
|
var $lines = inner$("div");
|
2017-02-20 10:03:56 +01:00
|
|
|
var $startLine = $lines.eq(1);
|
|
|
|
var $endLine = $lines.eq(3);
|
2016-06-21 11:48:10 +02:00
|
|
|
|
|
|
|
helper.selectLines($startLine, $endLine, startOffset, endOffset);
|
|
|
|
|
|
|
|
var selection = inner$.document.getSelection();
|
2020-03-22 21:32:39 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
* replace() is required here because Firefox keeps the line breaks.
|
|
|
|
*
|
|
|
|
* I'm not sure this is ideal behavior of getSelection() where the text
|
|
|
|
* is not consistent between browsers but that's the situation so that's
|
|
|
|
* how I'm covering it in this test.
|
|
|
|
*/
|
|
|
|
expect(cleanText(selection.toString().replace(/(\r\n|\n|\r)/gm,""))).to.be("ort lines to test");
|
2016-06-21 11:48:10 +02:00
|
|
|
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("selects all text between beginning of $startLine and end of $endLine when no offset is provided", function(done){
|
|
|
|
var inner$ = helper.padInner$;
|
|
|
|
|
|
|
|
var $lines = inner$("div");
|
2017-02-20 10:03:56 +01:00
|
|
|
var $startLine = $lines.eq(1);
|
|
|
|
var $endLine = $lines.eq(3);
|
2016-06-21 11:48:10 +02:00
|
|
|
|
|
|
|
helper.selectLines($startLine, $endLine);
|
|
|
|
|
|
|
|
var selection = inner$.document.getSelection();
|
2020-03-22 21:32:39 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
* replace() is required here because Firefox keeps the line breaks.
|
|
|
|
*
|
|
|
|
* I'm not sure this is ideal behavior of getSelection() where the text
|
|
|
|
* is not consistent between browsers but that's the situation so that's
|
|
|
|
* how I'm covering it in this test.
|
|
|
|
*/
|
|
|
|
expect(cleanText(selection.toString().replace(/(\r\n|\n|\r)/gm,""))).to.be("short lines to test");
|
2016-06-21 11:48:10 +02:00
|
|
|
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
});
|
2013-12-05 08:41:29 +01:00
|
|
|
});
|