2019-08-11 20:10:38 +02:00
|
|
|
#include <vector>
|
|
|
|
#include <initializer_list>
|
|
|
|
#include <string.h>
|
|
|
|
#include <sched.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <fstream>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <filesystem>
|
|
|
|
#include <sys/mount.h>
|
|
|
|
#include <sys/capability.h>
|
2020-09-26 17:03:26 +02:00
|
|
|
#include <qssb.h>
|
2019-08-11 20:10:38 +02:00
|
|
|
#include "../logger.h"
|
|
|
|
#include "../utils.h"
|
|
|
|
#include "../random.h"
|
|
|
|
|
|
|
|
#include "sandbox-linux.h"
|
|
|
|
|
|
|
|
/* TODO: make a whitelist approach. So far we simply blacklist
|
|
|
|
* obvious systemcalls. To whitelist, we need to analyse our
|
|
|
|
* dependencies (http library, sqlite wrapper, sqlite lib etc.) */
|
|
|
|
|
2020-09-26 17:03:26 +02:00
|
|
|
bool SandboxLinux::enableForInit()
|
2019-08-11 20:10:38 +02:00
|
|
|
{
|
2020-09-26 17:03:26 +02:00
|
|
|
umask(0027);
|
|
|
|
struct qssb_policy policy = {0};
|
|
|
|
int blacklisted_syscalls[] = {QSSB_SYS(execveat), QSSB_SYS(execve), -1};
|
|
|
|
policy.blacklisted_syscalls = blacklisted_syscalls;
|
|
|
|
policy.no_new_privs = 1;
|
|
|
|
int result = qssb_enable_policy(&policy);
|
|
|
|
if(result != 0)
|
2019-08-11 20:10:38 +02:00
|
|
|
{
|
2020-09-26 17:03:26 +02:00
|
|
|
Logger::error() << "Failed to install sandboxing policy (init): " << result;
|
2019-08-11 20:10:38 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-09-26 17:03:26 +02:00
|
|
|
bool SandboxLinux::enablePreWorker(std::vector<std::string> fsPaths)
|
2019-08-11 20:10:38 +02:00
|
|
|
{
|
2020-09-26 17:03:26 +02:00
|
|
|
std::sort(fsPaths.begin(), fsPaths.end(),
|
|
|
|
[](const std::string &a, const std::string &b) { return a.length() < b.length(); });
|
2019-08-11 20:10:38 +02:00
|
|
|
|
2020-09-26 17:03:26 +02:00
|
|
|
struct qssb_path_policy *policies = new qssb_path_policy[fsPaths.size()];
|
|
|
|
for(unsigned int i = 0; i < fsPaths.size(); i++)
|
2019-08-11 20:10:38 +02:00
|
|
|
{
|
2020-09-26 17:03:26 +02:00
|
|
|
policies[i].next = policies + (i + 1);
|
|
|
|
policies[i].mountpoint = fsPaths[i].c_str();
|
|
|
|
policies[i].policy = QSSB_MOUNT_ALLOW_READ | QSSB_MOUNT_ALLOW_WRITE;
|
2019-08-11 20:10:38 +02:00
|
|
|
}
|
2020-09-26 17:03:26 +02:00
|
|
|
policies[fsPaths.size() - 1].next = NULL;
|
2019-08-11 20:10:38 +02:00
|
|
|
|
2020-09-26 17:03:26 +02:00
|
|
|
struct qssb_policy policy = {0};
|
|
|
|
policy.path_policies = policies;
|
|
|
|
policy.namespace_options |= QSSB_UNSHARE_MOUNT;
|
|
|
|
policy.namespace_options |= QSSB_UNSHARE_USER;
|
|
|
|
int blacklisted_syscalls[] = {QSSB_SYS(execveat), QSSB_SYS(execve), -1};
|
|
|
|
policy.blacklisted_syscalls = blacklisted_syscalls;
|
|
|
|
int result = qssb_enable_policy(&policy);
|
|
|
|
if(result != 0)
|
2019-08-11 20:10:38 +02:00
|
|
|
{
|
2020-09-26 17:03:26 +02:00
|
|
|
Logger::error() << "Failed to install sandboxing policy (preworker): %i" << result;
|
2019-08-11 20:10:38 +02:00
|
|
|
return false;
|
|
|
|
}
|
2020-09-26 17:03:26 +02:00
|
|
|
delete[] policies;
|
2019-08-11 20:10:38 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SandboxLinux::supported()
|
|
|
|
{
|
2019-08-12 09:06:32 +02:00
|
|
|
std::fstream stream;
|
|
|
|
stream.open("/proc/sys/kernel/unprivileged_userns_clone");
|
|
|
|
if(stream.is_open())
|
|
|
|
{
|
|
|
|
std::string str;
|
|
|
|
stream >> str;
|
|
|
|
if(str[0] == '0')
|
|
|
|
{
|
2020-09-26 17:03:26 +02:00
|
|
|
Logger::error() << "Please write '1' to /proc/sys/kernel/unprivileged_userns_clone in order to enable "
|
|
|
|
"sandboxing support on this system";
|
2019-08-12 09:06:32 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2019-08-11 20:10:38 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
bool SandboxLinux::enableForWorker()
|
|
|
|
{
|
2020-09-26 17:03:26 +02:00
|
|
|
struct qssb_policy policy = {0};
|
|
|
|
policy.drop_caps = 1;
|
|
|
|
policy.not_dumpable = 1;
|
|
|
|
policy.no_new_privs = 1;
|
2019-08-11 20:10:38 +02:00
|
|
|
|
|
|
|
/* TODO: as said, a whitelist approach is better. As such, this list is bound to be incomplete in the
|
|
|
|
* sense that more could be listed here and some critical ones are probably missing */
|
2020-09-26 17:03:26 +02:00
|
|
|
int blacklisted_syscalls[] = {QSSB_SYS(setuid),
|
|
|
|
QSSB_SYS(connect),
|
|
|
|
QSSB_SYS(chroot),
|
|
|
|
QSSB_SYS(pivot_root),
|
|
|
|
QSSB_SYS(mount),
|
|
|
|
QSSB_SYS(setns),
|
|
|
|
QSSB_SYS(unshare),
|
|
|
|
QSSB_SYS(ptrace),
|
|
|
|
QSSB_SYS(personality),
|
|
|
|
QSSB_SYS(prctl),
|
|
|
|
-1};
|
|
|
|
policy.blacklisted_syscalls = blacklisted_syscalls;
|
|
|
|
if(qssb_enable_policy(&policy) != 0)
|
2019-08-11 20:10:38 +02:00
|
|
|
{
|
|
|
|
Logger::error() << "Sandbox: Activation of seccomp blacklist failed!";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|