simple-sandbox
Version:
A simple sandbox for Node.js using Linux namespaces and cgroup.
106 lines (76 loc) • 4.07 kB
Markdown
# simple-sandbox
A simple linux sandbox with Node.js API. Used by SYZOJ.
## Prerequisites
### Packages
You need to have the `build-essentials` (`g++`, `make`, etc.) and the `fmt` library installed in your system in order to build the C++ part.
The minimal `g++` version required is `g++-8`. A newer version of `clang++` with C++17 file system support is recommended.
Install them by (in Ubuntu 18.04):
```bash
apt install build-essential clang++-9 libfmt-dev
```
### Kernel
You need to have the memory swap account (disabled by default in Debian 8) enabled with your kernel. You can verify this by checking the existence of `/sys/fs/cgroup/memory/memory.memsw.usage_in_bytes`
If that file does not exist, then you may have to turn on that with your grub.
Add `swapaccount=1` to `GRUB_CMDLINE_LINUX_DEFAULT` section in `/etc/default/grub`.
```bash
GRUB_CMDLINE_LINUX_DEFAULT="quiet splash cgroup_enable=memory swapaccount=1"
```
Some distro [enables cgroup v2 by default in their new versions](https://rootlesscontaine.rs/getting-started/common/cgroup2/), including Arch Linux (since April 2021), Fedora (since 31) and Debian. If you cannot find the directory `/sys/fs/cgroup/memory/`, this is the case for you. In this case, you also need to add the parameter `systemd.unified_cgroup_hierarchy=0` to enable cgroup v1:
```bash
GRUB_CMDLINE_LINUX_DEFAULT="quiet splash cgroup_enable=memory swapaccount=1 systemd.unified_cgroup_hierarchy=0"
```
And then run
```bash
update-grub && reboot
```
To enjoy the modified kernel.
## Build
Pull the repository to somewhere on your computer and run
```bash
CXX=clang++-9 yarn install # Install required packages and compile C++ code with the clang++-9 compiler
yarn run build # Compile typescript code.
```
To build the project. If you don't want to use `yarn`, just change `yarn` to `npm`.
You can use `yarn run build:watch` to watch for the change of typescript file.
## Use
The library is with a simple API.
To start the sandbox, use the following code:
```js
const sandbox = require('simple-sandbox');
const myProcess = await sandbox.startSandbox(parameters);
```
where `parameters` is an instance of the [SandboxParameters interface](src/interfaces.ts). The detail explanation is available in the comments in that file.
The startSandbox function returns a Promise, from which you can get an instance of the [sandboxProcess](src/sandboxProcess.ts) class, reprensenting your sandboxed Process.
To terminate the sandboxed process, just use the `stop()` function:
```js
myProcess.stop();
```
To wait for the sandboxed process to end, use the `waitForStop()` function, which returns a `Promise`:
```js
myProcess.waitForStop().then((result) => {
console.log("OK! " + JSON.stringify(result));
}).catch((err) => {
console.log("Whooops!" + err.toString());
});
```
Note that `myProcess` itself is a EventEmitter, so you can register `exit` (indicates that the child process exited), and `error` (indicates that some error happens) event listener on it.
### Note
When a sandbox is started, a event listener for the `exit` event on the `process` object is registered. When Node.js is about to exit, it will kill the sandboxed process.
However, the `exit` event won't be called if there are `SIGTERM` or `SIGINT` (Ctrl-C) signals sent to the Node.js process. You should let the `SIGTERM` and `SIGINT` handler to call `process.exit()` on the main process:
```js
const terminationHandler = () => {
process.exit(1);
};
process.on('SIGTERM', terminationHandler);
process.on('SIGINT', terminationHandler);
```
## Example
A demostration is available in the `demo` directory.
In order to get the demostration running for every one, we create the directory `/opt/sandbox-test`.
```bash
sudo mkdir /opt/sandbox-test
sudo chown $(whoami) /opt/sandbox-test
mkdir /opt/sandbox-test/rootfs /opt/sandbox-test/binary /opt/sandbox-test/working
curl -L https://nl.alpinelinux.org/alpine/v3.5/releases/x86_64/alpine-minirootfs-3.5.2-x86_64.tar.gz | tar -xzvf - -C /opt/sandbox-test/rootfs
```
**(to be continued)**