2022-03-17 14:28:08 +00:00
|
|
|
|
#include <sys/time.h>
|
2022-03-17 15:13:29 +00:00
|
|
|
|
#include <filesystem>
|
2022-03-17 14:28:08 +00:00
|
|
|
|
|
|
|
|
|
#include "util.hh"
|
|
|
|
|
#include "types.hh"
|
|
|
|
|
|
2022-03-17 15:13:29 +00:00
|
|
|
|
namespace fs = std::filesystem;
|
|
|
|
|
|
2022-03-17 14:28:08 +00:00
|
|
|
|
namespace nix {
|
|
|
|
|
|
|
|
|
|
void createSymlink(const Path & target, const Path & link,
|
|
|
|
|
std::optional<time_t> mtime)
|
|
|
|
|
{
|
|
|
|
|
if (symlink(target.c_str(), link.c_str()))
|
|
|
|
|
throw SysError("creating symlink from '%1%' to '%2%'", link, target);
|
|
|
|
|
if (mtime) {
|
|
|
|
|
struct timeval times[2];
|
|
|
|
|
times[0].tv_sec = *mtime;
|
|
|
|
|
times[0].tv_usec = 0;
|
|
|
|
|
times[1].tv_sec = *mtime;
|
|
|
|
|
times[1].tv_usec = 0;
|
|
|
|
|
if (lutimes(link.c_str(), times))
|
|
|
|
|
throw SysError("setting time of symlink '%s'", link);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void replaceSymlink(const Path & target, const Path & link,
|
|
|
|
|
std::optional<time_t> mtime)
|
|
|
|
|
{
|
|
|
|
|
for (unsigned int n = 0; true; n++) {
|
|
|
|
|
Path tmp = canonPath(fmt("%s/.%d_%s", dirOf(link), n, baseNameOf(link)));
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
createSymlink(target, tmp, mtime);
|
|
|
|
|
} catch (SysError & e) {
|
|
|
|
|
if (e.errNo == EEXIST) continue;
|
|
|
|
|
throw;
|
|
|
|
|
}
|
|
|
|
|
|
2022-04-13 12:10:36 +00:00
|
|
|
|
renameFile(tmp, link);
|
2022-03-17 14:28:08 +00:00
|
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-03-18 13:25:56 +00:00
|
|
|
|
void setWriteTime(const fs::path & p, const struct stat & st)
|
|
|
|
|
{
|
|
|
|
|
struct timeval times[2];
|
|
|
|
|
times[0] = {
|
|
|
|
|
.tv_sec = st.st_atime,
|
|
|
|
|
.tv_usec = 0,
|
|
|
|
|
};
|
|
|
|
|
times[1] = {
|
|
|
|
|
.tv_sec = st.st_mtime,
|
|
|
|
|
.tv_usec = 0,
|
|
|
|
|
};
|
|
|
|
|
if (lutimes(p.c_str(), times) != 0)
|
|
|
|
|
throw SysError("changing modification time of '%s'", p);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void copy(const fs::directory_entry & from, const fs::path & to, bool andDelete)
|
|
|
|
|
{
|
|
|
|
|
// TODO: Rewrite the `is_*` to use `symlink_status()`
|
|
|
|
|
auto statOfFrom = lstat(from.path().c_str());
|
|
|
|
|
auto fromStatus = from.symlink_status();
|
|
|
|
|
|
|
|
|
|
// Mark the directory as writable so that we can delete its children
|
|
|
|
|
if (andDelete && fs::is_directory(fromStatus)) {
|
|
|
|
|
fs::permissions(from.path(), fs::perms::owner_write, fs::perm_options::add | fs::perm_options::nofollow);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (fs::is_symlink(fromStatus) || fs::is_regular_file(fromStatus)) {
|
|
|
|
|
fs::copy(from.path(), to, fs::copy_options::copy_symlinks | fs::copy_options::overwrite_existing);
|
|
|
|
|
} else if (fs::is_directory(fromStatus)) {
|
|
|
|
|
fs::create_directory(to);
|
|
|
|
|
for (auto & entry : fs::directory_iterator(from.path())) {
|
|
|
|
|
copy(entry, to / entry.path().filename(), andDelete);
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
throw Error("file '%s' has an unsupported type", from.path());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
setWriteTime(to, statOfFrom);
|
|
|
|
|
if (andDelete) {
|
|
|
|
|
if (!fs::is_symlink(fromStatus))
|
|
|
|
|
fs::permissions(from.path(), fs::perms::owner_write, fs::perm_options::add | fs::perm_options::nofollow);
|
|
|
|
|
fs::remove(from.path());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-04-13 12:10:36 +00:00
|
|
|
|
void renameFile(const Path & oldName, const Path & newName)
|
2022-03-17 14:28:46 +00:00
|
|
|
|
{
|
2022-04-13 12:19:42 +00:00
|
|
|
|
fs::rename(oldName, newName);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void moveFile(const Path & oldName, const Path & newName)
|
|
|
|
|
{
|
2022-03-17 15:13:29 +00:00
|
|
|
|
try {
|
2022-04-13 12:19:42 +00:00
|
|
|
|
renameFile(oldName, newName);
|
2022-03-17 15:13:29 +00:00
|
|
|
|
} catch (fs::filesystem_error & e) {
|
2022-04-13 12:19:42 +00:00
|
|
|
|
auto oldPath = fs::path(oldName);
|
|
|
|
|
auto newPath = fs::path(newName);
|
2022-03-17 15:13:29 +00:00
|
|
|
|
if (e.code().value() == EXDEV) {
|
2022-03-18 13:25:56 +00:00
|
|
|
|
fs::remove(newPath);
|
2022-04-13 12:19:42 +00:00
|
|
|
|
warn("Can’t rename %s as %s, copying instead", oldName, newName);
|
2022-03-18 13:25:56 +00:00
|
|
|
|
copy(fs::directory_entry(oldPath), newPath, true);
|
2022-03-17 15:13:29 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2022-03-17 14:28:46 +00:00
|
|
|
|
}
|
|
|
|
|
|
2022-03-17 14:28:08 +00:00
|
|
|
|
}
|