2003-06-16 13:33:38 +00:00
|
|
|
#include <iostream>
|
|
|
|
|
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;
|
|
|
|
|
|
|
|
|
2003-06-27 14:56:12 +00:00
|
|
|
Error::Error(const format & f)
|
|
|
|
{
|
|
|
|
err = f.str();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
SysError::SysError(const format & f)
|
|
|
|
: Error(format("%1%: %2%") % f.str() % strerror(errno))
|
2003-06-16 13:33:38 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2003-05-26 13:45:00 +00:00
|
|
|
|
2003-06-16 13:33:38 +00:00
|
|
|
string absPath(string path, string dir)
|
2003-05-26 13:45:00 +00:00
|
|
|
{
|
2003-06-16 13:33:38 +00:00
|
|
|
if (path[0] != '/') {
|
2003-05-26 13:45:00 +00:00
|
|
|
if (dir == "") {
|
|
|
|
char buf[PATH_MAX];
|
|
|
|
if (!getcwd(buf, sizeof(buf)))
|
2003-06-16 13:33:38 +00:00
|
|
|
throw SysError("cannot get cwd");
|
2003-05-26 13:45:00 +00:00
|
|
|
dir = buf;
|
|
|
|
}
|
2003-06-16 13:33:38 +00:00
|
|
|
path = dir + "/" + path;
|
2003-05-26 13:45:00 +00:00
|
|
|
}
|
2003-07-07 09:25:26 +00:00
|
|
|
return canonPath(path);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
string canonPath(const string & path)
|
|
|
|
{
|
2003-07-08 19:58:41 +00:00
|
|
|
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-06-16 13:33:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
string dirOf(string 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-06-16 13:33:38 +00:00
|
|
|
return string(path, 0, pos);
|
2003-05-26 13:45:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-06-16 13:33:38 +00:00
|
|
|
string baseNameOf(string path)
|
2003-05-26 13:45:00 +00:00
|
|
|
{
|
2003-06-16 13:33:38 +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);
|
2003-06-16 13:33:38 +00:00
|
|
|
return string(path, pos + 1);
|
2003-05-26 13:45:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-07-08 13:22:08 +00:00
|
|
|
bool pathExists(const string & 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-08-22 20:12:44 +00:00
|
|
|
void deletePath(const string & path)
|
2003-06-23 14:40:49 +00:00
|
|
|
{
|
2003-08-08 14:55:56 +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))
|
2003-08-22 20:12:44 +00:00
|
|
|
throw SysError(format("getting attributes of path `%1%'") % path);
|
2003-06-23 14:40:49 +00:00
|
|
|
|
|
|
|
if (S_ISDIR(st.st_mode)) {
|
2003-08-08 14:55:56 +00:00
|
|
|
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;
|
2003-08-08 14:55:56 +00:00
|
|
|
names.push_back(name);
|
2003-06-23 14:40:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
closedir(dir); /* !!! close on exception */
|
2003-08-08 14:55:56 +00:00
|
|
|
|
2003-08-22 20:12:44 +00:00
|
|
|
/* 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"));
|
|
|
|
}
|
|
|
|
|
2003-08-08 14:55:56 +00:00
|
|
|
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)
|
2003-08-22 20:12:44 +00:00
|
|
|
throw SysError(format("cannot unlink `%1%'") % path);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void makePathReadOnly(const string & path)
|
|
|
|
{
|
|
|
|
struct stat st;
|
|
|
|
if (lstat(path.c_str(), &st))
|
|
|
|
throw SysError(format("getting attributes of path `%1%'") % path);
|
|
|
|
|
2003-08-28 10:51:14 +00:00
|
|
|
if (!S_ISLNK(st.st_mode) && (st.st_mode & S_IWUSR)) {
|
2003-08-22 20:12:44 +00:00
|
|
|
if (chmod(path.c_str(), st.st_mode & ~S_IWUSR) == -1)
|
2003-08-28 10:51:14 +00:00
|
|
|
throw SysError(format("making `%1%' read-only") % path);
|
2003-08-22 20:12:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
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-07-24 13:43:16 +00:00
|
|
|
Verbosity verbosity = lvlError;
|
2003-07-24 08:53:43 +00:00
|
|
|
|
2003-07-04 12:18:06 +00:00
|
|
|
static int nestingLevel = 0;
|
|
|
|
|
|
|
|
|
2003-07-24 08:53:43 +00:00
|
|
|
Nest::Nest(Verbosity level, const format & f)
|
2003-07-04 12:18:06 +00:00
|
|
|
{
|
2003-07-24 08:53:43 +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--;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-07-24 08:53:43 +00:00
|
|
|
void msg(Verbosity level, const format & f)
|
2003-07-04 12:18:06 +00:00
|
|
|
{
|
2003-07-24 08:53:43 +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
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-06-27 13:55:12 +00:00
|
|
|
void debug(const format & f)
|
2003-05-26 13:45:00 +00:00
|
|
|
{
|
2003-07-24 13:43:16 +00:00
|
|
|
msg(lvlDebug, f);
|
2003-05-26 13:45:00 +00:00
|
|
|
}
|
2003-07-20 21:11:43 +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;
|
|
|
|
}
|
|
|
|
}
|