2020-10-11 16:35:19 +00:00
|
|
|
#include "lock.hh"
|
2020-05-19 21:25:44 +00:00
|
|
|
#include "globals.hh"
|
|
|
|
#include "pathlocks.hh"
|
|
|
|
#include "cgroup.hh"
|
|
|
|
|
2020-10-17 17:25:17 +00:00
|
|
|
#include <pwd.h>
|
|
|
|
#include <grp.h>
|
|
|
|
|
2020-05-19 21:25:44 +00:00
|
|
|
namespace nix {
|
|
|
|
|
|
|
|
struct SimpleUserLock : UserLock
|
|
|
|
{
|
|
|
|
AutoCloseFD fdUserLock;
|
|
|
|
uid_t uid;
|
|
|
|
gid_t gid;
|
|
|
|
std::vector<gid_t> supplementaryGIDs;
|
|
|
|
|
|
|
|
void kill() override
|
|
|
|
{
|
|
|
|
killUser(uid);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::pair<uid_t, uid_t> getUIDRange() override
|
|
|
|
{
|
|
|
|
assert(uid);
|
|
|
|
return {uid, uid};
|
|
|
|
}
|
|
|
|
|
|
|
|
gid_t getGID() override { assert(gid); return gid; }
|
|
|
|
|
|
|
|
std::vector<gid_t> getSupplementaryGIDs() override { return supplementaryGIDs; }
|
|
|
|
|
|
|
|
static std::unique_ptr<UserLock> acquire()
|
|
|
|
{
|
|
|
|
assert(settings.buildUsersGroup != "");
|
|
|
|
createDirs(settings.nixStateDir + "/userpool");
|
|
|
|
|
|
|
|
/* Get the members of the build-users-group. */
|
|
|
|
struct group * gr = getgrnam(settings.buildUsersGroup.get().c_str());
|
|
|
|
if (!gr)
|
|
|
|
throw Error("the group '%s' specified in 'build-users-group' does not exist", settings.buildUsersGroup);
|
|
|
|
|
|
|
|
/* Copy the result of getgrnam. */
|
|
|
|
Strings users;
|
|
|
|
for (char * * p = gr->gr_mem; *p; ++p) {
|
|
|
|
debug("found build user '%s'", *p);
|
|
|
|
users.push_back(*p);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (users.empty())
|
|
|
|
throw Error("the build users group '%s' has no members", settings.buildUsersGroup);
|
|
|
|
|
|
|
|
/* Find a user account that isn't currently in use for another
|
|
|
|
build. */
|
|
|
|
for (auto & i : users) {
|
|
|
|
debug("trying user '%s'", i);
|
|
|
|
|
|
|
|
struct passwd * pw = getpwnam(i.c_str());
|
|
|
|
if (!pw)
|
|
|
|
throw Error("the user '%s' in the group '%s' does not exist", i, settings.buildUsersGroup);
|
|
|
|
|
|
|
|
auto fnUserLock = fmt("%s/userpool/%s", settings.nixStateDir,pw->pw_uid);
|
|
|
|
|
|
|
|
AutoCloseFD fd = open(fnUserLock.c_str(), O_RDWR | O_CREAT | O_CLOEXEC, 0600);
|
|
|
|
if (!fd)
|
|
|
|
throw SysError("opening user lock '%s'", fnUserLock);
|
|
|
|
|
|
|
|
if (lockFile(fd.get(), ltWrite, false)) {
|
|
|
|
auto lock = std::make_unique<SimpleUserLock>();
|
|
|
|
|
|
|
|
lock->fdUserLock = std::move(fd);
|
|
|
|
lock->uid = pw->pw_uid;
|
|
|
|
lock->gid = gr->gr_gid;
|
|
|
|
|
|
|
|
/* Sanity check... */
|
|
|
|
if (lock->uid == getuid() || lock->uid == geteuid())
|
|
|
|
throw Error("the Nix user should not be a member of '%s'", settings.buildUsersGroup);
|
|
|
|
|
|
|
|
#if __linux__
|
|
|
|
/* Get the list of supplementary groups of this build
|
|
|
|
user. This is usually either empty or contains a
|
|
|
|
group such as "kvm". */
|
2022-11-03 16:43:40 +00:00
|
|
|
int ngroups = 32; // arbitrary initial guess
|
|
|
|
lock->supplementaryGIDs.resize(ngroups);
|
|
|
|
|
|
|
|
int err = getgrouplist(
|
|
|
|
pw->pw_name, pw->pw_gid,
|
|
|
|
lock->supplementaryGIDs.data(),
|
|
|
|
&ngroups);
|
|
|
|
|
|
|
|
/* Our initial size of 32 wasn't sufficient, the
|
|
|
|
correct size has been stored in ngroups, so we try
|
|
|
|
again. */
|
|
|
|
if (err == -1) {
|
|
|
|
lock->supplementaryGIDs.resize(ngroups);
|
|
|
|
err = getgrouplist(
|
|
|
|
pw->pw_name, pw->pw_gid,
|
|
|
|
lock->supplementaryGIDs.data(),
|
|
|
|
&ngroups);
|
|
|
|
}
|
|
|
|
|
|
|
|
// If it failed once more, then something must be broken.
|
2020-05-19 21:25:44 +00:00
|
|
|
if (err == -1)
|
|
|
|
throw Error("failed to get list of supplementary groups for '%s'", pw->pw_name);
|
|
|
|
|
2022-11-03 16:43:40 +00:00
|
|
|
// Finally, trim back the GID list to its real size.
|
2020-05-19 21:25:44 +00:00
|
|
|
lock->supplementaryGIDs.resize(ngroups);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return lock;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
#if __linux__
|
|
|
|
struct CgroupUserLock : UserLock
|
|
|
|
{
|
|
|
|
AutoCloseFD fdUserLock;
|
|
|
|
uid_t uid;
|
|
|
|
|
|
|
|
void kill() override
|
|
|
|
{
|
|
|
|
if (cgroup) {
|
|
|
|
destroyCgroup(*cgroup);
|
|
|
|
cgroup.reset();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
std::pair<uid_t, uid_t> getUIDRange() override
|
|
|
|
{
|
|
|
|
assert(uid);
|
|
|
|
return {uid, uid + settings.idsPerBuild - 1};
|
|
|
|
}
|
|
|
|
|
|
|
|
gid_t getGID() override
|
|
|
|
{
|
|
|
|
// We use the same GID ranges as for the UIDs.
|
|
|
|
assert(uid);
|
|
|
|
return uid;
|
|
|
|
}
|
|
|
|
|
2020-05-20 09:57:33 +00:00
|
|
|
std::vector<gid_t> getSupplementaryGIDs() override { return {}; }
|
2020-05-19 21:25:44 +00:00
|
|
|
|
|
|
|
static std::unique_ptr<UserLock> acquire()
|
|
|
|
{
|
2022-02-28 23:54:20 +00:00
|
|
|
settings.requireExperimentalFeature(Xp::AutoAllocateUids);
|
2020-05-19 21:25:44 +00:00
|
|
|
assert(settings.startId > 0);
|
|
|
|
assert(settings.startId % settings.idsPerBuild == 0);
|
|
|
|
assert(settings.uidCount % settings.idsPerBuild == 0);
|
|
|
|
assert((uint64_t) settings.startId + (uint64_t) settings.uidCount <= std::numeric_limits<uid_t>::max());
|
|
|
|
|
|
|
|
// FIXME: check whether the id range overlaps any known users
|
|
|
|
|
|
|
|
createDirs(settings.nixStateDir + "/userpool2");
|
|
|
|
|
|
|
|
size_t nrSlots = settings.uidCount / settings.idsPerBuild;
|
|
|
|
|
|
|
|
for (size_t i = 0; i < nrSlots; i++) {
|
|
|
|
debug("trying user slot '%d'", i);
|
|
|
|
|
|
|
|
createDirs(settings.nixStateDir + "/userpool2");
|
|
|
|
|
|
|
|
auto fnUserLock = fmt("%s/userpool2/slot-%d", settings.nixStateDir, i);
|
|
|
|
|
|
|
|
AutoCloseFD fd = open(fnUserLock.c_str(), O_RDWR | O_CREAT | O_CLOEXEC, 0600);
|
|
|
|
if (!fd)
|
|
|
|
throw SysError("opening user lock '%s'", fnUserLock);
|
|
|
|
|
|
|
|
if (lockFile(fd.get(), ltWrite, false)) {
|
|
|
|
auto lock = std::make_unique<CgroupUserLock>();
|
|
|
|
lock->fdUserLock = std::move(fd);
|
|
|
|
lock->uid = settings.startId + i * settings.idsPerBuild;
|
|
|
|
auto s = drainFD(lock->fdUserLock.get());
|
|
|
|
if (s != "") lock->cgroup = s;
|
|
|
|
return lock;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::optional<Path> cgroup;
|
|
|
|
|
|
|
|
std::optional<Path> getCgroup() override
|
|
|
|
{
|
|
|
|
if (!cgroup) {
|
|
|
|
/* Create a systemd cgroup since that's the minimum
|
|
|
|
required by systemd-nspawn. */
|
|
|
|
auto ourCgroups = getCgroups("/proc/self/cgroup");
|
|
|
|
auto systemdCgroup = ourCgroups["systemd"];
|
|
|
|
if (systemdCgroup == "")
|
|
|
|
throw Error("'systemd' cgroup does not exist");
|
|
|
|
|
|
|
|
auto hostCgroup = canonPath("/sys/fs/cgroup/systemd/" + systemdCgroup);
|
|
|
|
|
|
|
|
if (!pathExists(hostCgroup))
|
|
|
|
throw Error("expected cgroup directory '%s'", hostCgroup);
|
|
|
|
|
|
|
|
cgroup = fmt("%s/nix-%d", hostCgroup, uid);
|
|
|
|
|
|
|
|
destroyCgroup(*cgroup);
|
|
|
|
|
|
|
|
if (mkdir(cgroup->c_str(), 0755) == -1)
|
|
|
|
throw SysError("creating cgroup '%s'", *cgroup);
|
|
|
|
|
|
|
|
/* Record the cgroup in the lock file. This ensures that
|
|
|
|
if we subsequently get executed under a different parent
|
|
|
|
cgroup, we kill the previous cgroup first. */
|
|
|
|
if (ftruncate(fdUserLock.get(), 0) == -1)
|
|
|
|
throw Error("truncating user lock");
|
|
|
|
writeFull(fdUserLock.get(), *cgroup);
|
|
|
|
}
|
|
|
|
|
|
|
|
return cgroup;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
#endif
|
|
|
|
|
|
|
|
std::unique_ptr<UserLock> acquireUserLock()
|
|
|
|
{
|
|
|
|
#if __linux__
|
|
|
|
if (settings.autoAllocateUids)
|
|
|
|
return CgroupUserLock::acquire();
|
|
|
|
else
|
|
|
|
#endif
|
|
|
|
return SimpleUserLock::acquire();
|
|
|
|
}
|
|
|
|
|
2020-05-20 09:24:21 +00:00
|
|
|
bool useBuildUsers()
|
|
|
|
{
|
|
|
|
#if __linux__
|
|
|
|
static bool b = (settings.buildUsersGroup != "" || settings.startId.get() != 0) && getuid() == 0;
|
|
|
|
return b;
|
|
|
|
#elif __APPLE__
|
|
|
|
static bool b = settings.buildUsersGroup != "" && getuid() == 0;
|
|
|
|
return b;
|
|
|
|
#else
|
|
|
|
return false;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2020-05-19 21:25:44 +00:00
|
|
|
}
|