#!/usr/bin/env node 'use strict'; require('chromedriver'); var execSync = require('child_process').execSync, expect = require('expect.js'), fs = require('fs'), net = require('net'), path = require('path'), superagent = require('superagent'), util = require('util'), webdriver = require('selenium-webdriver'); var by = webdriver.By, until = webdriver.until, Builder = require('selenium-webdriver').Builder; process.env.NODE_TLS_REJECT_UNAUTHORIZED = '0'; if (!process.env.USERNAME || !process.env.PASSWORD) { console.log('USERNAME and PASSWORD env vars need to be set'); process.exit(1); } describe('Application life cycle test', function () { this.timeout(0); var server, browser = new Builder().forBrowser('chrome').build(); before(function (done) { var seleniumJar= require('selenium-server-standalone-jar'); var SeleniumServer = require('selenium-webdriver/remote').SeleniumServer; server = new SeleniumServer(seleniumJar.path, { port: 4444 }); server.start(); done(); }); after(function (done) { browser.quit(); server.stop(); done(); }); var LOCATION = 'test'; var TEST_TIMEOUT = 50000; var app; function waitForElement(elem) { return browser.wait(until.elementLocated(elem), TEST_TIMEOUT).then(function () { return browser.wait(until.elementIsVisible(browser.findElement(elem)), TEST_TIMEOUT); }); } function welcomePage(callback) { browser.get('https://' + app.fqdn).then(function () { return waitForElement(by.xpath('//*[text()="Cloudron LAMP App"]')); }).then(function () { return waitForElement(by.xpath('//h1[contains(text(), "7.2.10-0ubuntu0.18.04.1")]')); }).then(function () { callback(); }); } function uploadedFileExists(callback) { browser.get('https://' + app.fqdn + '/test.php').then(function () { return waitForElement(by.xpath('//*[text()="this works"]')); }).then(function () { return waitForElement(by.xpath('//*[text()="' + app.fqdn + '"]')); }).then(function () { callback(); }); } function checkIonCube(callback) { browser.get('https://' + app.fqdn).then(function () { return waitForElement(by.xpath('//td[contains(text(), "ionCube Loader")]')); // return waitForElement(by.xpath('//*[contains(text(), "Intrusion Protection&nsbp;from ioncube24.com")]')); }).then(function () { callback(); }); } function checkPhpMyAdmin(callback) { superagent.get('https://' + app.fqdn + '/phpmyadmin').end(function (error, result) { if (error && !error.response) return callback(error); // network error if (result.statusCode !== 401) return callback('Expecting 401 error'); superagent.get('https://' + app.fqdn + '/phpmyadmin') .auth(process.env.USERNAME, process.env.PASSWORD) .end(function (error, result) { if (error) return callback(error); if (result.text.indexOf(`${app.fqdn} / mysql | phpMyAdmin`) === -1) { // in the console.log(result.text); return callback(new Error('could not detect phpmyadmin')); } callback(); }); }); } function checkCron(callback) { this.timeout(60000 * 2); fs.writeFileSync('/tmp/crontab', '* * * * * echo -n "$MYSQL_HOST" > /app/data/public/cron\n', 'utf8'); execSync('cloudron push /tmp/crontab /app/data/crontab'); fs.unlinkSync('/tmp/crontab'); execSync('cloudron restart --wait'); console.log('Waiting for crontab to trigger'); setTimeout(function () { superagent.get('https://' + app.fqdn + '/cron').end(function (error, result) { if (error && !error.response) return callback(error); // network error if (result.statusCode !== 200) return callback('Expecting 200, got ' + result.statusCode); if (result.text !== 'mysql') return callback('Unexpected text: ' + result.text); callback(); }); }, 60 * 1000); // give it a minute to run the crontab } xit('build app', function () { execSync('cloudron build', { cwd: path.resolve(__dirname, '..'), stdio: 'inherit' }); }); it('install app', function () { execSync('cloudron install --new --wait --location ' + LOCATION, { cwd: path.resolve(__dirname, '..'), stdio: 'inherit' }); }); it('can get app information', function () { var inspect = JSON.parse(execSync('cloudron inspect')); app = inspect.apps.filter(function (a) { return a.location === LOCATION; })[0]; expect(app).to.be.an('object'); }); it('can view welcome page', welcomePage); it('can access ioncube', checkIonCube); it('can upload file with sftp', function () { // remove from known hosts in case this test was run on other apps with the same domain already // if the tests fail here you want below in ~/.ssh/config // Host test.cloudron.xyz // StrictHostKeyChecking no // HashKnownHosts no console.log('If this test fails, see the comment above this log message'); execSync(util.format('sed -i \'/%s/d\' -i ~/.ssh/known_hosts', app.fqdn)); execSync(util.format('lftp sftp://%s:%s@%s:%s -e "set sftp:auto-confirm yes; cd public/; put test.php; bye"', process.env.USERNAME, process.env.PASSWORD, app.fqdn, app.portBindings.SFTP_PORT)); }); it('can get uploaded file', uploadedFileExists); it('can access phpmyadmin', checkPhpMyAdmin); it('executes cron tasks', checkCron); it('backup app', function () { execSync('cloudron backup create --app ' + app.id, { cwd: path.resolve(__dirname, '..'), stdio: 'inherit' }); }); it('restore app', function () { execSync('cloudron restore --app ' + app.id, { cwd: path.resolve(__dirname, '..'), stdio: 'inherit' }); }); it('can get uploaded file', uploadedFileExists); it('move to different location', function () { browser.manage().deleteAllCookies(); execSync('cloudron configure --wait --location ' + LOCATION + '2 --app ' + app.id, { cwd: path.resolve(__dirname, '..'), stdio: 'inherit' }); var inspect = JSON.parse(execSync('cloudron inspect')); app = inspect.apps.filter(function (a) { return a.location === LOCATION + '2'; })[0]; expect(app).to.be.an('object'); }); it('can get uploaded file', uploadedFileExists); it('can access phpmyadmin', checkPhpMyAdmin); it('can access ioncube', checkIonCube); // disable SFTP it('can disable sftp', function () { execSync('cloudron configure --wait -p SFTP_PORT=', { cwd: path.resolve(__dirname, '..'), stdio: 'inherit' }); }); it('(nosftp) can view welcome page', welcomePage); it('(nosftp cannot upload file with sftp', function (done) { var client = new net.Socket(); client.setTimeout(10000); client.connect(2222, app.fqdn, function() { client.destroy(); done(new Error('Connected')); }); client.on('timeout', function () { client.destroy(); done(); }); // the packet just got dropped (good) client.on('error', function (error) { client.destroy(); done(new Error('Should have got timeout but got error:' + error.message)); }); }); it('(nosftp) cannot access phpmyadmin', function (done) { superagent.get('https://' + app.fqdn + '/phpmyadmin').end(function (error, result) { if (error && !error.response) return done(error); // network error if (result.statusCode !== 404) return done('Expecting 404 error'); done(); }); }); it('uninstall app', function () { execSync('cloudron uninstall --app ' + app.id, { cwd: path.resolve(__dirname, '..'), stdio: 'inherit' }); }); // test update it('can install app', function () { execSync('cloudron install --new --wait --appstore-id lamp.cloudronapp --location ' + LOCATION, { cwd: path.resolve(__dirname, '..'), stdio: 'inherit' }); var inspect = JSON.parse(execSync('cloudron inspect')); app = inspect.apps.filter(function (a) { return a.location === LOCATION; })[0]; expect(app).to.be.an('object'); }); it('can upload file with sftp', function () { // remove from known hosts in case this test was run on other apps with the same domain already // if the tests fail here you want to set "HashKnownHosts no" in ~/.ssh/config execSync(util.format('sed -i \'/%s/d\' -i ~/.ssh/known_hosts', app.fqdn)); execSync(util.format('lftp sftp://%s:%s@%s:%s -e "set sftp:auto-confirm yes; cd public/; put test.php; bye"', process.env.USERNAME, process.env.PASSWORD, app.fqdn, app.portBindings.SFTP_PORT)); }); it('can update', function () { execSync('cloudron install --wait --app ' + LOCATION, { cwd: path.resolve(__dirname, '..'), stdio: 'inherit' }); }); it('can get uploaded file', uploadedFileExists); it('can access phpmyadmin', checkPhpMyAdmin); it('can access ioncube', checkIonCube); it('uninstall app', function () { execSync('cloudron uninstall --app ' + app.id, { cwd: path.resolve(__dirname, '..'), stdio: 'inherit' }); }); });