1
0
mirror of https://git.cloudron.io/cloudron/minio-app synced 2026-05-03 15:35:50 +00:00

Compare commits

...

8 Commits

Author SHA1 Message Date
Girish Ramakrishnan 231a7485a4 test: simplify clearCache, drop logout helpers, simplify button selectors
- Drop `await clearCache()` from login/auth helpers; clearing cache before
  every login is no longer necessary now that charlie isolates sessions.
- Replace `logout` helpers with `clearCache` directly in `it()` calls; the
  helpers were either thin wrappers around `clearCache()` or did real
  navigation that is no longer worth the maintenance.
- Simplify `xpath=//button[text()=...]` / `[contains(., ...)]` button text
  selectors to charlie's bare-string (or RegExp for OR-clauses) form.
2026-05-01 10:40:49 +02:00
Girish Ramakrishnan b3fce86d02 test: pass full Mocha test to takeScreenshot for unique screenshot names 2026-04-29 17:11:05 +02:00
Girish Ramakrishnan 5517962ee8 test: port to charlie
Made-with: Cursor
2026-04-24 20:44:33 +02:00
Girish Ramakrishnan bbf33e6cdd test: remove package.json and package-lock.json, charlie provides all deps
Made-with: Cursor
2026-04-16 10:43:47 +02:00
Girish Ramakrishnan a2a90e130b test: remove chromedriver dependency
Selenium Manager auto-detects the system ChromeDriver, making the
chromedriver npm package unnecessary.

Made-with: Cursor
2026-04-15 17:18:56 +02:00
Girish Ramakrishnan 5fd9ba74ba test: use @cloudron scoped safetydance and superagent
Made-with: Cursor
2026-04-15 16:39:12 +02:00
Girish Ramakrishnan b1f79dfa1e test: convert test to ESM
Replace CommonJS require() with ESM imports, add "type": "module"
to test/package.json, remove 'use strict' and jshint directives,
replace __dirname with import.meta.dirname.

Made-with: Cursor
2026-04-15 16:16:58 +02:00
Girish Ramakrishnan 3d5d79ba0b Replace expect.js with node:assert/strict
expect.js is unmaintained and unnecessary — Node's built-in assert
module covers all our assertion patterns. This also removes expect.js
from package.json dependencies.

Made-with: Cursor
2026-04-15 15:38:00 +02:00
3 changed files with 43 additions and 2477 deletions
-2285
View File
File diff suppressed because it is too large Load Diff
-19
View File
@@ -1,19 +0,0 @@
{
"name": "test",
"version": "1.0.0",
"description": "",
"main": "index.js",
"scripts": {
"test": "echo \"Error: no test specified\" && exit 1"
},
"author": "",
"license": "ISC",
"dependencies": {
"@eslint/js": "^9.37.0",
"chromedriver": "^141.0.3",
"expect.js": "^0.3.1",
"mocha": "^11.7.4",
"selenium-webdriver": "^4.36.0",
"superagent": "^10.2.3"
}
}
+43 -173
View File
@@ -1,241 +1,111 @@
#!/usr/bin/env node
/* jshint esversion: 8 */
/* global it, xit, describe, before, after, afterEach */
import assert from 'node:assert/strict';
import fs from 'node:fs';
import path from 'node:path';
import superagent from '@cloudron/superagent';
'use strict';
import { app, clearCache, click, cloudronCli, goto, LOCATION, sendKeys, setupBrowser, takeScreenshot, teardownBrowser, waitFor } from '@cloudron/charlie';
require('chromedriver');
const execSync = require('child_process').execSync,
expect = require('expect.js'),
fs = require('fs'),
path = require('path'),
superagent = require('superagent'),
timers = require('timers/promises'),
{ Builder, By, until } = require('selenium-webdriver'),
{ Options } = require('selenium-webdriver/chrome');
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(__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;
expect(this.currentTest.title).to.be.a('string');
const screenshotData = await browser.takeScreenshot();
fs.writeFileSync(`./screenshots/${new Date().getTime()}-${this.currentTest.title.replaceAll(' ', '_')}.png`, screenshotData, 'base64');
await takeScreenshot(this.currentTest);
});
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];
expect(app).to.be.an('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 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);
}
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('Create Bucket');
}
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('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() {
const response = await superagent.get(`https://${app.secondaryDomains[0].fqdn}`).set('User-Agent', 'Mozilla/5.0').redirects(0).ok(() => true);
expect(response.status).to.be(307);
expect(response.headers.location).to.be(`https://${app.fqdn}`);
assert.strictEqual(response.status, 307);
assert.strictEqual(response.headers.location, `https://${app.fqdn}`);
}
async function checkApi() {
const response = await superagent.get(`https://${app.secondaryDomains[0].fqdn}`).ok(() => true);
expect(response.status).to.be(403);
expect(response.body.toString('utf8')).to.contain('<Code>AccessDenied</Code>');
assert.strictEqual(response.status, 403);
assert.ok(response.body.toString('utf8').includes('<Code>AccessDenied</Code>'));
}
async function changeAdminCredentials() {
let data = fs.readFileSync(path.join(__dirname, '../env.sh.template'), 'utf8');
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);
it('can logout', clearCache);
it('does redirect', checkRedirect);
it('check api', checkApi);
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);
it('can logout', logout);
it('can logout', clearCache);
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);
it('can logout', clearCache);
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);
it('can logout', logout);
it('can logout', clearCache);
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);
});