cloudcmd/server/route.spec.js
2022-04-23 01:34:07 +03:00

499 lines
9.9 KiB
JavaScript

'use strict';
const {Readable} = require('stream');
const path = require('path');
const fs = require('fs');
const tryToCatch = require('try-to-catch');
const {test, stub} = require('supertape');
const mockRequire = require('mock-require');
const {
reRequire,
stopAll,
} = mockRequire;
const fixtureDir = path.join(__dirname, '..', 'test', 'fixture');
const routePath = './route';
const cloudcmdPath = './cloudcmd';
const cloudcmd = require(cloudcmdPath);
const {createConfigManager} = cloudcmd;
const serveOnce = require('serve-once');
const defaultConfig = {
auth: false,
dropbox: false,
};
const {request} = serveOnce(cloudcmd, {
config: defaultConfig,
});
const {stringify} = JSON;
const {assign} = Object;
test('cloudcmd: route: buttons: no console', async (t) => {
const options = {
config: {
console: false,
},
};
const {body} = await request.get('/', {
options,
});
t.ok(/icon-console none/.test(body), 'should hide console');
t.end();
});
test('cloudcmd: route: buttons: console', async (t) => {
const config = {
console: true,
};
const options = {
config,
};
const {body} = await request.get('/', {
options,
});
t.notOk(/icon-console none/.test(body), 'should not hide console');
t.end();
});
test('cloudcmd: route: buttons: no config', async (t) => {
const config = {
configDialog: false,
};
const options = {
config,
};
const {body} = await request.get('/', {
options,
});
t.ok(/icon-config none/.test(body), 'should hide config');
t.end();
});
test('cloudcmd: route: buttons: no contact', async (t) => {
const config = {
contact: false,
};
const options = {
config,
};
const {body} = await request.get('/', {
options,
});
t.ok(/icon-contact none/.test(body), 'should hide contact');
t.end();
});
test('cloudcmd: route: buttons: one file panel: move', async (t) => {
const config = {
oneFilePanel: true,
};
const options = {
config,
};
const {body} = await request.get('/', {
options,
});
t.ok(/icon-move none/.test(body), 'should hide move button');
t.end();
});
test('cloudcmd: route: buttons: no one file panel: move', async (t) => {
const config = {
oneFilePanel: false,
};
const options = {
config,
};
const {body} = await request.get('/', {
options,
});
t.notOk(/icon-move none/.test(body), 'should not hide move button');
t.end();
});
test('cloudcmd: route: buttons: one file panel: move', async (t) => {
const config = {
oneFilePanel: true,
};
const options = {
config,
};
const {body} = await request.get('/', {
options,
});
t.ok(/icon-copy none/.test(body), 'should hide copy button');
t.end();
});
test('cloudcmd: route: keys panel: hide', async (t) => {
const config = {
keysPanel: false,
};
const options = {
config,
};
const {body} = await request.get('/', {
options,
});
t.ok(/keyspanel hidden/.test(body), 'should hide keyspanel');
t.end();
});
test('cloudcmd: route: keys panel', async (t) => {
const config = {
keysPanel: true,
};
const options = {
config,
};
const {body} = await request.get('/', {
options,
});
t.notOk(/keyspanel hidden/.test(body), 'should show keyspanel');
t.end();
});
test('cloudcmd: route: symlink', async (t) => {
const emptyDir = path.join(fixtureDir, 'empty-dir');
const root = fixtureDir;
const symlink = path.join(root, 'symlink-dir');
const config = {
root,
};
const options = {
config,
};
fs.symlinkSync(emptyDir, symlink);
const {body} = await request.get('/fs/symlink-dir', {
options,
});
fs.unlinkSync(symlink);
t.ok(body.length, 'should return html document');
t.end();
});
test('cloudcmd: route: not found', async (t) => {
const root = fixtureDir;
const config = {
root,
};
const options = {
config,
};
const {body} = await request.get('/fs/file-not-found', {
options,
});
t.match(body, 'ENOENT: no such file or directory', 'should return error');
t.end();
});
test('cloudcmd: route: sendIndex: encode', async (t) => {
const name = '"><svg onload=alert(3);>';
const nameEncoded = '&quot;&gt;&lt;svg onload=alert(3);&gt;';
const path = '/';
const files = [{
name,
}];
const stream = Readable.from(stringify({
path,
files,
}));
assign(stream, {
path,
files,
type: 'directory',
});
const read = stub().resolves(stream);
mockRequire('win32', {
read,
});
reRequire(routePath);
const cloudcmd = reRequire(cloudcmdPath);
const {request} = serveOnce(cloudcmd, {
configManager: createConfigManager(),
});
const {body} = await request.get('/');
stopAll();
t.ok(body.includes(nameEncoded), 'should encode name');
t.end();
});
test('cloudcmd: route: sendIndex: encode: not encoded', async (t) => {
const name = '"><svg onload=alert(3);>';
const path = '/';
const files = [{
name,
}];
const stream = Readable.from(stringify({
path,
files,
}));
assign(stream, {
path,
files,
type: 'directory',
});
const read = stub().resolves(stream);
mockRequire('win32', {
read,
});
reRequire(routePath);
const cloudcmd = reRequire(cloudcmdPath);
const {request} = serveOnce(cloudcmd);
const {body} = await request.get('/');
stopAll();
t.notOk(body.includes(name), 'should not put not encoded name');
t.end();
});
test('cloudcmd: route: sendIndex: ddos: render', async (t) => {
const name = `$$$'&quot;`;
const path = '/';
const files = [{
name,
}];
const stream = Readable.from(stringify({
path,
files,
}));
assign(stream, {
path,
files,
type: 'directory',
});
const read = stub().resolves(stream);
mockRequire('win32', {
read,
});
reRequire(routePath);
const cloudcmd = reRequire(cloudcmdPath);
const {request} = serveOnce(cloudcmd, {
config: defaultConfig,
});
const {status} = await request.get('/');
stopAll();
t.equal(status, 200, 'should not hang up');
t.end();
});
test('cloudcmd: route: buttons: no terminal', async (t) => {
const config = {
terminal: false,
};
const options = {
config,
};
const {body} = await request.get('/', {
options,
});
t.ok(/icon-terminal none/.test(body), 'should hide terminal');
t.end();
});
test('cloudcmd: route: no termianl: /fs', async (t) => {
const config = {
terminal: false,
};
const options = {
config,
configManager: createConfigManager(),
};
const {request} = serveOnce(cloudcmd);
const {body} = await request.get('/fs', {
options,
});
t.ok(/icon-terminal none/.test(body), 'should hide terminal');
t.end();
});
test('cloudcmd: route: buttons: terminal: can not load', async (t) => {
const config = {
terminal: true,
terminalPath: 'xxxxxxxxxxxx',
};
const options = {
config,
};
const {body} = await request.get('/', {
options,
});
t.ok(/icon-terminal none/.test(body), 'should not enable terminal');
t.end();
});
test('cloudcmd: route: buttons: terminal', async (t) => {
const config = {
terminal: true,
terminalPath: 'console-io',
};
const options = {
config,
};
const {body} = await request.get('/', {
options,
});
t.notOk(/icon-terminal none/.test(body), 'should not enable terminal');
t.end();
});
test('cloudcmd: route: buttons: contact', async (t) => {
const config = {
contact: true,
};
const options = {
config,
};
const {request} = serveOnce(cloudcmd);
const {body} = await request.get('/', {
options,
});
t.notOk(/icon-contact none/.test(body), 'should enable terminal');
t.end();
});
test('cloudcmd: route: dropbox', async (t) => {
const config = createConfigManager();
config('dropbox', true);
config('dropboxToken', '');
const {_getReadDir} = reRequire(routePath);
const readdir = _getReadDir(config);
const [e] = await tryToCatch(readdir, '/root');
t.ok(/token/.test(e.message), 'should contain word token in message');
t.end();
});
test('cloudcmd: route: content length', async (t) => {
const options = {
root: fixtureDir,
};
const {headers} = await request.get('/route.js', {
options,
});
const result = headers.get('content-length');
t.ok(result);
t.end();
});
test('cloudcmd: route: read: root', async (t) => {
const stream = Readable.from('hello');
stream.contentLength = 5;
const read = stub().returns(stream);
mockRequire('win32', {
read,
});
reRequire(routePath);
const cloudcmd = reRequire(cloudcmdPath);
const configManager = createConfigManager();
const root = '/hello';
configManager('root', root);
const {request} = serveOnce(cloudcmd, {
configManager,
});
await request.get('/fs/route.js');
const expected = [
'/hello/route.js', {
root,
},
];
stopAll();
t.calledWith(read, expected);
t.end();
});