test.js 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379
  1. #!/usr/bin/env node
  2. /* jslint node:true */
  3. /* global it:false */
  4. /* global xit:false */
  5. /* global describe:false */
  6. /* global before:false */
  7. /* global after:false */
  8. 'use strict';
  9. require('chromedriver');
  10. var execSync = require('child_process').execSync,
  11. ejs = require('ejs'),
  12. expect = require('expect.js'),
  13. fs = require('fs'),
  14. mkdirp = require('mkdirp'),
  15. path = require('path'),
  16. rimraf = require('rimraf'),
  17. superagent = require('superagent'),
  18. util = require('util'),
  19. webdriver = require('selenium-webdriver');
  20. var by = require('selenium-webdriver').By,
  21. until = require('selenium-webdriver').until,
  22. Key = require('selenium-webdriver').Key,
  23. Builder = require('selenium-webdriver').Builder;
  24. process.env.NODE_TLS_REJECT_UNAUTHORIZED = '0';
  25. describe('Application life cycle test', function () {
  26. this.timeout(0);
  27. var server, browser = new Builder().forBrowser('chrome').build();
  28. var LOCATION = 'test';
  29. var app;
  30. var username = process.env.USERNAME;
  31. var password = process.env.PASSWORD;
  32. var adminUsername = 'admin';
  33. var adminPassword = 'changeme';
  34. var TIMEOUT = parseInt(process.env.TIMEOUT, 10) || 5000;
  35. var email, token;
  36. before(function (done) {
  37. if (!process.env.USERNAME) return done(new Error('USERNAME env var not set'));
  38. if (!process.env.PASSWORD) return done(new Error('PASSWORD env var not set'));
  39. var seleniumJar= require('selenium-server-standalone-jar');
  40. var SeleniumServer = require('selenium-webdriver/remote').SeleniumServer;
  41. server = new SeleniumServer(seleniumJar.path, { port: 4444 });
  42. server.start();
  43. done();
  44. });
  45. after(function (done) {
  46. browser.quit();
  47. server.stop();
  48. done();
  49. });
  50. function login(username, password, done) {
  51. browser.manage().deleteAllCookies().then(function () {
  52. return browser.get('https://' + app.fqdn + '/wp-login.php');
  53. }).then(function () {
  54. return browser.sleep(2000); // there seems to be some javascript that gives auto-focus to username
  55. }).then(function () {
  56. return browser.findElement(by.id('user_login')).sendKeys(username);
  57. }).then(function () {
  58. return browser.findElement(by.id('user_pass')).sendKeys(password);
  59. }).then(function () {
  60. return browser.findElement(by.tagName('form')).submit();
  61. }).then(function () {
  62. return browser.wait(until.elementLocated(by.xpath('//h1[text()="Dashboard"]')), TIMEOUT);
  63. }).then(function () {
  64. done();
  65. });
  66. }
  67. function logout(done) {
  68. browser.manage().deleteAllCookies().then(function () {
  69. browser.executeScript('localStorage.clear();');
  70. browser.executeScript('sessionStorage.clear();');
  71. done();
  72. });
  73. }
  74. function checkHtaccess(done) {
  75. var out = execSync(util.format('cloudron exec --app %s -- cat /app/data/htaccess', app.id));
  76. expect(out.toString('utf8').indexOf('RewriteEngine On')).to.not.be(-1); // wp generates this with permalinks in hard mode
  77. done();
  78. }
  79. function checkPermalink(done) {
  80. browser.get('https://' + app.fqdn + '/hello-world').then(function () {
  81. return browser.findElement(by.xpath('//h1[text()="Hello Cloudron!"]'));
  82. }).then(function () {
  83. done();
  84. });
  85. }
  86. function checkPost(done) {
  87. browser.get('https://' + app.fqdn).then(function () {
  88. if (app.manifest.version === '1.7.0') {
  89. return browser.wait(until.elementLocated(by.xpath('//h3/a[text()="Hello Cloudron!"]')), TIMEOUT);
  90. } else {
  91. return browser.wait(until.elementLocated(by.xpath('//h2/a[text()="Hello Cloudron!"]')), TIMEOUT);
  92. }
  93. }).then(function () {
  94. done();
  95. });
  96. }
  97. function editPostWordpress4(done) {
  98. browser.get('https://' + app.fqdn + '/wp-admin/post.php?post=1&action=edit').then(function () {
  99. return browser.wait(until.elementLocated(by.xpath('//input[@id="title"]')), TIMEOUT);
  100. }).then(function () {
  101. return browser.findElement(by.xpath('//input[@id="title"]')).sendKeys(Key.chord(Key.CONTROL, 'a'));
  102. }).then(function () {
  103. return browser.findElement(by.xpath('//input[@id="title"]')).sendKeys('Hello Cloudron!');
  104. }).then(function () {
  105. return browser.findElement(by.xpath('//input[@id="publish"]')).click();
  106. }).then(function () {
  107. return browser.wait(until.elementLocated(by.xpath('//*[contains(text(), "Post updated.")]')), TIMEOUT);
  108. }).then(function () {
  109. done();
  110. });
  111. }
  112. function editPost(done) {
  113. browser.get('https://' + app.fqdn + '/wp-admin/post.php?post=1&action=edit').then(function () {
  114. return browser.wait(until.elementLocated(by.xpath('//button[@aria-label="Disable tips"]')), TIMEOUT);
  115. }).then(function () {
  116. return browser.findElement(by.xpath('//button[@aria-label="Disable tips"]')).click();
  117. }).then(function () {
  118. return browser.findElement(by.xpath('//textarea[@id="post-title-0"]')).sendKeys(Key.chord(Key.CONTROL, 'a'));
  119. }).then(function () {
  120. return browser.findElement(by.xpath('//textarea[@id="post-title-0"]')).sendKeys('Hello Cloudron!');
  121. }).then(function () {
  122. return browser.findElement(by.xpath('//button[text()="Update"]')).click();
  123. }).then(function () {
  124. return browser.wait(until.elementLocated(by.xpath('//*[contains(text(), "Post updated.")]')), TIMEOUT);
  125. }).then(function () {
  126. return browser.sleep(3000);
  127. }).then(function () {
  128. done();
  129. });
  130. }
  131. function uploadMedia(done) {
  132. browser.get('https://' + app.fqdn + '/wp-admin/media-new.php?browser-uploader').then(function () {
  133. return browser.wait(until.elementLocated(by.id('async-upload')), TIMEOUT);
  134. }).then(function () {
  135. return browser.findElement(by.xpath('//input[@id="async-upload" and @type="file"]')).sendKeys(path.resolve(__dirname, '../logo.png'));
  136. }).then(function () {
  137. return browser.findElement(by.id('html-upload')).click();
  138. }).then(function () {
  139. return browser.wait(function () {
  140. return browser.getCurrentUrl().then(function (url) {
  141. return url === 'https://' + app.fqdn + '/wp-admin/upload.php';
  142. });
  143. }, TIMEOUT);
  144. }).then(function () {
  145. done();
  146. });
  147. }
  148. function checkMedia(item, done) {
  149. browser.get(`https://${app.fqdn}/wp-admin/upload.php?item=${item}`).then(function () { // there's got to be a better way..
  150. return browser.wait(until.elementLocated(by.xpath('//*[text()="Attachment Details"]')), TIMEOUT);
  151. }).then(function () {
  152. return browser.findElement(by.xpath('//img[@class="details-image"]')).getAttribute('src');
  153. }).then(function (srcLink) {
  154. console.log('media is located at ', srcLink);
  155. mediaLink = srcLink;
  156. done();
  157. });
  158. }
  159. function checkMediaLink(done) {
  160. superagent.get(mediaLink).end(function (error, result) {
  161. expect(error).to.be(null);
  162. expect(result.statusCode).to.be(200);
  163. done();
  164. });
  165. }
  166. xit('build app', function () {
  167. execSync('cloudron build', { cwd: path.resolve(__dirname, '..'), stdio: 'inherit' });
  168. });
  169. it('can login', function (done) {
  170. var inspect = JSON.parse(execSync('cloudron inspect'));
  171. superagent.post('https://' + inspect.apiEndpoint + '/api/v1/developer/login').send({
  172. username: username,
  173. password: password
  174. }).end(function (error, result) {
  175. if (error) return done(error);
  176. if (result.statusCode !== 200) return done(new Error('Login failed with status ' + result.statusCode));
  177. token = result.body.accessToken;
  178. superagent.get('https://' + inspect.apiEndpoint + '/api/v1/profile')
  179. .query({ access_token: token }).end(function (error, result) {
  180. if (error) return done(error);
  181. if (result.statusCode !== 200) return done(new Error('Get profile failed with status ' + result.statusCode));
  182. email = result.body.email;
  183. done();
  184. });
  185. });
  186. });
  187. it('install app', function () {
  188. execSync('cloudron install --new --wait --location ' + LOCATION, { cwd: path.resolve(__dirname, '..'), stdio: 'inherit' });
  189. });
  190. it('can get app information', function () {
  191. var inspect = JSON.parse(execSync('cloudron inspect'));
  192. app = inspect.apps.filter(function (a) { return a.location === LOCATION; })[0];
  193. expect(app).to.be.an('object');
  194. });
  195. it('can get the main page', function (done) {
  196. superagent.get('https://' + app.fqdn).end(function (error, result) {
  197. expect(error).to.be(null);
  198. expect(result.status).to.eql(200);
  199. done();
  200. });
  201. });
  202. it('can login', login.bind(null, username, password));
  203. it('can edit', editPost);
  204. it('is an admin dashboard', function (done) {
  205. browser.wait(until.elementLocated(by.xpath('//div[@class="wp-menu-name" and contains(text(), "Plugins")]')), TIMEOUT).then(function () { done(); });
  206. });
  207. it('can upload media', uploadMedia);
  208. var mediaLink;
  209. it('can see media', checkMedia.bind(null, 6));
  210. it('can see media link', checkMediaLink);
  211. it('has correct htaccess', checkHtaccess);
  212. it('can access permalink', checkPermalink);
  213. it('can restart app', function (done) {
  214. execSync('cloudron restart --wait --app ' + app.id);
  215. done();
  216. });
  217. it('can login', login.bind(null, username, password));
  218. it('can see updated post', checkPost);
  219. it('can see media link', checkMediaLink);
  220. it('has correct htaccess', checkHtaccess);
  221. it('can access permalink', checkPermalink);
  222. it('backup app', function () {
  223. execSync('cloudron backup create --app ' + app.id, { cwd: path.resolve(__dirname, '..'), stdio: 'inherit' });
  224. });
  225. it('restore app', function () {
  226. execSync('cloudron restore --app ' + app.id, { cwd: path.resolve(__dirname, '..'), stdio: 'inherit' });
  227. });
  228. it('can login', login.bind(null, username, password));
  229. it('can see updated post', checkPost);
  230. it('can see media link', checkMediaLink);
  231. it('has correct htaccess', checkHtaccess);
  232. it('can access permalink', checkPermalink);
  233. it('can login', login.bind(null, username, password));
  234. it('runs cron jobs', function (done) {
  235. this.timeout(6 * 60 * 1000); // cron runs only every 5 minutes
  236. console.log('It can take upto 6 mins to detect that cron is working');
  237. function checkLogs() {
  238. var logs = execSync('cloudron logs --lines 1000 --app ' + app.id).toString('utf8');
  239. if (logs.indexOf('Executed the cron event \'wp_version_check\'') !== -1) return done();
  240. setTimeout(checkLogs, 45000);
  241. }
  242. setTimeout(checkLogs, 45000);
  243. });
  244. it('move to different location', function () {
  245. browser.manage().deleteAllCookies();
  246. execSync('cloudron configure --wait --location ' + LOCATION + '2 --app ' + app.id, { cwd: path.resolve(__dirname, '..'), stdio: 'inherit' });
  247. var inspect = JSON.parse(execSync('cloudron inspect'));
  248. app = inspect.apps.filter(function (a) { return a.location === LOCATION + '2'; })[0];
  249. expect(app).to.be.an('object');
  250. mediaLink = mediaLink.replace(LOCATION, LOCATION + '2');
  251. });
  252. it('can login', login.bind(null, username, password));
  253. it('can see updated post', checkPost);
  254. it('can see media link', checkMediaLink);
  255. it('has correct htaccess', checkHtaccess);
  256. it('can access permalink', checkPermalink);
  257. it('can login', login.bind(null, username, password));
  258. it('can logout', logout);
  259. it('uninstall app', function () {
  260. execSync('cloudron uninstall --app ' + app.id, { cwd: path.resolve(__dirname, '..'), stdio: 'inherit' });
  261. });
  262. // check if the _first_ login via email succeeds
  263. it('can login via email', function (done) {
  264. execSync('cloudron install --new --wait --location ' + LOCATION, { cwd: path.resolve(__dirname, '..'), stdio: 'inherit' });
  265. var inspect = JSON.parse(execSync('cloudron inspect'));
  266. app = inspect.apps.filter(function (a) { return a.location === LOCATION; })[0];
  267. expect(app).to.be.an('object');
  268. login(email, password, function () {
  269. execSync('cloudron uninstall --app ' + app.id, { cwd: path.resolve(__dirname, '..'), stdio: 'inherit' });
  270. logout(done);
  271. });
  272. });
  273. // No SSO
  274. it('install app (no sso)', function () {
  275. execSync('cloudron install --new --wait --no-sso --location ' + LOCATION, { cwd: path.resolve(__dirname, '..'), stdio: 'inherit' });
  276. });
  277. it('can get app information', function () {
  278. var inspect = JSON.parse(execSync('cloudron inspect'));
  279. app = inspect.apps.filter(function (a) { return a.location === LOCATION; })[0];
  280. expect(app).to.be.an('object');
  281. });
  282. it('can login (no sso)', login.bind(null, adminUsername, adminPassword));
  283. it('is an admin dashboard (no sso)', function (done) {
  284. browser.wait(until.elementLocated(by.xpath('//div[@class="wp-menu-name" and contains(text(), "Plugins")]')), TIMEOUT).then(function () { done(); });
  285. });
  286. it('can logout', logout);
  287. it('uninstall app (no sso)', function () {
  288. execSync('cloudron uninstall --app ' + app.id, { cwd: path.resolve(__dirname, '..'), stdio: 'inherit' });
  289. });
  290. // test update
  291. it('can install app', function () {
  292. execSync('cloudron install --new --wait --appstore-id org.wordpress.cloudronapp --location ' + LOCATION, { cwd: path.resolve(__dirname, '..'), stdio: 'inherit' });
  293. var inspect = JSON.parse(execSync('cloudron inspect'));
  294. app = inspect.apps.filter(function (a) { return a.location === LOCATION; })[0];
  295. expect(app).to.be.an('object');
  296. });
  297. it('can login', login.bind(null, username, password));
  298. it('can edit', editPost);
  299. it('can upload media', uploadMedia);
  300. it('can update', function () {
  301. execSync('cloudron install --wait --app ' + LOCATION, { cwd: path.resolve(__dirname, '..'), stdio: 'inherit' });
  302. });
  303. it('can login', login.bind(null, username, password));
  304. it('can see updated post', checkPost);
  305. it('is an admin dashboard', function (done) {
  306. browser.wait(until.elementLocated(by.xpath('//div[@class="wp-menu-name" and contains(text(), "Plugins")]')), TIMEOUT).then(function () { done(); });
  307. });
  308. it('can see media', checkMedia.bind(null, 6));
  309. it('can see media link', checkMediaLink);
  310. it('can access permalink', checkPermalink);
  311. it('uninstall app', function () {
  312. execSync('cloudron uninstall --app ' + app.id, { cwd: path.resolve(__dirname, '..'), stdio: 'inherit' });
  313. });
  314. });