96dc79cfe6
- Convert all require()/module.exports to import/export across 260+ files - Add "type": "module" to package.json to enable ESM by default - Add migrations/package.json with "type": "commonjs" to keep db-migrate compatible - Convert eslint.config.js to ESM with sourceType: "module" - Replace __dirname/__filename with import.meta.dirname/import.meta.filename - Replace require.main === module with process.argv[1] === import.meta.filename - Remove 'use strict' directives (implicit in ESM) - Convert dynamic require() in switch statements to static import lookup maps (dns.js, domains.js, backupformats.js, backupsites.js, network.js) - Extract self-referencing exports.CONSTANT patterns into standalone const declarations (apps.js, services.js, locks.js, users.js, mail.js, etc.) - Lazify SERVICES object in services.js to avoid circular dependency TDZ issues - Add clearMailQueue() to mailer.js for ESM-safe queue clearing in tests - Add _setMockApp() to ldapserver.js for ESM-safe test mocking - Add _setMockResolve() wrapper to dig.js for ESM-safe DNS mocking in tests - Convert backupupload.js to use dynamic imports so --check exits before loading the module graph (which requires BOX_ENV) - Update check-install to use ESM import for infra_version.js - Convert scripts/ (hotfix, release, remote_hotfix.js, find-unused-translations) - All 1315 tests passing Migration stats (AI-assisted using Cursor with Claude): - Wall clock time: ~3-4 hours - Assistant completions: ~80-100 - Estimated token usage: ~1-2M tokens Co-authored-by: Cursor <cursoragent@cursor.com>
59 lines
2.1 KiB
JavaScript
59 lines
2.1 KiB
JavaScript
import assert from 'node:assert';
|
|
import BoxError from './boxerror.js';
|
|
import debugModule from 'debug';
|
|
import Docker from 'dockerode';
|
|
import safe from 'safetydance';
|
|
import * as tokens from './tokens.js';
|
|
|
|
const debug = debugModule('box:janitor');
|
|
|
|
export {
|
|
cleanupTokens,
|
|
cleanupDockerVolumes
|
|
};
|
|
|
|
const gConnection = new Docker({ socketPath: '/var/run/docker.sock' });
|
|
|
|
async function cleanupTokens() {
|
|
debug('Cleaning up expired tokens');
|
|
|
|
const [error, result] = await safe(tokens.delExpired());
|
|
if (error) return debug('cleanupTokens: error removing expired tokens. %o', error);
|
|
|
|
debug(`Cleaned up ${result} expired tokens`);
|
|
}
|
|
|
|
async function cleanupTmpVolume(containerInfo) {
|
|
assert.strictEqual(typeof containerInfo, 'object');
|
|
|
|
const cmd = 'find /tmp -type f -mtime +10 -exec rm -rf {} +'.split(' '); // 10 day old files
|
|
|
|
debug(`cleanupTmpVolume ${JSON.stringify(containerInfo.Names)}`);
|
|
|
|
const [error, execContainer] = await safe(gConnection.getContainer(containerInfo.Id).exec({ Cmd: cmd, AttachStdout: true, AttachStderr: true, Tty: false }));
|
|
if (error) throw new BoxError(BoxError.DOCKER_ERROR, `Failed to exec container: ${error.message}`);
|
|
|
|
const [startError, stream] = await safe(execContainer.start({ hijack: true }));
|
|
if (startError) throw new BoxError(BoxError.DOCKER_ERROR, `Failed to start exec container: ${startError.message}`);
|
|
|
|
gConnection.modem.demuxStream(stream, process.stdout, process.stderr);
|
|
|
|
return new Promise((resolve, reject) => {
|
|
stream.on('error', (error) => reject(new BoxError(BoxError.DOCKER_ERROR, `Failed to cleanup in exec container: ${error.message}`)));
|
|
stream.on('end', resolve);
|
|
});
|
|
}
|
|
|
|
async function cleanupDockerVolumes() {
|
|
debug('Cleaning up docker volumes');
|
|
|
|
const [error, containers] = await safe(gConnection.listContainers({ all: 0 }));
|
|
if (error) throw new BoxError(BoxError.DOCKER_ERROR, error);
|
|
|
|
for (const container of containers) {
|
|
await safe(cleanupTmpVolume(container), { debug }); // intentionally ignore error
|
|
}
|
|
|
|
debug('Cleaned up docker volumes');
|
|
}
|