2003-04-04 19:14:56 +03:00
|
|
|
#ifndef __UTIL_H
|
|
|
|
#define __UTIL_H
|
|
|
|
|
2003-04-08 15:00:51 +03:00
|
|
|
#include <string>
|
2003-06-20 17:11:31 +03:00
|
|
|
#include <list>
|
2003-07-17 15:27:55 +03:00
|
|
|
#include <set>
|
2003-04-08 15:00:51 +03:00
|
|
|
#include <sstream>
|
|
|
|
|
2003-10-22 13:48:22 +03:00
|
|
|
#include <sys/types.h>
|
|
|
|
#include <dirent.h>
|
2003-04-08 15:00:51 +03:00
|
|
|
#include <unistd.h>
|
2004-01-15 22:23:55 +02:00
|
|
|
#include <signal.h>
|
2003-04-08 15:00:51 +03:00
|
|
|
|
2003-06-27 16:55:12 +03:00
|
|
|
#include <boost/format.hpp>
|
|
|
|
|
2003-04-04 19:14:56 +03:00
|
|
|
using namespace std;
|
2003-06-27 16:55:12 +03:00
|
|
|
using namespace boost;
|
2003-04-04 19:14:56 +03:00
|
|
|
|
|
|
|
|
|
|
|
class Error : public exception
|
|
|
|
{
|
2003-06-16 16:33:38 +03:00
|
|
|
protected:
|
2003-04-04 19:14:56 +03:00
|
|
|
string err;
|
|
|
|
public:
|
2003-06-27 17:56:12 +03:00
|
|
|
Error(const format & f);
|
|
|
|
~Error() throw () { };
|
2003-04-04 19:14:56 +03:00
|
|
|
const char * what() const throw () { return err.c_str(); }
|
2003-07-07 12:25:26 +03:00
|
|
|
const string & msg() const throw () { return err; }
|
2003-04-04 19:14:56 +03:00
|
|
|
};
|
|
|
|
|
2003-06-16 16:33:38 +03:00
|
|
|
class SysError : public Error
|
|
|
|
{
|
|
|
|
public:
|
2003-06-27 17:56:12 +03:00
|
|
|
SysError(const format & f);
|
2003-06-16 16:33:38 +03:00
|
|
|
};
|
|
|
|
|
2003-04-04 19:14:56 +03:00
|
|
|
class UsageError : public Error
|
|
|
|
{
|
|
|
|
public:
|
2003-06-27 17:56:12 +03:00
|
|
|
UsageError(const format & f) : Error(f) { };
|
2003-04-04 19:14:56 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2003-06-20 17:11:31 +03:00
|
|
|
typedef list<string> Strings;
|
2003-07-17 15:27:55 +03:00
|
|
|
typedef set<string> StringSet;
|
2003-04-04 19:14:56 +03:00
|
|
|
|
|
|
|
|
2003-10-07 17:37:41 +03:00
|
|
|
/* Paths are just strings. */
|
|
|
|
typedef string Path;
|
|
|
|
typedef list<Path> Paths;
|
|
|
|
typedef set<Path> PathSet;
|
|
|
|
|
|
|
|
|
2003-04-08 15:00:51 +03:00
|
|
|
/* The canonical system name, as returned by config.guess. */
|
2003-05-26 16:45:00 +03:00
|
|
|
extern string thisSystem;
|
2003-04-08 15:00:51 +03:00
|
|
|
|
|
|
|
|
2004-05-12 12:35:51 +03:00
|
|
|
/* Return an environment variable. */
|
|
|
|
string getEnv(const string & key, const string & def = "");
|
|
|
|
|
2003-06-16 16:33:38 +03:00
|
|
|
/* Return an absolutized path, resolving paths relative to the
|
2003-07-07 12:25:26 +03:00
|
|
|
specified directory, or the current directory otherwise. The path
|
|
|
|
is also canonicalised. */
|
2003-10-07 17:37:41 +03:00
|
|
|
Path absPath(Path path, Path dir = "");
|
2003-06-16 16:33:38 +03:00
|
|
|
|
2004-03-27 19:58:04 +02:00
|
|
|
/* Canonicalise a path by removing all `.' or `..' components and
|
|
|
|
double or trailing slashes. */
|
2003-10-07 17:37:41 +03:00
|
|
|
Path canonPath(const Path & path);
|
2003-07-07 12:25:26 +03:00
|
|
|
|
2004-03-27 19:58:04 +02:00
|
|
|
/* Return the directory part of the given canonical path, i.e.,
|
|
|
|
everything before the final `/'. If the path is the root or an
|
|
|
|
immediate child thereof (e.g., `/foo'), this means an empty string
|
|
|
|
is returned. */
|
2003-10-07 17:37:41 +03:00
|
|
|
Path dirOf(const Path & path);
|
2003-06-16 16:33:38 +03:00
|
|
|
|
2004-03-27 19:58:04 +02:00
|
|
|
/* Return the base name of the given canonical path, i.e., everything
|
|
|
|
following the final `/'. */
|
2003-10-07 17:37:41 +03:00
|
|
|
string baseNameOf(const Path & path);
|
2003-04-08 15:00:51 +03:00
|
|
|
|
2003-07-08 16:22:08 +03:00
|
|
|
/* Return true iff the given path exists. */
|
2003-10-07 17:37:41 +03:00
|
|
|
bool pathExists(const Path & path);
|
2003-04-08 15:00:51 +03:00
|
|
|
|
2004-01-05 18:26:43 +02:00
|
|
|
/* Read the contents (target) of a symbolic link. The result is not
|
|
|
|
in any way canonicalised. */
|
|
|
|
Path readLink(const Path & path);
|
|
|
|
|
2003-11-19 19:27:16 +02:00
|
|
|
/* Read the contents of a directory. The entries `.' and `..' are
|
|
|
|
removed. */
|
|
|
|
Strings readDirectory(const Path & path);
|
|
|
|
|
2003-06-23 17:40:49 +03:00
|
|
|
/* Delete a path; i.e., in the case of a directory, it is deleted
|
|
|
|
recursively. Don't use this at home, kids. */
|
2003-10-07 17:37:41 +03:00
|
|
|
void deletePath(const Path & path);
|
2003-08-22 23:12:44 +03:00
|
|
|
|
|
|
|
/* Make a path read-only recursively. */
|
2003-10-07 17:37:41 +03:00
|
|
|
void makePathReadOnly(const Path & path);
|
2003-06-23 17:40:49 +03:00
|
|
|
|
2003-10-02 14:55:38 +03:00
|
|
|
/* Create a temporary directory. */
|
2003-10-07 17:37:41 +03:00
|
|
|
Path createTempDir();
|
2003-10-02 14:55:38 +03:00
|
|
|
|
2003-11-22 17:58:34 +02:00
|
|
|
/* Create a file and write the given text to it. The file is written
|
|
|
|
in binary mode (i.e., no end-of-line conversions). The path should
|
|
|
|
not already exist. */
|
|
|
|
void writeStringToFile(const Path & path, const string & s);
|
|
|
|
|
2003-06-23 17:40:49 +03:00
|
|
|
|
2003-07-04 15:18:06 +03:00
|
|
|
/* Messages. */
|
|
|
|
|
2004-03-22 22:53:49 +02:00
|
|
|
|
|
|
|
typedef enum {
|
|
|
|
ltPretty, /* nice, nested output */
|
|
|
|
ltEscapes, /* nesting indicated using escape codes (for log2xml) */
|
|
|
|
ltFlat /* no nesting */
|
|
|
|
} LogType;
|
|
|
|
|
2003-07-24 11:53:43 +03:00
|
|
|
typedef enum {
|
* Upgrade operation in `nix-env'. For instance, you can say
nix-env -u foo.nix strategoxt
to replace the installed `strategoxt' derivation with the one from `foo.nix', if
the latter has a higher version number. This is a no-op if `strategoxt' is not
installed. Wildcards are also accepted, so
nix-env -u foo.nix '*'
will replace any installed derivation with newer versions from `foo.nix', if
available.
The notion of "version number" is somewhat ad hoc, but should be useful in most
cases, as evidenced by the following unit tests for the version comparator:
TEST("1.0", "2.3", -1);
TEST("2.1", "2.3", -1);
TEST("2.3", "2.3", 0);
TEST("2.5", "2.3", 1);
TEST("3.1", "2.3", 1);
TEST("2.3.1", "2.3", 1);
TEST("2.3.1", "2.3a", 1);
TEST("2.3pre1", "2.3", -1);
TEST("2.3pre3", "2.3pre12", -1);
TEST("2.3a", "2.3c", -1);
TEST("2.3pre1", "2.3c", -1);
TEST("2.3pre1", "2.3q", -1);
(-1 = less, 0 = equal, 1 = greater)
* A new verbosity level `lvlInfo', between `lvlError' and `lvlTalkative'. This is
the default for `nix-env', so without any `-v' flags users should get useful
output, e.g.,
$ nix-env -u foo.nix strategoxt
upgrading `strategoxt-0.9.2' to `strategoxt-0.9.3'
2003-12-22 18:04:00 +02:00
|
|
|
lvlError,
|
|
|
|
lvlInfo,
|
2003-07-24 16:43:16 +03:00
|
|
|
lvlTalkative,
|
|
|
|
lvlChatty,
|
|
|
|
lvlDebug,
|
|
|
|
lvlVomit
|
2003-07-24 11:53:43 +03:00
|
|
|
} Verbosity;
|
|
|
|
|
2004-03-22 22:53:49 +02:00
|
|
|
extern LogType logType;
|
|
|
|
extern Verbosity verbosity; /* suppress msgs > this */
|
2003-07-24 11:53:43 +03:00
|
|
|
|
2003-07-04 15:18:06 +03:00
|
|
|
class Nest
|
|
|
|
{
|
|
|
|
private:
|
|
|
|
bool nest;
|
|
|
|
public:
|
2003-11-09 12:35:45 +02:00
|
|
|
Nest();
|
2003-07-04 15:18:06 +03:00
|
|
|
~Nest();
|
2003-11-09 12:35:45 +02:00
|
|
|
void open(Verbosity level, const format & f);
|
2004-03-22 23:42:28 +02:00
|
|
|
void close();
|
2003-07-04 15:18:06 +03:00
|
|
|
};
|
|
|
|
|
2003-11-09 12:35:45 +02:00
|
|
|
void printMsg_(Verbosity level, const format & f);
|
|
|
|
|
|
|
|
#define startNest(varName, level, f) \
|
|
|
|
Nest varName; \
|
|
|
|
if (level <= verbosity) { \
|
|
|
|
varName.open(level, (f)); \
|
|
|
|
}
|
|
|
|
|
|
|
|
#define printMsg(level, f) \
|
|
|
|
do { \
|
|
|
|
if (level <= verbosity) { \
|
|
|
|
printMsg_(level, (f)); \
|
|
|
|
} \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
#define debug(f) printMsg(lvlDebug, f)
|
2003-05-26 01:42:19 +03:00
|
|
|
|
|
|
|
|
2003-07-21 00:11:43 +03:00
|
|
|
/* Wrappers arount read()/write() that read/write exactly the
|
|
|
|
requested number of bytes. */
|
|
|
|
void readFull(int fd, unsigned char * buf, size_t count);
|
|
|
|
void writeFull(int fd, const unsigned char * buf, size_t count);
|
|
|
|
|
|
|
|
|
2003-10-22 13:48:22 +03:00
|
|
|
/* Automatic cleanup of resources. */
|
|
|
|
|
|
|
|
class AutoDelete
|
|
|
|
{
|
|
|
|
string path;
|
|
|
|
bool del;
|
|
|
|
public:
|
|
|
|
AutoDelete(const string & p);
|
|
|
|
~AutoDelete();
|
|
|
|
void cancel();
|
|
|
|
};
|
|
|
|
|
|
|
|
class AutoCloseFD
|
|
|
|
{
|
|
|
|
int fd;
|
|
|
|
public:
|
|
|
|
AutoCloseFD();
|
|
|
|
AutoCloseFD(int fd);
|
|
|
|
~AutoCloseFD();
|
|
|
|
void operator =(int fd);
|
|
|
|
operator int();
|
2004-06-15 16:49:42 +03:00
|
|
|
void close();
|
|
|
|
bool isOpen();
|
|
|
|
};
|
|
|
|
|
|
|
|
class Pipe
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
AutoCloseFD readSide, writeSide;
|
|
|
|
void create();
|
2003-10-22 13:48:22 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
class AutoCloseDir
|
|
|
|
{
|
|
|
|
DIR * dir;
|
|
|
|
public:
|
|
|
|
AutoCloseDir();
|
|
|
|
AutoCloseDir(DIR * dir);
|
|
|
|
~AutoCloseDir();
|
|
|
|
void operator =(DIR * dir);
|
|
|
|
operator DIR *();
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2004-01-15 22:23:55 +02:00
|
|
|
/* User interruption. */
|
|
|
|
|
|
|
|
extern volatile sig_atomic_t _isInterrupted;
|
|
|
|
|
|
|
|
void _interrupted();
|
|
|
|
|
|
|
|
void inline checkInterrupt()
|
|
|
|
{
|
|
|
|
if (_isInterrupted) _interrupted();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-04-04 19:14:56 +03:00
|
|
|
#endif /* !__UTIL_H */
|