123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392 |
- #pragma once
- #include "types.hh"
- #include <sys/types.h>
- #include <sys/stat.h>
- #include <dirent.h>
- #include <unistd.h>
- #include <signal.h>
- #include <functional>
- #include <cstdio>
- namespace nix {
- #define foreach(it_type, it, collection) \
- for (it_type it = (collection).begin(); it != (collection).end(); ++it)
- #define foreach_reverse(it_type, it, collection) \
- for (it_type it = (collection).rbegin(); it != (collection).rend(); ++it)
- /* Return an environment variable. */
- string getEnv(const string & key, const string & def = "");
- /* Return an absolutized path, resolving paths relative to the
- specified directory, or the current directory otherwise. The path
- is also canonicalised. */
- Path absPath(Path path, Path dir = "");
- /* Canonicalise a path by removing all `.' or `..' components and
- double or trailing slashes. Optionally resolves all symlink
- components such that each component of the resulting path is *not*
- a symbolic link. */
- Path canonPath(const Path & path, bool resolveSymlinks = false);
- /* 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. */
- Path dirOf(const Path & path);
- /* Return the base name of the given canonical path, i.e., everything
- following the final `/'. */
- string baseNameOf(const Path & path);
- /* Check whether a given path is a descendant of the given
- directory. */
- bool isInDir(const Path & path, const Path & dir);
- /* Get status of `path'. */
- struct stat lstat(const Path & path);
- /* Return true iff the given path exists. */
- bool pathExists(const Path & path);
- /* Read the contents (target) of a symbolic link. The result is not
- in any way canonicalised. */
- Path readLink(const Path & path);
- bool isLink(const Path & path);
- /* Read the contents of a directory. The entries `.' and `..' are
- removed. */
- struct DirEntry
- {
- string name;
- ino_t ino;
- unsigned char type; // one of DT_*
- DirEntry(const string & name, ino_t ino, unsigned char type)
- : name(name), ino(ino), type(type) { }
- };
- typedef vector<DirEntry> DirEntries;
- DirEntries readDirectory(const Path & path);
- unsigned char getFileType(const Path & path);
- /* Read the contents of a file into a string. */
- string readFile(int fd);
- string readFile(const Path & path, bool drain = false);
- /* Write a string to a file. */
- void writeFile(const Path & path, const string & s);
- /* Read a line from a file descriptor. */
- string readLine(int fd);
- /* Write a line to a file descriptor. */
- void writeLine(int fd, string s);
- /* Delete a path; i.e., in the case of a directory, it is deleted
- recursively. Don't use this at home, kids. The second variant
- returns the number of bytes and blocks freed, and 'linkThreshold' denotes
- the number of links under which a file is accounted for in 'bytesFreed'. */
- void deletePath(const Path & path);
- void deletePath(const Path & path, unsigned long long & bytesFreed,
- size_t linkThreshold = 1);
- /* Create a temporary directory. */
- Path createTempDir(const Path & tmpRoot = "", const Path & prefix = "nix",
- bool includePid = true, bool useGlobalCounter = true, mode_t mode = 0755);
- /* Create a directory and all its parents, if necessary. Returns the
- list of created directories, in order of creation. */
- Paths createDirs(const Path & path);
- /* Create a symlink. */
- void createSymlink(const Path & target, const Path & link);
- template<class T, class A>
- T singleton(const A & a)
- {
- T t;
- t.insert(a);
- return t;
- }
- /* Messages. */
- typedef enum {
- ltPretty, /* nice, nested output */
- ltEscapes, /* nesting indicated using escape codes (for log2xml) */
- ltFlat /* no nesting */
- } LogType;
- extern LogType logType;
- extern Verbosity verbosity; /* suppress msgs > this */
- class Nest
- {
- private:
- bool nest;
- public:
- Nest();
- ~Nest();
- void open(Verbosity level, const FormatOrString & fs);
- void close();
- };
- void printMsg_(Verbosity level, const FormatOrString & fs);
- #define startNest(varName, level, f) \
- Nest varName; \
- if (level <= verbosity) { \
- varName.open(level, (f)); \
- }
- #define printMsg(level, f) \
- do { \
- if (level <= nix::verbosity) { \
- nix::printMsg_(level, (f)); \
- } \
- } while (0)
- #define debug(f) printMsg(lvlDebug, f)
- void warnOnce(bool & haveWarned, const FormatOrString & fs);
- void writeToStderr(const string & s);
- extern void (*_writeToStderr) (const unsigned char * buf, size_t count);
- /* 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);
- void writeFull(int fd, const string & s);
- MakeError(EndOfFile, Error)
- /* Read a file descriptor until EOF occurs. */
- string drainFD(int fd);
- /* Automatic cleanup of resources. */
- template <class T>
- struct AutoDeleteArray
- {
- T * p;
- AutoDeleteArray(T * p) : p(p) { }
- ~AutoDeleteArray()
- {
- delete [] p;
- }
- };
- class AutoDelete
- {
- Path path;
- bool del;
- bool recursive;
- public:
- AutoDelete(const Path & p, bool recursive = true);
- ~AutoDelete();
- void cancel();
- };
- class AutoCloseFD
- {
- int fd;
- public:
- AutoCloseFD();
- AutoCloseFD(int fd);
- AutoCloseFD(const AutoCloseFD & fd);
- ~AutoCloseFD();
- void operator =(int fd);
- operator int() const;
- void close();
- bool isOpen();
- int borrow();
- };
- class Pipe
- {
- public:
- AutoCloseFD readSide, writeSide;
- void create();
- };
- class AutoCloseDir
- {
- DIR * dir;
- public:
- AutoCloseDir();
- AutoCloseDir(DIR * dir);
- ~AutoCloseDir();
- void operator =(DIR * dir);
- operator DIR *();
- void close();
- };
- class Pid
- {
- pid_t pid;
- bool separatePG;
- int killSignal;
- public:
- Pid();
- Pid(pid_t pid);
- ~Pid();
- void operator =(pid_t pid);
- operator pid_t();
- void kill(bool quiet = false);
- int wait(bool block);
- void setSeparatePG(bool separatePG);
- void setKillSignal(int signal);
- };
- /* An "agent" is a helper program that runs in the background and that we talk
- to over pipes, such as the "guix offload" program. */
- struct Agent
- {
- /* Pipes for talking to the agent. */
- Pipe toAgent;
- /* Pipe for the agent's standard output/error. */
- Pipe fromAgent;
- /* Pipe for build standard output/error--e.g., for build processes started
- by "guix offload". */
- Pipe builderOut;
- /* The process ID of the agent. */
- Pid pid;
- /* The command and arguments passed to the agent. */
- Agent(const string &command, const Strings &args);
- ~Agent();
- };
- /* Kill all processes running under the specified uid by sending them
- a SIGKILL. */
- void killUser(uid_t uid);
- /* Fork a process that runs the given function, and return the child
- pid to the caller. */
- pid_t startProcess(std::function<void()> fun, bool dieWithParent = true,
- const string & errorPrefix = "error: ", bool runExitHandlers = false);
- /* Run a program and return its stdout in a string (i.e., like the
- shell backtick operator). */
- string runProgram(Path program, bool searchPath = false,
- const Strings & args = Strings());
- MakeError(ExecError, Error)
- /* Convert a list of strings to a null-terminated vector of char
- *'s. The result must not be accessed beyond the lifetime of the
- list of strings. */
- std::vector<char *> stringsToCharPtrs(const Strings & ss);
- /* Close all file descriptors except stdin, stdout, stderr, and those
- listed in the given set. Good practice in child processes. */
- void closeMostFDs(const set<int> & exceptions);
- /* Set the close-on-exec flag for the given file descriptor. */
- void closeOnExec(int fd);
- /* Common initialisation performed in child processes. */
- void commonChildInit(Pipe & logPipe);
- /* User interruption. */
- extern volatile sig_atomic_t _isInterrupted;
- void _interrupted();
- void inline checkInterrupt()
- {
- if (_isInterrupted) _interrupted();
- }
- MakeError(Interrupted, BaseError)
- /* String tokenizer. */
- template<class C> C tokenizeString(const string & s, const string & separators = " \t\n\r");
- /* Concatenate the given strings with a separator between the
- elements. */
- string concatStringsSep(const string & sep, const Strings & ss);
- string concatStringsSep(const string & sep, const StringSet & ss);
- /* Remove trailing whitespace from a string. */
- string chomp(const string & s);
- /* Convert the exit status of a child as returned by wait() into an
- error string. */
- string statusToString(int status);
- bool statusOk(int status);
- /* Parse a string into an integer. */
- template<class N> bool string2Int(const string & s, N & n)
- {
- std::istringstream str(s);
- str >> n;
- return str && str.get() == EOF;
- }
- /* Return true iff `s' ends in `suffix'. */
- bool hasSuffix(const string & s, const string & suffix);
- /* Read string `s' from stream `str'. */
- void expect(std::istream & str, const string & s);
- MakeError(FormatError, Error)
- /* Read a C-style string from stream `str'. */
- string parseString(std::istream & str);
- /* Utility function used to parse legacy ATerms. */
- bool endOfList(std::istream & str);
- /* Exception handling in destructors: print an error message, then
- ignore the exception. */
- void ignoreException();
- }
|