lix-releng-staging/src/libnix/util.cc

254 lines
5.3 KiB
C++
Raw Normal View History

#include <iostream>
2003-09-11 08:31:29 +00:00
#include <cerrno>
#include <cstdio>
2003-06-23 14:40:49 +00:00
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <dirent.h>
2003-05-26 13:45:00 +00:00
#include "util.hh"
string thisSystem = SYSTEM;
Error::Error(const format & f)
{
err = f.str();
}
SysError::SysError(const format & f)
: Error(format("%1%: %2%") % f.str() % strerror(errno))
{
}
2003-05-26 13:45:00 +00:00
2003-10-07 14:37:41 +00:00
Path absPath(Path path, Path dir)
2003-05-26 13:45:00 +00:00
{
if (path[0] != '/') {
2003-05-26 13:45:00 +00:00
if (dir == "") {
char buf[PATH_MAX];
if (!getcwd(buf, sizeof(buf)))
throw SysError("cannot get cwd");
2003-05-26 13:45:00 +00:00
dir = buf;
}
path = dir + "/" + path;
2003-05-26 13:45:00 +00:00
}
return canonPath(path);
}
2003-10-07 14:37:41 +00:00
Path canonPath(const Path & path)
{
string s;
if (path[0] != '/')
throw Error(format("not an absolute path: `%1%'") % path);
string::const_iterator i = path.begin(), end = path.end();
while (1) {
/* Skip slashes. */
while (i != end && *i == '/') i++;
if (i == end) break;
/* Ignore `.'. */
if (*i == '.' && (i + 1 == end || i[1] == '/'))
i++;
/* If `..', delete the last component. */
else if (*i == '.' && i + 1 < end && i[1] == '.' &&
(i + 2 == end || i[2] == '/'))
{
if (!s.empty()) s.erase(s.rfind('/'));
i += 2;
}
/* Normal component; copy it. */
else {
s += '/';
while (i != end && *i != '/') s += *i++;
}
}
return s.empty() ? "/" : s;
}
2003-10-07 14:37:41 +00:00
Path dirOf(const Path & path)
{
unsigned int pos = path.rfind('/');
2003-07-04 12:18:06 +00:00
if (pos == string::npos)
throw Error(format("invalid file name: %1%") % path);
2003-10-07 14:37:41 +00:00
return Path(path, 0, pos);
2003-05-26 13:45:00 +00:00
}
2003-10-07 14:37:41 +00:00
string baseNameOf(const Path & path)
2003-05-26 13:45:00 +00:00
{
unsigned int pos = path.rfind('/');
2003-07-04 12:18:06 +00:00
if (pos == string::npos)
throw Error(format("invalid file name %1% ") % path);
return string(path, pos + 1);
2003-05-26 13:45:00 +00:00
}
2003-10-07 14:37:41 +00:00
bool pathExists(const Path & path)
{
int res;
struct stat st;
res = stat(path.c_str(), &st);
if (!res) return true;
if (errno != ENOENT)
throw SysError(format("getting status of %1%") % path);
return false;
}
2003-10-07 14:37:41 +00:00
void deletePath(const Path & path)
2003-06-23 14:40:49 +00:00
{
msg(lvlVomit, format("deleting path `%1%'") % path);
2003-06-23 14:40:49 +00:00
struct stat st;
if (lstat(path.c_str(), &st))
throw SysError(format("getting attributes of path `%1%'") % path);
2003-06-23 14:40:49 +00:00
if (S_ISDIR(st.st_mode)) {
Strings names;
2003-06-23 14:40:49 +00:00
DIR * dir = opendir(path.c_str());
struct dirent * dirent;
while (errno = 0, dirent = readdir(dir)) {
string name = dirent->d_name;
if (name == "." || name == "..") continue;
names.push_back(name);
2003-06-23 14:40:49 +00:00
}
closedir(dir); /* !!! close on exception */
/* Make the directory writable. */
if (!(st.st_mode & S_IWUSR)) {
if (chmod(path.c_str(), st.st_mode | S_IWUSR) == -1)
throw SysError(format("making `%1%' writable"));
}
for (Strings::iterator i = names.begin(); i != names.end(); i++)
deletePath(path + "/" + *i);
2003-06-23 14:40:49 +00:00
}
if (remove(path.c_str()) == -1)
throw SysError(format("cannot unlink `%1%'") % path);
}
2003-10-07 14:37:41 +00:00
void makePathReadOnly(const Path & path)
{
struct stat st;
if (lstat(path.c_str(), &st))
throw SysError(format("getting attributes of path `%1%'") % path);
if (!S_ISLNK(st.st_mode) && (st.st_mode & S_IWUSR)) {
if (chmod(path.c_str(), st.st_mode & ~S_IWUSR) == -1)
throw SysError(format("making `%1%' read-only") % path);
}
if (S_ISDIR(st.st_mode)) {
DIR * dir = opendir(path.c_str());
struct dirent * dirent;
while (errno = 0, dirent = readdir(dir)) {
string name = dirent->d_name;
if (name == "." || name == "..") continue;
makePathReadOnly(path + "/" + name);
}
closedir(dir); /* !!! close on exception */
}
2003-07-04 12:18:06 +00:00
}
2003-10-07 14:37:41 +00:00
static Path tempName()
{
static int counter = 0;
char * s = getenv("TMPDIR");
2003-10-07 14:37:41 +00:00
Path tmpRoot = s ? canonPath(Path(s)) : "/tmp";
return (format("%1%/nix-%2%-%3%") % tmpRoot % getpid() % counter++).str();
}
2003-10-07 14:37:41 +00:00
Path createTempDir()
{
while (1) {
2003-10-07 14:37:41 +00:00
Path tmpDir = tempName();
if (mkdir(tmpDir.c_str(), 0777) == 0) return tmpDir;
if (errno != EEXIST)
throw SysError(format("creating directory `%1%'") % tmpDir);
}
}
Verbosity verbosity = lvlError;
2003-07-04 12:18:06 +00:00
static int nestingLevel = 0;
Nest::Nest(Verbosity level, const format & f)
2003-07-04 12:18:06 +00:00
{
if (level > verbosity)
nest = false;
else {
msg(level, f);
nest = true;
nestingLevel++;
}
2003-07-04 12:18:06 +00:00
}
Nest::~Nest()
{
if (nest) nestingLevel--;
}
void msg(Verbosity level, const format & f)
2003-07-04 12:18:06 +00:00
{
if (level > verbosity) return;
2003-07-04 12:18:06 +00:00
string spaces;
for (int i = 0; i < nestingLevel; i++)
2003-07-20 19:29:38 +00:00
spaces += "| ";
2003-07-04 12:18:06 +00:00
cerr << format("%1%%2%\n") % spaces % f.str();
2003-06-23 14:40:49 +00:00
}
void debug(const format & f)
2003-05-26 13:45:00 +00:00
{
msg(lvlDebug, f);
2003-05-26 13:45:00 +00:00
}
void readFull(int fd, unsigned char * buf, size_t count)
{
while (count) {
ssize_t res = read(fd, (char *) buf, count);
if (res == -1) throw SysError("reading from file");
if (res == 0) throw Error("unexpected end-of-file");
count -= res;
buf += res;
}
}
void writeFull(int fd, const unsigned char * buf, size_t count)
{
while (count) {
ssize_t res = write(fd, (char *) buf, count);
if (res == -1) throw SysError("writing to file");
count -= res;
buf += res;
}
}