Files
cloudron-box/src/sftp.js
T

105 lines
4.1 KiB
JavaScript
Raw Normal View History

2019-04-04 20:46:01 -07:00
'use strict';
exports = module.exports = {
2021-01-21 12:53:38 -08:00
start,
2021-02-09 22:57:21 -08:00
rebuild,
2021-01-21 12:53:38 -08:00
DEFAULT_MEMORY_LIMIT: 256 * 1024 * 1024
2019-04-04 20:46:01 -07:00
};
2021-05-11 17:50:48 -07:00
const apps = require('./apps.js'),
assert = require('assert'),
BoxError = require('./boxerror.js'),
debug = require('debug')('box:sftp'),
2020-10-19 17:26:20 -07:00
hat = require('./hat.js'),
2019-04-04 20:46:01 -07:00
infra = require('./infra_version.js'),
path = require('path'),
paths = require('./paths.js'),
2020-08-09 12:10:20 -07:00
safe = require('safetydance'),
2020-08-20 11:04:31 +02:00
shell = require('./shell.js'),
2021-01-21 12:53:38 -08:00
system = require('./system.js'),
volumes = require('./volumes.js');
2019-04-04 20:46:01 -07:00
2021-08-25 19:41:46 -07:00
async function rebuild(serviceConfig, options) {
2021-01-21 12:53:38 -08:00
assert.strictEqual(typeof serviceConfig, 'object');
2021-02-09 22:57:21 -08:00
assert.strictEqual(typeof options, 'object');
debug('rebuilding container');
2019-04-04 20:46:01 -07:00
const tag = infra.images.sftp.tag;
2021-01-21 12:53:38 -08:00
const memoryLimit = serviceConfig.memoryLimit || exports.DEFAULT_MEMORY_LIMIT;
const memory = system.getMemoryAllocation(memoryLimit);
2020-10-19 17:26:20 -07:00
const cloudronToken = hat(8 * 128);
2019-04-04 20:46:01 -07:00
2021-09-26 18:36:33 -07:00
const resolvedAppDataDir = safe.fs.realpathSync(paths.APPS_DATA_DIR);
if (!resolvedAppDataDir) throw new BoxError(BoxError.FS_ERROR, `Could not resolve apps data dir: ${safe.error.message}`);
2021-09-25 17:07:07 -07:00
const dataDirs = [{ hostDir: resolvedAppDataDir, mountDir: '/mnt/appsdata' }];
2021-08-25 19:41:46 -07:00
2021-09-25 17:07:07 -07:00
// custom app data directories
const allApps = await apps.list();
for (const app of allApps) {
if (!app.manifest.addons['localstorage'] || !app.dataDir) continue;
2021-08-25 19:41:46 -07:00
const hostDir = apps.getDataDir(app, app.dataDir), mountDir = `/mnt/${app.id}`;
if (!safe.fs.existsSync(hostDir)) { // this can fail if external mount does not have permissions for yellowtent user
// do not create host path when cloudron is restoring. this will then create dir with root perms making restore logic fail
debug(`Ignoring app data dir ${hostDir} for ${app.id} since it does not exist`);
2021-09-25 17:07:07 -07:00
continue;
2021-08-25 19:41:46 -07:00
}
dataDirs.push({ hostDir, mountDir });
2021-09-25 17:07:07 -07:00
}
2021-08-25 19:41:46 -07:00
2021-09-25 17:07:07 -07:00
// volume directories
2021-08-25 19:41:46 -07:00
dataDirs.push({ hostDir: '/mnt/volumes', mountDir: '/mnt/volumes' });
2021-09-25 17:07:07 -07:00
const allVolumes = await volumes.list();
for (const volume of allVolumes) {
if (volume.hostPath.startsWith('/mnt/volumes/')) continue;
2021-08-25 19:41:46 -07:00
if (!safe.fs.existsSync(volume.hostPath)) {
debug(`Ignoring volume host path ${volume.hostPath} since it does not exist`);
2021-09-25 17:07:07 -07:00
continue;
2021-08-25 19:41:46 -07:00
}
dataDirs.push({ hostDir: volume.hostPath, mountDir: `/mnt/${volume.id}` });
2021-09-25 17:07:07 -07:00
}
2021-08-25 19:41:46 -07:00
// mail data dir
2021-09-26 18:36:33 -07:00
const resolvedMailDataDir = safe.fs.realpathSync(paths.MAIL_DATA_DIR);
if (!resolvedMailDataDir) throw new BoxError(BoxError.FS_ERROR, `Could not resolve mail data dir: ${safe.error.message}`);
dataDirs.push({ hostDir: resolvedMailDataDir, mountDir: '/mnt/maildata' });
2021-09-25 17:07:07 -07:00
const mounts = dataDirs.map(v => `-v "${v.hostDir}:${v.mountDir}"`).join(' ');
2021-08-25 19:41:46 -07:00
const cmd = `docker run --restart=always -d --name="sftp" \
--hostname sftp \
--net cloudron \
--net-alias sftp \
--log-driver syslog \
--log-opt syslog-address=udp://127.0.0.1:2514 \
--log-opt syslog-format=rfc5424 \
--log-opt tag=sftp \
-m ${memory} \
--memory-swap ${memoryLimit} \
--dns 172.18.0.1 \
--dns-search=. \
-p 222:22 \
${mounts} \
-e CLOUDRON_SFTP_TOKEN="${cloudronToken}" \
-v "${paths.SFTP_KEYS_DIR}:/etc/ssh:ro" \
--label isCloudronManaged=true \
--read-only -v /tmp -v /run "${tag}"`;
// ignore error if container not found (and fail later) so that this code works across restarts
await shell.promises.exec('stopSftp', 'docker stop sftp || true');
await shell.promises.exec('removeSftp', 'docker rm -f sftp || true');
await shell.promises.exec('startSftp', cmd);
2019-04-04 20:46:01 -07:00
}
2021-01-21 12:53:38 -08:00
2021-08-25 19:41:46 -07:00
async function start(existingInfra, serviceConfig) {
2021-01-21 12:53:38 -08:00
assert.strictEqual(typeof existingInfra, 'object');
assert.strictEqual(typeof serviceConfig, 'object');
2021-08-25 19:41:46 -07:00
await rebuild(serviceConfig, { force: true }); // force rebuild when infra changed
2021-01-21 12:53:38 -08:00
}