diff --git a/test/test.js b/test/test.js index 251bd69..dc4db59 100644 --- a/test/test.js +++ b/test/test.js @@ -1,124 +1,60 @@ #!/usr/bin/env node -import { execSync } from 'node:child_process'; import assert from 'node:assert/strict'; import fs from 'node:fs'; import path from 'node:path'; import superagent from '@cloudron/superagent'; -import timers from 'node:timers/promises'; -import { Builder, By, until } from 'selenium-webdriver'; -import { Options } from 'selenium-webdriver/chrome'; -/* global it, xit, describe, before, after, afterEach */ +import { app, clearCache, click, cloudronCli, goto, LOCATION, sendKeys, setupBrowser, takeScreenshot, teardownBrowser, waitFor } from '@cloudron/charlie'; -if (!process.env.USERNAME || !process.env.PASSWORD) { - console.log('USERNAME and PASSWORD env vars need to be set'); - process.exit(1); -} +/* global it, describe, before, after, afterEach */ describe('Application life cycle test', function () { - this.timeout(0); - - const LOCATION = process.env.LOCATION || 'test'; - const TEST_TIMEOUT = parseInt(process.env.TIMEOUT, 10) || 30000; const BUCKET = 'cloudrontestbucket'; - const EXEC_ARGS = { cwd: path.resolve(import.meta.dirname, '..'), stdio: 'inherit' }; - let browser, app; - let rootPassword; - const username = process.env.USERNAME; - const password = process.env.PASSWORD; - - before(function () { - const chromeOptions = new Options().windowSize({ width: 1600, height: 1024 }); - chromeOptions.addArguments('guest'); // disable password checks - if (process.env.CI) chromeOptions.addArguments('no-sandbox', 'disable-dev-shm-usage', 'headless'); - browser = new Builder().forBrowser('chrome').setChromeOptions(chromeOptions).build(); - if (!fs.existsSync('./screenshots')) fs.mkdirSync('./screenshots'); - }); - - after(function () { - browser.quit(); - }); + before(setupBrowser); + after(teardownBrowser); afterEach(async function () { - if (!process.env.CI || !app) return; - - const currentUrl = await browser.getCurrentUrl(); - if (!currentUrl.includes(app.domain)) return; - assert.strictEqual(typeof this.currentTest.title, 'string'); - - const screenshotData = await browser.takeScreenshot(); - fs.writeFileSync(`./screenshots/${new Date().getTime()}-${this.currentTest.title.replaceAll(' ', '_')}.png`, screenshotData, 'base64'); + await takeScreenshot(this.currentTest.title); }); - async function waitForElement(elem) { - await browser.wait(until.elementLocated(elem), TEST_TIMEOUT); - await browser.wait(until.elementIsVisible(browser.findElement(elem)), TEST_TIMEOUT); - } - - function getAppInfo() { - var inspect = JSON.parse(execSync('cloudron inspect')); - app = inspect.apps.filter(function (a) { return a.location.indexOf(LOCATION) === 0; })[0]; - assert.ok(app && typeof app === 'object'); - } - async function confirmLicense() { - await timers.setTimeout(5000); - await waitForElement(By.id('acknowledge-confirm')); - const button = await browser.findElement(By.id('acknowledge-confirm')); - await browser.executeScript('arguments[0].scrollIntoView(false)', button); - await browser.sleep(4000); - await button.click(); - await browser.sleep(4000); + try { + await waitFor('css=#acknowledge-confirm', { timeout: 10000 }); + await click('css=#acknowledge-confirm'); + } catch { + // license modal not shown + } } - async function login(username, password, acceptLicense=false) { - await browser.manage().deleteAllCookies(); - await browser.get('about:blank'); - await browser.sleep(2000); - await browser.get(`https://${app.fqdn}/login`); - await browser.sleep(2000); - - await waitForElement(By.id('accessKey')); - await browser.findElement(By.id('accessKey')).sendKeys(username); - await browser.findElement(By.id('secretKey')).sendKeys(password); - await browser.findElement(By.xpath('//button[@id="do-login"]')).click(); + async function login(accessKey, secretKey, acceptLicense = false) { + await clearCache(); + await goto(`https://${app.fqdn}/login`, 'css=#accessKey'); + await sendKeys('css=#accessKey', accessKey); + await sendKeys('css=#secretKey', secretKey); + await click('css=#do-login'); if (acceptLicense) await confirmLicense(); - await waitForElement(By.xpath('//button[contains(., "Create Bucket")]')); - await timers.setTimeout(5000); + await waitFor('xpath=//button[contains(., "Create Bucket")]'); } async function logout() { - await browser.sleep(10000); - // await browser.get(`https://${app.fqdn}/`); - await waitForElement(By.xpath('//button[contains(., "Create Bucket")]')); - - const button = await browser.findElement(By.xpath('//button[@id="sign-out"]')); - await browser.executeScript('arguments[0].scrollIntoView(false)', button); - await button.click(); - await browser.sleep(10000); // needed! - await waitForElement(By.xpath('//*[@id="accessKey"]')); + await waitFor('xpath=//button[contains(., "Create Bucket")]'); + await click('css=#sign-out'); + await waitFor('css=#accessKey'); } async function addBucket() { - // await browser.get(`https://${app.fqdn}/buckets`); - await waitForElement(By.xpath('//button[contains(., "Create Bucket")]')); - await browser.findElement(By.xpath('//button[contains(., "Create Bucket")]')).click(); - await browser.sleep(1000); - await waitForElement(By.xpath('//input[@id="bucket-name"]')); - await browser.findElement(By.xpath('//input[@id="bucket-name"]')).sendKeys(BUCKET); - await browser.findElement(By.xpath('//button[@id="create-bucket"]')).click(); - // await browser.get(`https://${app.fqdn}/buckets`); - await waitForElement(By.xpath(`//h1[contains(text(), "${BUCKET}")]`)); + await click('xpath=//button[contains(., "Create Bucket")]'); + await sendKeys('css=#bucket-name', BUCKET); + await click('css=#create-bucket'); + await waitFor(`xpath=//h1[contains(text(), "${BUCKET}")]`); } async function checkBucket() { - await browser.sleep(10000); - // await browser.get(`https://${app.fqdn}/buckets`); - await waitForElement(By.xpath(`//h1[contains(text(), "${BUCKET}")]`)); + await waitFor(`xpath=//h1[contains(text(), "${BUCKET}")]`); } async function checkRedirect() { @@ -137,28 +73,12 @@ describe('Application life cycle test', function () { let data = fs.readFileSync(path.join(import.meta.dirname, '../env.sh.template'), 'utf8'); data += '\nexport MINIO_ROOT_USER=minioakey\nexport MINIO_ROOT_PASSWORD=minioskey\n'; fs.writeFileSync('/tmp/env.sh', data); - execSync(`cloudron push --app ${app.id} /tmp/env.sh /app/data/env.sh`, EXEC_ARGS); - execSync(`cloudron restart --app ${app.id}`, EXEC_ARGS); - await timers.setTimeout(10000); + await cloudronCli.push('/tmp/env.sh', '/app/data/env.sh'); + await cloudronCli.restart(); } - async function getAdminCredentials() { - execSync(`cloudron pull --app ${app.id} /app/data/env.sh /tmp/env.sh`, EXEC_ARGS); - const data = fs.readFileSync('/tmp/env.sh', 'utf8'); - const m = data.match(/MINIO_ROOT_PASSWORD=(.*)/); - if (!m) throw new Error('Could not detect root password'); - rootPassword = m[1].trim(); - console.log(`root password is [${rootPassword}]`); - } + it('install app', () => cloudronCli.install({ secondaryDomains: { API_SERVER_DOMAIN: `${LOCATION}-api` } })); - xit('build app', function () { execSync('cloudron build', EXEC_ARGS); }); - - it('install app', async function () { - execSync(`cloudron install --location ${LOCATION} --secondary-domains API_SERVER_DOMAIN=${LOCATION}-api`, EXEC_ARGS); - await timers.setTimeout(10000); - }); - - it('can get app information', getAppInfo); it('can admin login', login.bind(null, 'minioadmin', 'minioadmin', true)); it('can add bucket', addBucket); it('can logout', logout); @@ -167,10 +87,7 @@ describe('Application life cycle test', function () { it('can change admin credentials', changeAdminCredentials); - it('can restart app', async function () { - execSync(`cloudron restart --app ${app.id}`, EXEC_ARGS); - await timers.setTimeout(10000); - }); + it('can restart app', cloudronCli.restart); it('can admin login', login.bind(null, 'minioakey', 'minioskey', false)); it('has bucket', checkBucket); @@ -178,17 +95,9 @@ describe('Application life cycle test', function () { it('does redirect', checkRedirect); it('check api', checkApi); - it('backup app', function () { execSync('cloudron backup create --app ' + app.id, EXEC_ARGS); }); - it('restore app', async function () { - const backups = JSON.parse(execSync(`cloudron backup list --raw --app ${app.id}`)); - execSync('cloudron uninstall --app ' + app.id, EXEC_ARGS); - execSync('cloudron install --location ' + LOCATION, EXEC_ARGS); - getAppInfo(); - execSync(`cloudron restore --backup ${backups[0].id} --app ${app.id}`, EXEC_ARGS); - await timers.setTimeout(10000); - }); + it('backup app', cloudronCli.createBackup); + it('restore app', cloudronCli.restoreFromLatestBackup); - it('can get app information', getAppInfo); it('can admin login', login.bind(null, 'minioakey', 'minioskey', false)); it('has bucket', checkBucket); it('can logout', logout); @@ -196,12 +105,7 @@ describe('Application life cycle test', function () { it('does redirect', checkRedirect); it('check api', checkApi); - it('move to different location', async function () { - browser.manage().deleteAllCookies(); - execSync('cloudron configure --location ' + LOCATION + '2', EXEC_ARGS); - await timers.setTimeout(10000); - }); - it('can get app information', getAppInfo); + it('move to different location', cloudronCli.changeLocation); it('can admin login', login.bind(null, 'minioakey', 'minioskey', true)); it('has bucket', checkBucket); @@ -210,27 +114,5 @@ describe('Application life cycle test', function () { it('does redirect', checkRedirect); it('check api', checkApi); - it('uninstall app', function () { execSync('cloudron uninstall --app ' + app.id, EXEC_ARGS); }); - - // test update -// it('can install app for update', function () { execSync('cloudron install --appstore-id io.minio.cloudronapp --location ' + LOCATION, EXEC_ARGS); }); -// it('can get app information', getAppInfo); -// -// it('can get admin credentials', getAdminCredentials); -// it('can admin login', async () => await login('minioadmin', rootPassword, true)); -// it('can add buckets', addBucket); -// it('can logout', logout); -// -// it('can update', function () { execSync(`cloudron update --app ${LOCATION}`, EXEC_ARGS); }); -// it('can configure', function () { execSync(`cloudron configure --app ${LOCATION} --location ${LOCATION} --secondary-domains API_SERVER_DOMAIN=${LOCATION}-api`, EXEC_ARGS); }); -// it('can get app information', getAppInfo); -// -// it('can admin login', async () => await login('minioadmin', rootPassword, true)); -// it('has bucket', checkBucket); -// it('can logout', logout); -// -// it('does redirect', checkRedirect); -// it('check api', checkApi); -// -// it('uninstall app', function () { execSync('cloudron uninstall --app ' + app.id, EXEC_ARGS); }); + it('uninstall app', cloudronCli.uninstall); });