test.js 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258
  1. #!/usr/bin/env node
  2. 'use strict';
  3. var execSync = require('child_process').execSync,
  4. expect = require('expect.js'),
  5. path = require('path'),
  6. superagent = require('superagent'),
  7. webdriver = require('selenium-webdriver');
  8. var by = webdriver.By,
  9. Keys = webdriver.Key,
  10. until = webdriver.until;
  11. process.env.NODE_TLS_REJECT_UNAUTHORIZED = '0';
  12. describe('Application life cycle test', function () {
  13. this.timeout(0);
  14. var chrome = require('selenium-webdriver/chrome');
  15. var server, browser = new chrome.Driver(), uploadedImageUrl;
  16. var username = 'admin', password = 'changeme';
  17. before(function (done) {
  18. var seleniumJar= require('selenium-server-standalone-jar');
  19. var SeleniumServer = require('selenium-webdriver/remote').SeleniumServer;
  20. server = new SeleniumServer(seleniumJar.path, { port: 4444 });
  21. server.start();
  22. done();
  23. });
  24. after(function (done) {
  25. browser.quit();
  26. server.stop();
  27. done();
  28. });
  29. var LOCATION = 'test';
  30. var TEST_TIMEOUT = parseInt(process.env.TIMEOUT, 10) || 30000;
  31. var app;
  32. var email, token;
  33. function login(done) {
  34. browser.manage().window().setSize(1280,768);
  35. browser.manage().deleteAllCookies();
  36. browser.get('https://' + app.fqdn + '/login').then(function () {
  37. return browser.executeScript('localStorage.clear();');
  38. }).then(function () {
  39. return browser.get('https://' + app.fqdn + '/login');
  40. }).then(function () {
  41. return browser.sleep(5000); // takes some time for username to be visible
  42. }).then(function () {
  43. return browser.wait(until.elementLocated(by.id('username')), TEST_TIMEOUT);
  44. }).then(function () {
  45. return browser.findElement(by.id('username')).sendKeys(username);
  46. }).then(function () {
  47. return browser.findElement(by.id('password')).sendKeys(password);
  48. }).then(function () {
  49. return browser.findElement(by.id('login')).click();
  50. }).then(function () {
  51. return browser.wait(until.elementLocated(by.xpath('//a[contains(text(), "Announcements")]')), TEST_TIMEOUT);
  52. }).then(function () {
  53. done();
  54. });
  55. }
  56. function checkMailPlugin(done) {
  57. browser.get('https://' + app.fqdn + '/admin/emailers/local').then(function () {
  58. return browser.wait(until.elementLocated(by.id('host')), TEST_TIMEOUT);
  59. }).then(function () {
  60. return browser.sleep(5000);
  61. }).then(function () {
  62. return browser.findElement(by.id('host')).getAttribute('value');
  63. }).then(function (val) {
  64. if (val !== 'mail') return done(new Error('Incorrect mail server value: ' + val));
  65. done();
  66. });
  67. }
  68. function restartForum(done) {
  69. browser.get('https://' + app.fqdn + '/admin').then(function () {
  70. return browser.sleep(3000);
  71. }).then(function () {
  72. return browser.findElement(by.xpath('//button[text()="Restart"]')).click();
  73. }).then(function () {
  74. return browser.sleep(3000);
  75. }).then(function () {
  76. return browser.findElement(by.xpath('//button[text()="Confirm"]')).click();
  77. }).then(function () {
  78. console.log('Waiting for forum to restart');
  79. return browser.sleep(40000); // wait 30secs for reload
  80. }).then(function () {
  81. return browser.get('https://' + app.fqdn + '/admin');
  82. }).then(function () {
  83. return browser.sleep(3000);
  84. }).then(function () {
  85. return browser.wait(until.elementLocated(by.xpath('//h1[text()="Dashboard"]')), TEST_TIMEOUT);
  86. }).then(function () { done(); });
  87. }
  88. function installCustomPlugin(done) {
  89. execSync('cloudron exec --app ' + app.id + ' -- /usr/local/bin/gosu cloudron:cloudron npm install nodebb-plugin-beep', { cwd: path.resolve(__dirname, '..'), stdio: 'inherit' });
  90. done();
  91. }
  92. function activateCustomPlugin(done) {
  93. browser.get('https://' + app.fqdn + '/admin/extend/plugins#installed').then(function () {
  94. return browser.wait(until.elementLocated(by.xpath('//ul[contains(@class, "installed")]//strong[text()="nodebb-plugin-beep"]')), TEST_TIMEOUT);
  95. }).then(function () {
  96. return browser.sleep(10000);
  97. }).then(function () {
  98. var button = browser.findElement(by.xpath('//li[@id="nodebb-plugin-beep"]//button[@data-action="toggleActive"]'));
  99. return browser.executeScript('arguments[0].scrollIntoView(false)', button);
  100. }).then(function () {
  101. return browser.findElement(by.xpath('//li[@id="nodebb-plugin-beep"]//button[@data-action="toggleActive"]')).click(); // activate the plugin
  102. }).then(function () {
  103. return browser.wait(until.elementLocated(by.xpath('//button[text()="Confirm"]')), TEST_TIMEOUT);
  104. }).then(function () {
  105. var button = browser.findElement(by.xpath('//button[text()="Confirm"]'));
  106. return browser.executeScript('arguments[0].scrollIntoView(false)', button);
  107. }).then(function () {
  108. return browser.findElement(by.xpath('//button[text()="Confirm"]')).click();
  109. }).then(function () {
  110. return browser.sleep(20000); // wait for the action to succeed
  111. }).then(function() {
  112. done();
  113. });
  114. }
  115. function listCustomPlugin(done) {
  116. browser.get('https://' + app.fqdn + '/admin/extend/plugins#installed').then(function () {
  117. return browser.wait(until.elementLocated(by.xpath('//strong[text()="nodebb-plugin-beep"]')), TEST_TIMEOUT);
  118. }).then(function () {
  119. done();
  120. });
  121. }
  122. function uploadImage(done) {
  123. browser.get('https://' + app.fqdn + '/user/admin/edit').then(function () {
  124. return browser.wait(until.elementLocated(by.xpath('//a[text()="Change Picture"]')), TEST_TIMEOUT);
  125. }).then(function () {
  126. var button = browser.findElement(by.xpath('//a[text()="Change Picture"]'));
  127. return browser.executeScript('arguments[0].scrollIntoView(false)', button);
  128. }).then(function () {
  129. return browser.findElement(by.xpath('//a[text()="Change Picture"]')).click();
  130. }).then(function () {
  131. return browser.sleep(4000);
  132. }).then(function () {
  133. return browser.findElement(by.xpath('//button[@data-action="upload"]')).click();
  134. }).then(function () {
  135. return browser.sleep(4000);
  136. }).then(function () {
  137. return browser.findElement(by.xpath('//input[@type="file"]')).sendKeys(path.resolve(__dirname, '../logo.png'));
  138. }).then(function () {
  139. browser.sleep(3000);
  140. }).then(function () { // upload it
  141. return browser.findElement(by.id('fileUploadSubmitBtn')).click();
  142. }).then(function () {
  143. browser.sleep(3000);
  144. }).then(function () {
  145. return browser.findElement(by.xpath('//button[text()="Crop and upload"]')).click();
  146. }).then(function () {
  147. return browser.sleep(3000);
  148. }).then(function () {
  149. done();
  150. });
  151. }
  152. function checkImage(done) {
  153. browser.get('https://' + app.fqdn + '/user/admin').then(function () {
  154. return browser.wait(until.elementLocated(by.xpath('//img[@src="/assets/uploads/profile/1-profileavatar.png"]')), TEST_TIMEOUT);
  155. }).then(function () {
  156. var img = browser.findElement(by.xpath('//img[@src="/assets/uploads/profile/1-profileavatar.png"]'));
  157. return browser.executeScript('return arguments[0].complete && arguments[0].naturalWidth', img);
  158. }).then(function (imageWidth) {
  159. done(imageWidth === 200 ? null : new Error('failed to load image'));
  160. });
  161. }
  162. xit('build app', function () {
  163. execSync('cloudron build', { cwd: path.resolve(__dirname, '..'), stdio: 'inherit' });
  164. });
  165. it('install app', function () {
  166. execSync('cloudron install --new --wait --location ' + LOCATION, { cwd: path.resolve(__dirname, '..'), stdio: 'inherit' });
  167. });
  168. it('can get app information', function () {
  169. var inspect = JSON.parse(execSync('cloudron inspect'));
  170. app = inspect.apps.filter(function (a) { return a.location === LOCATION; })[0];
  171. expect(app).to.be.an('object');
  172. });
  173. it('can login', login);
  174. it('check mail plugin', checkMailPlugin);
  175. it('can install custom plugin', installCustomPlugin);
  176. it('can restart forum', restartForum); // required before activate!
  177. it('can activate custom plugin', activateCustomPlugin);
  178. it('can list custom plugin', listCustomPlugin);
  179. it('can upload image', uploadImage);
  180. it('can check image', checkImage);
  181. it('backup app', function () {
  182. execSync('cloudron backup create --app ' + app.id, { cwd: path.resolve(__dirname, '..'), stdio: 'inherit' });
  183. });
  184. it('restore app', function () {
  185. execSync('cloudron restore --app ' + app.id, { cwd: path.resolve(__dirname, '..'), stdio: 'inherit' });
  186. });
  187. it('can login', login);
  188. it('can list custom plugin', listCustomPlugin);
  189. it('can check image', checkImage);
  190. it('can restart app', function (done) {
  191. execSync('cloudron restart --wait --app ' + app.id);
  192. done();
  193. });
  194. it('can login', login);
  195. it('can list custom plugin', listCustomPlugin);
  196. it('can check image', checkImage);
  197. it('move to different location', function () {
  198. execSync('cloudron configure --wait --location ' + LOCATION + '2 --app ' + app.id, { cwd: path.resolve(__dirname, '..'), stdio: 'inherit' });
  199. var inspect = JSON.parse(execSync('cloudron inspect'));
  200. app = inspect.apps.filter(function (a) { return a.location === LOCATION + '2'; })[0];
  201. expect(app).to.be.an('object');
  202. });
  203. it('can login', login);
  204. it('can list custom plugin', listCustomPlugin);
  205. it('can check image', checkImage);
  206. it('uninstall app', function () {
  207. execSync('cloudron uninstall --app ' + app.id, { cwd: path.resolve(__dirname, '..'), stdio: 'inherit' });
  208. });
  209. // test update
  210. it('can install app', function () {
  211. execSync('cloudron install --new --wait --appstore-id org.nodebb.cloudronapp --location ' + LOCATION, { cwd: path.resolve(__dirname, '..'), stdio: 'inherit' });
  212. var inspect = JSON.parse(execSync('cloudron inspect'));
  213. app = inspect.apps.filter(function (a) { return a.location === LOCATION; })[0];
  214. expect(app).to.be.an('object');
  215. });
  216. it('can login', login);
  217. it('check mail plugin', checkMailPlugin);
  218. it('can update', function () {
  219. execSync('cloudron install --wait --app ' + LOCATION, { cwd: path.resolve(__dirname, '..'), stdio: 'inherit' });
  220. });
  221. it('check mail plugin', checkMailPlugin);
  222. it('uninstall app', function () {
  223. execSync('cloudron uninstall --app ' + app.id, { cwd: path.resolve(__dirname, '..'), stdio: 'inherit' });
  224. });
  225. });