2004-04-23 15:16:08 +00:00
|
|
|
#include "config.h"
|
|
|
|
|
2006-08-04 16:01:26 +00:00
|
|
|
#ifdef __CYGWIN__
|
|
|
|
#include <windows.h>
|
|
|
|
#endif
|
|
|
|
|
2003-06-16 13:33:38 +00:00
|
|
|
#include <iostream>
|
2003-09-11 08:31:29 +00:00
|
|
|
#include <cerrno>
|
|
|
|
#include <cstdio>
|
2004-03-22 20:53:49 +00:00
|
|
|
#include <sstream>
|
2003-06-16 13:33:38 +00:00
|
|
|
|
2003-06-23 14:40:49 +00:00
|
|
|
#include <sys/stat.h>
|
2004-06-22 09:51:44 +00:00
|
|
|
#include <sys/wait.h>
|
2006-09-27 21:04:07 +00:00
|
|
|
#include <sys/types.h>
|
2006-12-02 15:45:51 +00:00
|
|
|
#include <fcntl.h>
|
2006-09-27 21:04:07 +00:00
|
|
|
|
2003-05-26 13:45:00 +00:00
|
|
|
#include "util.hh"
|
|
|
|
|
|
|
|
|
2006-12-07 16:40:41 +00:00
|
|
|
extern char * * environ;
|
|
|
|
|
|
|
|
|
2006-09-04 21:06:23 +00:00
|
|
|
namespace nix {
|
|
|
|
|
|
|
|
|
2003-06-27 14:56:12 +00:00
|
|
|
Error::Error(const format & f)
|
|
|
|
{
|
|
|
|
err = f.str();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-03-08 14:11:19 +00:00
|
|
|
Error & Error::addPrefix(const format & f)
|
|
|
|
{
|
|
|
|
err = f.str() + err;
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-06-27 14:56:12 +00:00
|
|
|
SysError::SysError(const format & f)
|
|
|
|
: Error(format("%1%: %2%") % f.str() % strerror(errno))
|
2006-12-05 01:31:45 +00:00
|
|
|
, errNo(errno)
|
2003-06-16 13:33:38 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2003-05-26 13:45:00 +00:00
|
|
|
|
2004-05-12 09:35:51 +00:00
|
|
|
string getEnv(const string & key, const string & def)
|
|
|
|
{
|
|
|
|
char * value = getenv(key.c_str());
|
|
|
|
return value ? string(value) : def;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-10-07 14:37:41 +00:00
|
|
|
Path absPath(Path path, Path 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);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-01-08 17:16:03 +00:00
|
|
|
Path canonPath(const Path & path, bool resolveSymlinks)
|
2003-07-07 09:25:26 +00:00
|
|
|
{
|
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();
|
2006-01-08 17:16:03 +00:00
|
|
|
string temp;
|
|
|
|
|
|
|
|
/* Count the number of times we follow a symlink and stop at some
|
|
|
|
arbitrary (but high) limit to prevent infinite loops. */
|
|
|
|
unsigned int followCount = 0, maxFollow = 1024;
|
2003-07-08 19:58:41 +00:00
|
|
|
|
|
|
|
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++;
|
2006-01-08 17:16:03 +00:00
|
|
|
|
|
|
|
/* If s points to a symlink, resolve it and restart (since
|
|
|
|
the symlink target might contain new symlinks). */
|
|
|
|
if (resolveSymlinks && isLink(s)) {
|
|
|
|
followCount++;
|
|
|
|
if (followCount >= maxFollow)
|
|
|
|
throw Error(format("infinite symlink recursion in path `%1%'") % path);
|
|
|
|
temp = absPath(readLink(s), dirOf(s))
|
|
|
|
+ string(i, end);
|
|
|
|
i = temp.begin(); /* restart */
|
|
|
|
end = temp.end();
|
|
|
|
s = "";
|
|
|
|
/* !!! potential for infinite loop */
|
|
|
|
}
|
2003-07-08 19:58:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return s.empty() ? "/" : s;
|
2003-06-16 13:33:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-10-07 14:37:41 +00:00
|
|
|
Path dirOf(const Path & path)
|
2003-06-16 13:33:38 +00:00
|
|
|
{
|
2006-05-11 02:19:43 +00:00
|
|
|
Path::size_type pos = path.rfind('/');
|
2003-07-04 12:18:06 +00:00
|
|
|
if (pos == string::npos)
|
2006-12-02 16:41:36 +00:00
|
|
|
throw Error(format("invalid file name `%1%'") % path);
|
2006-01-09 14:52:46 +00:00
|
|
|
return pos == 0 ? "/" : 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
|
|
|
{
|
2006-05-11 02:19:43 +00:00
|
|
|
Path::size_type pos = path.rfind('/');
|
2003-07-04 12:18:06 +00:00
|
|
|
if (pos == string::npos)
|
2006-12-02 16:41:36 +00:00
|
|
|
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-10-07 14:37:41 +00:00
|
|
|
bool pathExists(const Path & path)
|
2003-07-08 13:22:08 +00:00
|
|
|
{
|
|
|
|
int res;
|
|
|
|
struct stat st;
|
2004-02-06 10:59:06 +00:00
|
|
|
res = lstat(path.c_str(), &st);
|
2003-07-08 13:22:08 +00:00
|
|
|
if (!res) return true;
|
2004-08-04 09:25:21 +00:00
|
|
|
if (errno != ENOENT && errno != ENOTDIR)
|
2003-07-08 13:22:08 +00:00
|
|
|
throw SysError(format("getting status of %1%") % path);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-01-05 16:26:43 +00:00
|
|
|
Path readLink(const Path & path)
|
|
|
|
{
|
|
|
|
struct stat st;
|
|
|
|
if (lstat(path.c_str(), &st))
|
|
|
|
throw SysError(format("getting status of `%1%'") % path);
|
|
|
|
if (!S_ISLNK(st.st_mode))
|
|
|
|
throw Error(format("`%1%' is not a symlink") % path);
|
|
|
|
char buf[st.st_size];
|
|
|
|
if (readlink(path.c_str(), buf, st.st_size) != st.st_size)
|
|
|
|
throw SysError(format("reading symbolic link `%1%'") % path);
|
|
|
|
return string(buf, st.st_size);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-02-01 13:48:46 +00:00
|
|
|
bool isLink(const Path & path)
|
|
|
|
{
|
|
|
|
struct stat st;
|
|
|
|
if (lstat(path.c_str(), &st))
|
|
|
|
throw SysError(format("getting status of `%1%'") % path);
|
|
|
|
return S_ISLNK(st.st_mode);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-11-19 17:27:16 +00:00
|
|
|
Strings readDirectory(const Path & path)
|
|
|
|
{
|
|
|
|
Strings names;
|
|
|
|
|
|
|
|
AutoCloseDir dir = opendir(path.c_str());
|
|
|
|
if (!dir) throw SysError(format("opening directory `%1%'") % path);
|
|
|
|
|
|
|
|
struct dirent * dirent;
|
|
|
|
while (errno = 0, dirent = readdir(dir)) { /* sic */
|
2004-01-15 20:23:55 +00:00
|
|
|
checkInterrupt();
|
2003-11-19 17:27:16 +00:00
|
|
|
string name = dirent->d_name;
|
|
|
|
if (name == "." || name == "..") continue;
|
|
|
|
names.push_back(name);
|
|
|
|
}
|
|
|
|
if (errno) throw SysError(format("reading directory `%1%'") % path);
|
|
|
|
|
|
|
|
return names;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-02-01 22:07:48 +00:00
|
|
|
string readFile(int fd)
|
|
|
|
{
|
|
|
|
struct stat st;
|
|
|
|
if (fstat(fd, &st) == -1)
|
|
|
|
throw SysError("statting file");
|
2006-10-30 11:56:09 +00:00
|
|
|
|
|
|
|
unsigned char * buf = new unsigned char[st.st_size];
|
|
|
|
AutoDeleteArray<unsigned char> d(buf);
|
2005-02-01 22:07:48 +00:00
|
|
|
readFull(fd, buf, st.st_size);
|
|
|
|
|
|
|
|
return string((char *) buf, st.st_size);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
string readFile(const Path & path)
|
|
|
|
{
|
|
|
|
AutoCloseFD fd = open(path.c_str(), O_RDONLY);
|
|
|
|
if (fd == -1)
|
|
|
|
throw SysError(format("opening file `%1%'") % path);
|
|
|
|
return readFile(fd);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-02-09 09:50:29 +00:00
|
|
|
void writeFile(const Path & path, const string & s)
|
|
|
|
{
|
|
|
|
AutoCloseFD fd = open(path.c_str(), O_WRONLY | O_TRUNC | O_CREAT, 0666);
|
|
|
|
if (fd == -1)
|
|
|
|
throw SysError(format("opening file `%1%'") % path);
|
|
|
|
writeFull(fd, (unsigned char *) s.c_str(), s.size());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-11-24 20:24:14 +00:00
|
|
|
unsigned long long computePathSize(const Path & path)
|
|
|
|
{
|
|
|
|
unsigned long long size = 0;
|
|
|
|
|
|
|
|
checkInterrupt();
|
|
|
|
|
|
|
|
struct stat st;
|
|
|
|
if (lstat(path.c_str(), &st))
|
|
|
|
throw SysError(format("getting attributes of path `%1%'") % path);
|
|
|
|
|
|
|
|
size += st.st_size;
|
|
|
|
|
|
|
|
if (S_ISDIR(st.st_mode)) {
|
|
|
|
Strings names = readDirectory(path);
|
|
|
|
|
|
|
|
for (Strings::iterator i = names.begin(); i != names.end(); ++i)
|
|
|
|
size += computePathSize(path + "/" + *i);
|
|
|
|
}
|
|
|
|
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-12-15 21:11:39 +00:00
|
|
|
static void _deletePath(const Path & path, unsigned long long & bytesFreed)
|
2003-06-23 14:40:49 +00:00
|
|
|
{
|
2004-01-15 20:23:55 +00:00
|
|
|
checkInterrupt();
|
|
|
|
|
2004-03-22 21:42:28 +00:00
|
|
|
printMsg(lvlVomit, format("%1%") % path);
|
2003-08-08 14:55:56 +00:00
|
|
|
|
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
|
|
|
|
2005-12-15 21:11:39 +00:00
|
|
|
bytesFreed += st.st_size;
|
|
|
|
|
2003-06-23 14:40:49 +00:00
|
|
|
if (S_ISDIR(st.st_mode)) {
|
2003-11-19 17:27:16 +00:00
|
|
|
Strings names = readDirectory(path);
|
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)
|
2004-11-08 15:20:52 +00:00
|
|
|
throw SysError(format("making `%1%' writable") % path);
|
2003-08-22 20:12:44 +00:00
|
|
|
}
|
|
|
|
|
2003-11-19 17:27:16 +00:00
|
|
|
for (Strings::iterator i = names.begin(); i != names.end(); ++i)
|
2005-12-15 21:11:39 +00:00
|
|
|
_deletePath(path + "/" + *i, bytesFreed);
|
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);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-22 21:42:28 +00:00
|
|
|
void deletePath(const Path & path)
|
2005-12-15 21:11:39 +00:00
|
|
|
{
|
|
|
|
unsigned long long dummy;
|
|
|
|
deletePath(path, dummy);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void deletePath(const Path & path, unsigned long long & bytesFreed)
|
2004-03-22 21:42:28 +00:00
|
|
|
{
|
|
|
|
startNest(nest, lvlDebug,
|
|
|
|
format("recursively deleting path `%1%'") % path);
|
2005-12-15 21:11:39 +00:00
|
|
|
bytesFreed = 0;
|
|
|
|
_deletePath(path, bytesFreed);
|
2004-03-22 21:42:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-10-07 14:37:41 +00:00
|
|
|
void makePathReadOnly(const Path & path)
|
2003-08-22 20:12:44 +00:00
|
|
|
{
|
2004-01-15 20:23:55 +00:00
|
|
|
checkInterrupt();
|
|
|
|
|
2003-08-22 20:12:44 +00:00
|
|
|
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)) {
|
2003-11-19 17:27:16 +00:00
|
|
|
Strings names = readDirectory(path);
|
|
|
|
for (Strings::iterator i = names.begin(); i != names.end(); ++i)
|
|
|
|
makePathReadOnly(path + "/" + *i);
|
2003-08-22 20:12:44 +00:00
|
|
|
}
|
2003-07-04 12:18:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-10-07 14:37:41 +00:00
|
|
|
static Path tempName()
|
2003-10-02 11:55:38 +00:00
|
|
|
{
|
|
|
|
static int counter = 0;
|
2006-01-08 17:16:03 +00:00
|
|
|
Path tmpRoot = canonPath(getEnv("TMPDIR", "/tmp"), true);
|
2003-10-02 11:55:38 +00:00
|
|
|
return (format("%1%/nix-%2%-%3%") % tmpRoot % getpid() % counter++).str();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-10-07 14:37:41 +00:00
|
|
|
Path createTempDir()
|
2003-10-02 11:55:38 +00:00
|
|
|
{
|
|
|
|
while (1) {
|
2004-01-15 20:23:55 +00:00
|
|
|
checkInterrupt();
|
2003-10-07 14:37:41 +00:00
|
|
|
Path tmpDir = tempName();
|
2006-06-14 11:53:55 +00:00
|
|
|
if (mkdir(tmpDir.c_str(), 0777) == 0) {
|
|
|
|
/* Explicitly set the group of the directory. This is to
|
|
|
|
work around around problems caused by BSD's group
|
|
|
|
ownership semantics (directories inherit the group of
|
|
|
|
the parent). For instance, the group of /tmp on
|
|
|
|
FreeBSD is "wheel", so all directories created in /tmp
|
|
|
|
will be owned by "wheel"; but if the user is not in
|
|
|
|
"wheel", then "tar" will fail to unpack archives that
|
|
|
|
have the setgid bit set on directories. */
|
|
|
|
if (chown(tmpDir.c_str(), (uid_t) -1, getegid()) != 0)
|
|
|
|
throw SysError(format("setting group of directory `%1%'") % tmpDir);
|
|
|
|
return tmpDir;
|
|
|
|
}
|
2003-10-02 11:55:38 +00:00
|
|
|
if (errno != EEXIST)
|
|
|
|
throw SysError(format("creating directory `%1%'") % tmpDir);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-03-24 17:46:38 +00:00
|
|
|
void createDirs(const Path & path)
|
|
|
|
{
|
2006-01-09 14:52:46 +00:00
|
|
|
if (path == "/") return;
|
2005-03-24 17:46:38 +00:00
|
|
|
createDirs(dirOf(path));
|
|
|
|
if (!pathExists(path))
|
|
|
|
if (mkdir(path.c_str(), 0777) == -1)
|
|
|
|
throw SysError(format("creating directory `%1%'") % path);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-11-22 15:58:34 +00:00
|
|
|
void writeStringToFile(const Path & path, const string & s)
|
|
|
|
{
|
2005-01-31 10:27:25 +00:00
|
|
|
AutoCloseFD fd(open(path.c_str(),
|
|
|
|
O_CREAT | O_EXCL | O_WRONLY, 0666));
|
2003-11-22 15:58:34 +00:00
|
|
|
if (fd == -1)
|
|
|
|
throw SysError(format("creating file `%1%'") % path);
|
|
|
|
writeFull(fd, (unsigned char *) s.c_str(), s.size());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-22 20:53:49 +00:00
|
|
|
LogType logType = ltPretty;
|
2004-08-18 12:19:06 +00:00
|
|
|
Verbosity verbosity = lvlInfo;
|
2003-07-24 08:53:43 +00:00
|
|
|
|
2003-07-04 12:18:06 +00:00
|
|
|
static int nestingLevel = 0;
|
|
|
|
|
|
|
|
|
2003-11-09 10:35:45 +00:00
|
|
|
Nest::Nest()
|
2003-07-04 12:18:06 +00:00
|
|
|
{
|
2003-11-09 10:35:45 +00:00
|
|
|
nest = false;
|
2003-07-04 12:18:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Nest::~Nest()
|
|
|
|
{
|
2004-03-22 21:42:28 +00:00
|
|
|
close();
|
2004-03-22 20:53:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static string escVerbosity(Verbosity level)
|
|
|
|
{
|
2006-08-26 16:48:01 +00:00
|
|
|
return int2String((int) level);
|
2003-07-04 12:18:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-11-09 10:35:45 +00:00
|
|
|
void Nest::open(Verbosity level, const format & f)
|
|
|
|
{
|
|
|
|
if (level <= verbosity) {
|
2004-03-22 20:53:49 +00:00
|
|
|
if (logType == ltEscapes)
|
2006-09-04 21:06:23 +00:00
|
|
|
std::cerr << "\033[" << escVerbosity(level) << "p"
|
|
|
|
<< f.str() << "\n";
|
2004-03-22 21:42:28 +00:00
|
|
|
else
|
|
|
|
printMsg_(level, f);
|
2003-11-09 10:35:45 +00:00
|
|
|
nest = true;
|
|
|
|
nestingLevel++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-22 21:42:28 +00:00
|
|
|
void Nest::close()
|
|
|
|
{
|
|
|
|
if (nest) {
|
|
|
|
nestingLevel--;
|
|
|
|
if (logType == ltEscapes)
|
2006-09-04 21:06:23 +00:00
|
|
|
std::cerr << "\033[q";
|
2004-03-27 15:33:19 +00:00
|
|
|
nest = false;
|
2004-03-22 21:42:28 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-11-09 10:35:45 +00:00
|
|
|
void printMsg_(Verbosity level, const format & f)
|
2003-07-04 12:18:06 +00:00
|
|
|
{
|
2004-01-15 20:23:55 +00:00
|
|
|
checkInterrupt();
|
2003-07-24 08:53:43 +00:00
|
|
|
if (level > verbosity) return;
|
2004-03-22 20:53:49 +00:00
|
|
|
string prefix;
|
|
|
|
if (logType == ltPretty)
|
|
|
|
for (int i = 0; i < nestingLevel; i++)
|
|
|
|
prefix += "| ";
|
|
|
|
else if (logType == ltEscapes && level != lvlInfo)
|
|
|
|
prefix = "\033[" + escVerbosity(level) + "s";
|
2006-06-19 14:37:35 +00:00
|
|
|
string s = (format("%1%%2%\n") % prefix % f.str()).str();
|
2006-12-03 02:08:13 +00:00
|
|
|
writeToStderr((const unsigned char *) s.c_str(), s.size());
|
2003-06-23 14:40:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-08-29 15:29:38 +00:00
|
|
|
void warnOnce(bool & haveWarned, const format & f)
|
|
|
|
{
|
|
|
|
if (!haveWarned) {
|
|
|
|
printMsg(lvlError, format("warning: %1%") % f.str());
|
|
|
|
haveWarned = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-12-03 02:08:13 +00:00
|
|
|
static void defaultWriteToStderr(const unsigned char * buf, size_t count)
|
|
|
|
{
|
|
|
|
writeFull(STDERR_FILENO, buf, count);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void (*writeToStderr) (const unsigned char * buf, size_t count) = defaultWriteToStderr;
|
|
|
|
|
|
|
|
|
2003-07-20 21:11:43 +00:00
|
|
|
void readFull(int fd, unsigned char * buf, size_t count)
|
|
|
|
{
|
|
|
|
while (count) {
|
2004-01-15 20:23:55 +00:00
|
|
|
checkInterrupt();
|
2003-07-20 21:11:43 +00:00
|
|
|
ssize_t res = read(fd, (char *) buf, count);
|
2004-05-11 13:48:25 +00:00
|
|
|
if (res == -1) {
|
|
|
|
if (errno == EINTR) continue;
|
|
|
|
throw SysError("reading from file");
|
|
|
|
}
|
2006-12-04 17:17:13 +00:00
|
|
|
if (res == 0) throw EndOfFile("unexpected end-of-file");
|
2003-07-20 21:11:43 +00:00
|
|
|
count -= res;
|
|
|
|
buf += res;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void writeFull(int fd, const unsigned char * buf, size_t count)
|
|
|
|
{
|
|
|
|
while (count) {
|
2004-01-15 20:23:55 +00:00
|
|
|
checkInterrupt();
|
2003-07-20 21:11:43 +00:00
|
|
|
ssize_t res = write(fd, (char *) buf, count);
|
2004-05-11 13:48:25 +00:00
|
|
|
if (res == -1) {
|
|
|
|
if (errno == EINTR) continue;
|
|
|
|
throw SysError("writing to file");
|
|
|
|
}
|
2003-07-20 21:11:43 +00:00
|
|
|
count -= res;
|
|
|
|
buf += res;
|
|
|
|
}
|
|
|
|
}
|
2003-10-22 10:48:22 +00:00
|
|
|
|
|
|
|
|
2006-07-20 12:17:25 +00:00
|
|
|
string drainFD(int fd)
|
|
|
|
{
|
|
|
|
string result;
|
|
|
|
unsigned char buffer[4096];
|
|
|
|
while (1) {
|
|
|
|
ssize_t rd = read(fd, buffer, sizeof buffer);
|
|
|
|
if (rd == -1) {
|
|
|
|
if (errno != EINTR)
|
|
|
|
throw SysError("reading from file");
|
|
|
|
}
|
|
|
|
else if (rd == 0) break;
|
|
|
|
else result.append((char *) buffer, rd);
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-06-22 09:51:44 +00:00
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
2003-10-22 10:48:22 +00:00
|
|
|
AutoDelete::AutoDelete(const string & p) : path(p)
|
|
|
|
{
|
|
|
|
del = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
AutoDelete::~AutoDelete()
|
|
|
|
{
|
|
|
|
if (del) deletePath(path);
|
|
|
|
}
|
|
|
|
|
|
|
|
void AutoDelete::cancel()
|
|
|
|
{
|
|
|
|
del = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-06-22 09:51:44 +00:00
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
2003-10-22 10:48:22 +00:00
|
|
|
AutoCloseFD::AutoCloseFD()
|
|
|
|
{
|
|
|
|
fd = -1;
|
|
|
|
}
|
|
|
|
|
2004-06-22 09:51:44 +00:00
|
|
|
|
2003-10-22 10:48:22 +00:00
|
|
|
AutoCloseFD::AutoCloseFD(int fd)
|
|
|
|
{
|
|
|
|
this->fd = fd;
|
|
|
|
}
|
|
|
|
|
2004-06-22 09:51:44 +00:00
|
|
|
|
2005-01-31 10:27:25 +00:00
|
|
|
AutoCloseFD::AutoCloseFD(const AutoCloseFD & fd)
|
|
|
|
{
|
|
|
|
abort();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-10-22 10:48:22 +00:00
|
|
|
AutoCloseFD::~AutoCloseFD()
|
|
|
|
{
|
2004-06-15 13:49:42 +00:00
|
|
|
try {
|
|
|
|
close();
|
|
|
|
} catch (Error & e) {
|
|
|
|
printMsg(lvlError, format("error (ignored): %1%") % e.msg());
|
|
|
|
}
|
2003-10-22 10:48:22 +00:00
|
|
|
}
|
|
|
|
|
2004-06-22 09:51:44 +00:00
|
|
|
|
2003-10-22 10:48:22 +00:00
|
|
|
void AutoCloseFD::operator =(int fd)
|
|
|
|
{
|
2004-06-15 13:49:42 +00:00
|
|
|
if (this->fd != fd) close();
|
2003-10-22 10:48:22 +00:00
|
|
|
this->fd = fd;
|
|
|
|
}
|
|
|
|
|
2004-06-22 09:51:44 +00:00
|
|
|
|
2005-01-31 10:27:25 +00:00
|
|
|
AutoCloseFD::operator int() const
|
2003-10-22 10:48:22 +00:00
|
|
|
{
|
|
|
|
return fd;
|
|
|
|
}
|
|
|
|
|
2004-06-22 09:51:44 +00:00
|
|
|
|
2004-06-15 13:49:42 +00:00
|
|
|
void AutoCloseFD::close()
|
|
|
|
{
|
|
|
|
if (fd != -1) {
|
|
|
|
if (::close(fd) == -1)
|
|
|
|
/* This should never happen. */
|
|
|
|
throw SysError("closing file descriptor");
|
|
|
|
fd = -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-06-22 09:51:44 +00:00
|
|
|
|
2004-06-15 13:49:42 +00:00
|
|
|
bool AutoCloseFD::isOpen()
|
|
|
|
{
|
|
|
|
return fd != -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-01-27 12:19:25 +00:00
|
|
|
/* Pass responsibility for closing this fd to the caller. */
|
|
|
|
int AutoCloseFD::borrow()
|
|
|
|
{
|
|
|
|
int oldFD = fd;
|
|
|
|
fd = -1;
|
|
|
|
return oldFD;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-06-15 13:49:42 +00:00
|
|
|
void Pipe::create()
|
|
|
|
{
|
|
|
|
int fds[2];
|
|
|
|
if (pipe(fds) != 0) throw SysError("creating pipe");
|
|
|
|
readSide = fds[0];
|
|
|
|
writeSide = fds[1];
|
|
|
|
}
|
|
|
|
|
2003-10-22 10:48:22 +00:00
|
|
|
|
2004-06-22 09:51:44 +00:00
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
2003-10-22 10:48:22 +00:00
|
|
|
AutoCloseDir::AutoCloseDir()
|
|
|
|
{
|
|
|
|
dir = 0;
|
|
|
|
}
|
|
|
|
|
2004-06-22 09:51:44 +00:00
|
|
|
|
2003-10-22 10:48:22 +00:00
|
|
|
AutoCloseDir::AutoCloseDir(DIR * dir)
|
|
|
|
{
|
|
|
|
this->dir = dir;
|
|
|
|
}
|
|
|
|
|
2004-06-22 09:51:44 +00:00
|
|
|
|
2003-10-22 10:48:22 +00:00
|
|
|
AutoCloseDir::~AutoCloseDir()
|
|
|
|
{
|
|
|
|
if (dir) closedir(dir);
|
|
|
|
}
|
|
|
|
|
2004-06-22 09:51:44 +00:00
|
|
|
|
2003-10-22 10:48:22 +00:00
|
|
|
void AutoCloseDir::operator =(DIR * dir)
|
|
|
|
{
|
|
|
|
this->dir = dir;
|
|
|
|
}
|
|
|
|
|
2004-06-22 09:51:44 +00:00
|
|
|
|
2003-10-22 10:48:22 +00:00
|
|
|
AutoCloseDir::operator DIR *()
|
|
|
|
{
|
|
|
|
return dir;
|
|
|
|
}
|
|
|
|
|
2004-01-15 20:23:55 +00:00
|
|
|
|
2004-06-22 09:51:44 +00:00
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
|
|
|
Pid::Pid()
|
|
|
|
{
|
|
|
|
pid = -1;
|
|
|
|
separatePG = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Pid::~Pid()
|
|
|
|
{
|
|
|
|
kill();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Pid::operator =(pid_t pid)
|
|
|
|
{
|
|
|
|
if (this->pid != pid) kill();
|
|
|
|
this->pid = pid;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Pid::operator pid_t()
|
|
|
|
{
|
|
|
|
return pid;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Pid::kill()
|
|
|
|
{
|
|
|
|
if (pid == -1) return;
|
|
|
|
|
2004-06-25 15:36:09 +00:00
|
|
|
printMsg(lvlError, format("killing process %1%") % pid);
|
2004-06-22 09:51:44 +00:00
|
|
|
|
|
|
|
/* Send a KILL signal to the child. If it has its own process
|
|
|
|
group, send the signal to every process in the child process
|
|
|
|
group (which hopefully includes *all* its children). */
|
|
|
|
if (::kill(separatePG ? -pid : pid, SIGKILL) != 0)
|
2004-06-25 15:36:09 +00:00
|
|
|
printMsg(lvlError, (SysError(format("killing process %1%") % pid).msg()));
|
|
|
|
|
|
|
|
/* Wait until the child dies, disregarding the exit status. */
|
|
|
|
int status;
|
2006-12-03 02:12:26 +00:00
|
|
|
while (waitpid(pid, &status, 0) == -1) {
|
|
|
|
checkInterrupt();
|
2004-06-25 15:36:09 +00:00
|
|
|
if (errno != EINTR) printMsg(lvlError,
|
|
|
|
(SysError(format("waiting for process %1%") % pid).msg()));
|
2006-12-03 02:12:26 +00:00
|
|
|
}
|
2004-06-22 09:51:44 +00:00
|
|
|
|
|
|
|
pid = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int Pid::wait(bool block)
|
|
|
|
{
|
|
|
|
while (1) {
|
|
|
|
int status;
|
|
|
|
int res = waitpid(pid, &status, block ? 0 : WNOHANG);
|
|
|
|
if (res == pid) {
|
|
|
|
pid = -1;
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
if (res == 0 && !block) return -1;
|
|
|
|
if (errno != EINTR)
|
|
|
|
throw SysError("cannot get child exit status");
|
2006-12-04 17:17:13 +00:00
|
|
|
checkInterrupt();
|
2004-06-22 09:51:44 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Pid::setSeparatePG(bool separatePG)
|
|
|
|
{
|
|
|
|
this->separatePG = separatePG;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-12-07 00:16:07 +00:00
|
|
|
void killUser(uid_t uid)
|
|
|
|
{
|
|
|
|
debug(format("killing all processes running under uid `%1%'") % uid);
|
|
|
|
|
|
|
|
assert(uid != 0); /* just to be safe... */
|
|
|
|
|
|
|
|
/* The system call kill(-1, sig) sends the signal `sig' to all
|
|
|
|
users to which the current process can send signals. So we
|
|
|
|
fork a process, switch to uid, and send a mass kill. */
|
|
|
|
|
|
|
|
Pid pid;
|
|
|
|
pid = fork();
|
|
|
|
switch (pid) {
|
|
|
|
|
|
|
|
case -1:
|
|
|
|
throw SysError("unable to fork");
|
|
|
|
|
|
|
|
case 0:
|
|
|
|
try { /* child */
|
|
|
|
|
|
|
|
if (setuid(uid) == -1) abort();
|
|
|
|
|
|
|
|
while (true) {
|
|
|
|
if (kill(-1, SIGKILL) == 0) break;
|
|
|
|
if (errno == ESRCH) break; /* no more processes */
|
|
|
|
if (errno != EINTR)
|
|
|
|
throw SysError(format("cannot kill processes for uid `%1%'") % uid);
|
|
|
|
}
|
|
|
|
|
|
|
|
} catch (std::exception & e) {
|
|
|
|
std::cerr << format("killing processes beloging to uid `%1%': %1%\n")
|
|
|
|
% uid % e.what();
|
|
|
|
quickExit(1);
|
|
|
|
}
|
|
|
|
quickExit(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* parent */
|
|
|
|
if (pid.wait(true) != 0)
|
|
|
|
throw Error(format("cannot kill processes for uid `%1%'") % uid);
|
|
|
|
|
|
|
|
/* !!! We should really do some check to make sure that there are
|
|
|
|
no processes left running under `uid', but there is no portable
|
|
|
|
way to do so (I think). The most reliable way may be `ps -eo
|
|
|
|
uid | grep -q $uid'. */
|
|
|
|
}
|
|
|
|
|
2004-06-22 09:51:44 +00:00
|
|
|
|
2006-07-20 12:17:25 +00:00
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
2007-02-21 14:31:42 +00:00
|
|
|
string runProgram(Path program, bool searchPath, const Strings & args)
|
2006-07-20 12:17:25 +00:00
|
|
|
{
|
|
|
|
/* Create a pipe. */
|
|
|
|
Pipe pipe;
|
|
|
|
pipe.create();
|
|
|
|
|
|
|
|
/* Fork. */
|
|
|
|
Pid pid;
|
|
|
|
pid = fork();
|
|
|
|
switch (pid) {
|
|
|
|
|
|
|
|
case -1:
|
|
|
|
throw SysError("unable to fork");
|
|
|
|
|
|
|
|
case 0: /* child */
|
|
|
|
try {
|
|
|
|
pipe.readSide.close();
|
|
|
|
|
|
|
|
if (dup2(pipe.writeSide, STDOUT_FILENO) == -1)
|
|
|
|
throw SysError("dupping from-hook write side");
|
2007-02-21 14:31:42 +00:00
|
|
|
|
|
|
|
std::vector<const char *> cargs; /* careful with c_str()! */
|
|
|
|
cargs.push_back(program.c_str());
|
|
|
|
for (Strings::const_iterator i = args.begin(); i != args.end(); ++i)
|
|
|
|
cargs.push_back(i->c_str());
|
|
|
|
cargs.push_back(0);
|
|
|
|
|
|
|
|
if (searchPath)
|
|
|
|
execvp(program.c_str(), (char * *) &cargs[0]);
|
|
|
|
else
|
|
|
|
execv(program.c_str(), (char * *) &cargs[0]);
|
2006-07-20 12:17:25 +00:00
|
|
|
throw SysError(format("executing `%1%'") % program);
|
|
|
|
|
2006-09-04 21:06:23 +00:00
|
|
|
} catch (std::exception & e) {
|
|
|
|
std::cerr << "error: " << e.what() << std::endl;
|
2006-07-20 12:17:25 +00:00
|
|
|
}
|
|
|
|
quickExit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Parent. */
|
|
|
|
|
|
|
|
pipe.writeSide.close();
|
|
|
|
|
|
|
|
string result = drainFD(pipe.readSide);
|
|
|
|
|
|
|
|
/* Wait for the child to finish. */
|
|
|
|
int status = pid.wait(true);
|
|
|
|
if (!statusOk(status))
|
2006-12-07 14:14:35 +00:00
|
|
|
throw Error(format("program `%1%' %2%")
|
2006-07-20 12:17:25 +00:00
|
|
|
% program % statusToString(status));
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void quickExit(int status)
|
|
|
|
{
|
|
|
|
#ifdef __CYGWIN__
|
|
|
|
/* Hack for Cygwin: _exit() doesn't seem to work quite right,
|
|
|
|
since some Berkeley DB code appears to be called when a child
|
|
|
|
exits through _exit() (e.g., because execve() failed). So call
|
|
|
|
the Windows API directly. */
|
|
|
|
ExitProcess(status);
|
|
|
|
#else
|
|
|
|
_exit(status);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-12-07 16:40:41 +00:00
|
|
|
void setuidCleanup()
|
|
|
|
{
|
|
|
|
/* Don't trust the environment. */
|
|
|
|
environ = 0;
|
|
|
|
|
|
|
|
/* Make sure that file descriptors 0, 1, 2 are open. */
|
|
|
|
for (int fd = 0; fd <= 2; ++fd) {
|
|
|
|
struct stat st;
|
|
|
|
if (fstat(fd, &st) == -1) abort();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-06-22 09:51:44 +00:00
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
2004-01-15 20:23:55 +00:00
|
|
|
volatile sig_atomic_t _isInterrupted = 0;
|
|
|
|
|
|
|
|
void _interrupted()
|
|
|
|
{
|
2004-05-11 13:48:25 +00:00
|
|
|
/* Block user interrupts while an exception is being handled.
|
|
|
|
Throwing an exception while another exception is being handled
|
|
|
|
kills the program! */
|
2006-09-04 21:06:23 +00:00
|
|
|
if (!std::uncaught_exception()) {
|
2004-05-11 13:48:25 +00:00
|
|
|
_isInterrupted = 0;
|
2006-12-04 17:17:13 +00:00
|
|
|
throw Interrupted("interrupted by the user");
|
2004-05-11 13:48:25 +00:00
|
|
|
}
|
2004-01-15 20:23:55 +00:00
|
|
|
}
|
2004-06-20 13:37:51 +00:00
|
|
|
|
|
|
|
|
2004-06-22 09:51:44 +00:00
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
2004-06-20 13:37:51 +00:00
|
|
|
string packStrings(const Strings & strings)
|
|
|
|
{
|
|
|
|
string d;
|
|
|
|
for (Strings::const_iterator i = strings.begin();
|
|
|
|
i != strings.end(); ++i)
|
|
|
|
{
|
|
|
|
unsigned int len = i->size();
|
|
|
|
d += len & 0xff;
|
|
|
|
d += (len >> 8) & 0xff;
|
|
|
|
d += (len >> 16) & 0xff;
|
|
|
|
d += (len >> 24) & 0xff;
|
|
|
|
d += *i;
|
|
|
|
}
|
|
|
|
return d;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Strings unpackStrings(const string & s)
|
|
|
|
{
|
|
|
|
Strings strings;
|
|
|
|
|
|
|
|
string::const_iterator i = s.begin();
|
|
|
|
|
|
|
|
while (i != s.end()) {
|
|
|
|
|
|
|
|
if (i + 4 > s.end())
|
|
|
|
throw Error(format("short db entry: `%1%'") % s);
|
|
|
|
|
|
|
|
unsigned int len;
|
|
|
|
len = (unsigned char) *i++;
|
|
|
|
len |= ((unsigned char) *i++) << 8;
|
|
|
|
len |= ((unsigned char) *i++) << 16;
|
|
|
|
len |= ((unsigned char) *i++) << 24;
|
2005-01-20 14:10:19 +00:00
|
|
|
|
|
|
|
if (len == 0xffffffff) return strings; /* explicit end-of-list */
|
2004-06-20 13:37:51 +00:00
|
|
|
|
|
|
|
if (i + len > s.end())
|
|
|
|
throw Error(format("short db entry: `%1%'") % s);
|
|
|
|
|
|
|
|
strings.push_back(string(i, i + len));
|
|
|
|
i += len;
|
|
|
|
}
|
|
|
|
|
|
|
|
return strings;
|
|
|
|
}
|
2004-06-22 08:50:25 +00:00
|
|
|
|
|
|
|
|
2005-09-22 15:43:22 +00:00
|
|
|
Strings tokenizeString(const string & s, const string & separators)
|
|
|
|
{
|
|
|
|
Strings result;
|
|
|
|
string::size_type pos = s.find_first_not_of(separators, 0);
|
|
|
|
while (pos != string::npos) {
|
|
|
|
string::size_type end = s.find_first_of(separators, pos + 1);
|
|
|
|
if (end == string::npos) end = s.size();
|
|
|
|
string token(s, pos, end - pos);
|
|
|
|
result.push_back(token);
|
|
|
|
pos = s.find_first_not_of(separators, end);
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-06-22 08:50:25 +00:00
|
|
|
string statusToString(int status)
|
|
|
|
{
|
|
|
|
if (!WIFEXITED(status) || WEXITSTATUS(status) != 0) {
|
|
|
|
if (WIFEXITED(status))
|
2004-06-22 17:04:10 +00:00
|
|
|
return (format("failed with exit code %1%") % WEXITSTATUS(status)).str();
|
2004-06-22 08:50:25 +00:00
|
|
|
else if (WIFSIGNALED(status))
|
2004-06-22 17:04:10 +00:00
|
|
|
return (format("failed due to signal %1%") % WTERMSIG(status)).str();
|
2004-06-22 08:50:25 +00:00
|
|
|
else
|
|
|
|
return "died abnormally";
|
|
|
|
} else return "succeeded";
|
|
|
|
}
|
2004-06-22 11:03:41 +00:00
|
|
|
|
|
|
|
|
|
|
|
bool statusOk(int status)
|
|
|
|
{
|
|
|
|
return WIFEXITED(status) && WEXITSTATUS(status) == 0;
|
|
|
|
}
|
2004-09-10 13:32:08 +00:00
|
|
|
|
|
|
|
|
2006-08-26 16:48:01 +00:00
|
|
|
string int2String(int n)
|
|
|
|
{
|
2006-09-04 21:06:23 +00:00
|
|
|
std::ostringstream str;
|
2006-08-26 16:48:01 +00:00
|
|
|
str << n;
|
|
|
|
return str.str();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-09-10 13:32:08 +00:00
|
|
|
bool string2Int(const string & s, int & n)
|
|
|
|
{
|
2006-09-04 21:06:23 +00:00
|
|
|
std::istringstream str(s);
|
2004-09-10 13:32:08 +00:00
|
|
|
str >> n;
|
2005-05-04 16:29:44 +00:00
|
|
|
return str && str.get() == EOF;
|
2004-09-10 13:32:08 +00:00
|
|
|
}
|
2006-09-04 21:06:23 +00:00
|
|
|
|
|
|
|
|
|
|
|
}
|