2013-10-28 20:04:30 +04:00
|
|
|
/*globals Ghost, casper, __utils__ */
|
2013-07-31 11:33:28 +04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Casper Tests
|
|
|
|
*
|
|
|
|
* Functional browser tests for checking that the Ghost Admin UI is working as expected
|
|
|
|
* The setup of these tests is a little hacky for now, which is why they are not wired in to grunt
|
|
|
|
* Requires that you are running Ghost locally and have already registered a single user
|
|
|
|
*
|
|
|
|
* Usage (from test/functional):
|
|
|
|
*
|
2013-08-05 13:31:04 +04:00
|
|
|
* casperjs test admin/ --includes=base.js [--host=localhost --port=2368 --noPort=false --email=ghost@tryghost.org --password=Sl1m3r]
|
2013-07-31 11:33:28 +04:00
|
|
|
*
|
|
|
|
* --host - your local host address e.g. localhost or local.tryghost.org
|
|
|
|
* --port - port number of your local Ghost
|
|
|
|
* --email - the email address your admin user is registered with
|
|
|
|
* --password - the password your admin user is registered with
|
2013-08-05 13:31:04 +04:00
|
|
|
* --noPort - don't include a port number
|
2013-07-31 11:33:28 +04:00
|
|
|
*
|
|
|
|
* Requirements:
|
|
|
|
* you must have phantomjs 1.9.1 and casperjs 1.1.0-DEV installed in order for these tests to work
|
|
|
|
*/
|
|
|
|
|
2014-01-01 18:31:29 +04:00
|
|
|
var DEBUG = false, // TOGGLE THIS TO GET MORE SCREENSHOTS
|
|
|
|
host = casper.cli.options.host || 'localhost',
|
2013-08-05 13:31:04 +04:00
|
|
|
noPort = casper.cli.options.noPort || false,
|
2013-07-31 11:33:28 +04:00
|
|
|
port = casper.cli.options.port || '2368',
|
2013-11-24 18:29:36 +04:00
|
|
|
email = casper.cli.options.email || 'jbloggs@example.com',
|
2013-08-27 01:28:41 +04:00
|
|
|
password = casper.cli.options.password || 'Sl1m3rson',
|
2013-08-05 13:31:04 +04:00
|
|
|
url = "http://" + host + (noPort ? '/' : ":" + port + "/"),
|
2013-09-12 12:59:58 +04:00
|
|
|
newUser = {
|
|
|
|
name: "Test User",
|
|
|
|
email: email,
|
|
|
|
password: password
|
|
|
|
},
|
2013-07-31 11:33:28 +04:00
|
|
|
user = {
|
|
|
|
email: email,
|
|
|
|
password: password
|
|
|
|
},
|
2013-08-22 23:48:36 +04:00
|
|
|
falseUser = {
|
|
|
|
email: email,
|
|
|
|
password: 'letmethrough'
|
|
|
|
},
|
2013-08-05 13:31:04 +04:00
|
|
|
testPost = {
|
|
|
|
title: "Bacon ipsum dolor sit amet",
|
2013-09-12 02:04:49 +04:00
|
|
|
html: "I am a test post.\n#I have some small content"
|
2013-08-05 13:31:04 +04:00
|
|
|
};
|
2013-07-31 11:33:28 +04:00
|
|
|
|
2013-08-05 13:31:04 +04:00
|
|
|
casper.writeContentToCodeMirror = function (content) {
|
|
|
|
var lines = content.split("\n");
|
2013-07-31 11:33:28 +04:00
|
|
|
|
2014-03-18 02:01:29 +04:00
|
|
|
casper.waitForSelector('.CodeMirror-wrap textarea', function onSuccess() {
|
|
|
|
casper.each(lines, function (self, line) {
|
|
|
|
self.sendKeys('.CodeMirror-wrap textarea', line, {keepFocus: true});
|
|
|
|
self.sendKeys('.CodeMirror-wrap textarea', casper.page.event.key.Enter, {keepFocus: true});
|
|
|
|
});
|
2013-08-05 13:31:04 +04:00
|
|
|
|
2014-03-18 02:01:29 +04:00
|
|
|
return this;
|
|
|
|
}, function onTimeout() {
|
|
|
|
casper.test.fail('CodeMirror was not found.');
|
|
|
|
}, 2000);
|
2013-08-05 13:31:04 +04:00
|
|
|
};
|
|
|
|
|
2013-09-12 12:59:58 +04:00
|
|
|
casper.waitForOpaque = function (classname, then, timeout) {
|
|
|
|
casper.waitFor(function checkOpaque() {
|
2013-12-15 22:15:52 +04:00
|
|
|
var value = this.evaluate(function (element) {
|
2013-09-12 12:59:58 +04:00
|
|
|
var target = document.querySelector(element);
|
2013-12-15 22:15:52 +04:00
|
|
|
if (target === null) {
|
|
|
|
return null;
|
|
|
|
}
|
2013-09-12 12:59:58 +04:00
|
|
|
return window.getComputedStyle(target).getPropertyValue('opacity') === "1";
|
|
|
|
}, classname);
|
2013-12-15 22:15:52 +04:00
|
|
|
if (value !== true && value !== false) {
|
|
|
|
casper.test.fail('Unable to find element: ' + classname);
|
|
|
|
}
|
|
|
|
return value;
|
2013-09-12 12:59:58 +04:00
|
|
|
}, then, timeout);
|
|
|
|
};
|
2013-08-30 13:50:42 +04:00
|
|
|
|
2014-05-08 11:44:48 +04:00
|
|
|
casper.failOnTimeout = function (test, message) {
|
|
|
|
return function onTimeout() {
|
|
|
|
test.fail(message);
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2014-06-18 00:53:56 +04:00
|
|
|
casper.echoConcise = function (message, style) {
|
|
|
|
if (!casper.cli.options.concise) {
|
|
|
|
casper.echo(message, style);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-30 13:50:42 +04:00
|
|
|
// ## Debugging
|
|
|
|
// output all errors to the console
|
|
|
|
casper.on('remote.message', function (msg) {
|
2014-06-18 00:53:56 +04:00
|
|
|
casper.echoConcise('GOT CONSOLE LOG: ' + msg);
|
2013-08-30 13:50:42 +04:00
|
|
|
});
|
|
|
|
|
2013-10-28 20:04:30 +04:00
|
|
|
casper.on('error', function (msg) {
|
2014-06-18 00:53:56 +04:00
|
|
|
casper.echoConcise('GOT ERROR, ' + msg);
|
2013-08-30 13:50:42 +04:00
|
|
|
});
|
|
|
|
|
2013-10-28 20:04:30 +04:00
|
|
|
casper.on("page.error", function (msg) {
|
2014-06-18 00:53:56 +04:00
|
|
|
casper.echoConcise("GOT PAGE ERROR: " + msg, "ERROR");
|
2013-08-30 13:50:42 +04:00
|
|
|
});
|
|
|
|
|
2014-01-01 18:31:29 +04:00
|
|
|
casper.captureScreenshot = function (filename, debugOnly) {
|
|
|
|
debugOnly = debugOnly !== false;
|
|
|
|
// If we are in debug mode, OR debugOnly is false
|
|
|
|
if (DEBUG || debugOnly === false) {
|
|
|
|
filename = filename || "casper_test_fail.png";
|
2014-06-04 16:21:56 +04:00
|
|
|
casper.then(function () {
|
|
|
|
casper.capture(new Date().getTime() + '_' + filename);
|
|
|
|
});
|
2014-01-01 18:31:29 +04:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2013-08-30 13:50:42 +04:00
|
|
|
// on failure, grab a screenshot
|
2014-05-08 11:44:48 +04:00
|
|
|
casper.test.on("fail", function captureFailure(test) {
|
2014-01-01 18:31:29 +04:00
|
|
|
casper.captureScreenshot(casper.test.filename || "casper_test_fail.png", false);
|
2013-10-08 05:39:33 +04:00
|
|
|
});
|
|
|
|
|
2013-10-28 20:04:30 +04:00
|
|
|
var CasperTest = (function () {
|
2013-10-08 05:39:33 +04:00
|
|
|
|
|
|
|
var _beforeDoneHandler,
|
|
|
|
_noop = function noop() { },
|
|
|
|
_isUserRegistered = false;
|
|
|
|
|
|
|
|
// Always log out at end of test.
|
|
|
|
casper.test.tearDown(function (done) {
|
|
|
|
casper.then(_beforeDoneHandler);
|
|
|
|
|
2013-11-26 13:38:54 +04:00
|
|
|
casper.thenOpen(url + 'ghost\/signout/');
|
2013-10-08 05:39:33 +04:00
|
|
|
|
|
|
|
casper.waitForResource(/ghost\/sign/);
|
2013-10-28 20:04:30 +04:00
|
|
|
|
2014-01-01 18:31:29 +04:00
|
|
|
casper.captureScreenshot('teardown.png');
|
|
|
|
|
2013-10-08 05:39:33 +04:00
|
|
|
casper.run(done);
|
|
|
|
});
|
|
|
|
|
|
|
|
// Wrapper around `casper.test.begin`
|
|
|
|
function begin(testName, expect, suite, doNotAutoLogin) {
|
|
|
|
_beforeDoneHandler = _noop;
|
|
|
|
|
|
|
|
var runTest = function (test) {
|
|
|
|
test.filename = testName.toLowerCase().replace(/ /g, '-').concat('.png');
|
|
|
|
|
2013-10-15 07:39:52 +04:00
|
|
|
casper.start('about:blank').viewport(1280, 1024);
|
2013-10-08 05:39:33 +04:00
|
|
|
|
|
|
|
if (!doNotAutoLogin) {
|
|
|
|
// Only call register once for the lifetime of Mindless
|
|
|
|
if (!_isUserRegistered) {
|
|
|
|
CasperTest.Routines.logout.run(test);
|
|
|
|
CasperTest.Routines.register.run(test);
|
|
|
|
|
|
|
|
_isUserRegistered = true;
|
|
|
|
}
|
2013-10-28 20:04:30 +04:00
|
|
|
|
2013-10-08 05:39:33 +04:00
|
|
|
/* Ensure we're logged out at the start of every test or we may get
|
|
|
|
unexpected failures. */
|
|
|
|
CasperTest.Routines.logout.run(test);
|
|
|
|
CasperTest.Routines.login.run(test);
|
|
|
|
}
|
|
|
|
|
|
|
|
suite.call(casper, test);
|
|
|
|
|
|
|
|
casper.run(function () {
|
|
|
|
test.done();
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
if (typeof expect === 'function') {
|
|
|
|
doNotAutoLogin = suite;
|
|
|
|
suite = expect;
|
|
|
|
|
|
|
|
casper.test.begin(testName, runTest);
|
2013-10-28 20:04:30 +04:00
|
|
|
} else {
|
2013-10-08 05:39:33 +04:00
|
|
|
casper.test.begin(testName, expect, runTest);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Sets a handler to be invoked right before `test.done` is invoked
|
|
|
|
function beforeDone(fn) {
|
|
|
|
if (fn) {
|
|
|
|
_beforeDoneHandler = fn;
|
2013-10-28 20:04:30 +04:00
|
|
|
} else {
|
2013-10-08 05:39:33 +04:00
|
|
|
_beforeDoneHandler = _noop;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
begin: begin,
|
|
|
|
beforeDone: beforeDone
|
|
|
|
};
|
|
|
|
|
|
|
|
}());
|
|
|
|
|
|
|
|
CasperTest.Routines = (function () {
|
|
|
|
|
|
|
|
function register(test) {
|
|
|
|
casper.thenOpen(url + 'ghost/signup/').viewport(1280, 1024);
|
|
|
|
|
|
|
|
casper.waitForOpaque('.signup-box', function then() {
|
|
|
|
this.fill('#signup', newUser, true);
|
|
|
|
});
|
|
|
|
|
|
|
|
casper.waitForSelectorTextChange('.notification-error', function onSuccess() {
|
2013-10-15 07:39:52 +04:00
|
|
|
var errorText = casper.evaluate(function () {
|
|
|
|
return document.querySelector('.notification-error').innerText;
|
|
|
|
});
|
2014-06-18 00:53:56 +04:00
|
|
|
casper.echoConcise('It appears as though a user is already registered. Error text: ' + errorText);
|
2013-10-08 05:39:33 +04:00
|
|
|
}, function onTimeout() {
|
2014-06-18 00:53:56 +04:00
|
|
|
casper.echoConcise('It appears as though a user was not already registered.');
|
2013-10-08 05:39:33 +04:00
|
|
|
}, 2000);
|
|
|
|
}
|
|
|
|
|
|
|
|
function login(test) {
|
|
|
|
casper.thenOpen(url + 'ghost/signin/');
|
|
|
|
|
2014-01-01 18:31:29 +04:00
|
|
|
casper.waitForResource(/ghost\/signin/);
|
|
|
|
|
2013-10-08 05:39:33 +04:00
|
|
|
casper.waitForOpaque('.login-box', function then() {
|
2014-01-01 18:31:29 +04:00
|
|
|
casper.captureScreenshot("got_sign_in.png");
|
2013-10-08 05:39:33 +04:00
|
|
|
this.fill("#login", user, true);
|
2014-01-01 18:31:29 +04:00
|
|
|
casper.captureScreenshot("filled_sign_in.png");
|
2013-10-08 05:39:33 +04:00
|
|
|
});
|
|
|
|
|
2014-01-01 18:31:29 +04:00
|
|
|
casper.waitForResource(/ghost\/$/).then(function () {
|
|
|
|
casper.captureScreenshot('have_logged_in.png');
|
|
|
|
});
|
2013-10-08 05:39:33 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
function logout(test) {
|
2013-11-26 13:38:54 +04:00
|
|
|
casper.thenOpen(url + 'ghost\/signout/');
|
2013-10-08 05:39:33 +04:00
|
|
|
|
2014-01-01 18:31:29 +04:00
|
|
|
casper.captureScreenshot("logging_out.png");
|
2013-10-08 05:39:33 +04:00
|
|
|
|
2014-01-01 18:31:29 +04:00
|
|
|
// Wait for signin or signup
|
|
|
|
casper.waitForResource(/ghost\/sign/);
|
2013-10-08 05:39:33 +04:00
|
|
|
}
|
|
|
|
|
2014-01-01 15:17:35 +04:00
|
|
|
function togglePermalinks(state) {
|
|
|
|
casper.thenOpen(url + "ghost/settings/general");
|
|
|
|
|
2014-01-01 18:31:29 +04:00
|
|
|
casper.waitForResource(/ghost\/settings\/general/);
|
|
|
|
|
2014-01-01 15:17:35 +04:00
|
|
|
casper.waitForSelector('#general');
|
|
|
|
casper.waitForOpaque('#general', function then() {
|
|
|
|
var currentState = this.evaluate(function () {
|
|
|
|
return document.querySelector('#permalinks') && document.querySelector('#permalinks').checked ? 'on' : 'off';
|
2013-12-30 11:03:29 +04:00
|
|
|
});
|
2014-01-01 15:17:35 +04:00
|
|
|
if (currentState !== state) {
|
|
|
|
casper.thenClick('#permalinks');
|
|
|
|
casper.thenClick('.button-save');
|
2014-01-01 18:31:29 +04:00
|
|
|
|
|
|
|
casper.captureScreenshot("saving.png");
|
|
|
|
|
|
|
|
casper.waitForSelector('.notification-success', function () {
|
|
|
|
casper.captureScreenshot("saved.png");
|
2014-01-01 15:17:35 +04:00
|
|
|
});
|
|
|
|
}
|
2013-12-30 11:03:29 +04:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2013-10-08 05:39:33 +04:00
|
|
|
function _createRunner(fn) {
|
|
|
|
fn.run = function run(test) {
|
|
|
|
var routine = this;
|
|
|
|
|
|
|
|
casper.then(function () {
|
|
|
|
routine.call(casper, test);
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
return fn;
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
register: _createRunner(register),
|
|
|
|
login: _createRunner(login),
|
|
|
|
logout: _createRunner(logout),
|
2013-12-30 11:03:29 +04:00
|
|
|
togglePermalinks: _createRunner(togglePermalinks)
|
2013-10-08 05:39:33 +04:00
|
|
|
};
|
|
|
|
|
|
|
|
}());
|