2012-07-18 18:59:03 +00:00
|
|
|
|
#pragma once
|
2003-04-04 16:14:56 +00:00
|
|
|
|
|
2006-09-04 21:06:23 +00:00
|
|
|
|
#include "types.hh"
|
2016-04-25 13:26:07 +00:00
|
|
|
|
#include "logging.hh"
|
2003-04-08 12:00:51 +00:00
|
|
|
|
|
2003-10-22 10:48:22 +00:00
|
|
|
|
#include <sys/types.h>
|
2010-12-13 13:32:58 +00:00
|
|
|
|
#include <sys/stat.h>
|
2003-10-22 10:48:22 +00:00
|
|
|
|
#include <dirent.h>
|
2003-04-08 12:00:51 +00:00
|
|
|
|
#include <unistd.h>
|
2004-01-15 20:23:55 +00:00
|
|
|
|
#include <signal.h>
|
2016-09-20 13:39:08 +00:00
|
|
|
|
|
2014-07-10 14:50:51 +00:00
|
|
|
|
#include <functional>
|
2016-01-19 12:00:02 +00:00
|
|
|
|
#include <limits>
|
2009-11-24 13:28:46 +00:00
|
|
|
|
#include <cstdio>
|
2016-09-20 13:39:08 +00:00
|
|
|
|
#include <map>
|
2009-11-24 12:56:26 +00:00
|
|
|
|
|
2015-11-07 03:51:33 +00:00
|
|
|
|
#ifndef HAVE_STRUCT_DIRENT_D_TYPE
|
|
|
|
|
#define DT_UNKNOWN 0
|
|
|
|
|
#define DT_REG 1
|
|
|
|
|
#define DT_LNK 2
|
|
|
|
|
#define DT_DIR 3
|
|
|
|
|
#endif
|
2003-06-27 13:55:12 +00:00
|
|
|
|
|
2006-09-04 21:06:23 +00:00
|
|
|
|
namespace nix {
|
2003-10-07 14:37:41 +00:00
|
|
|
|
|
|
|
|
|
|
2004-05-12 09:35:51 +00:00
|
|
|
|
/* Return an environment variable. */
|
|
|
|
|
string getEnv(const string & key, const string & def = "");
|
|
|
|
|
|
2016-09-20 13:39:08 +00:00
|
|
|
|
/* Get the entire environment. */
|
|
|
|
|
std::map<std::string, std::string> getEnv();
|
|
|
|
|
|
2003-06-16 13:33:38 +00:00
|
|
|
|
/* Return an absolutized path, resolving paths relative to the
|
2003-07-07 09:25:26 +00:00
|
|
|
|
specified directory, or the current directory otherwise. The path
|
|
|
|
|
is also canonicalised. */
|
2003-10-07 14:37:41 +00:00
|
|
|
|
Path absPath(Path path, Path dir = "");
|
2003-06-16 13:33:38 +00:00
|
|
|
|
|
2004-03-27 17:58:04 +00:00
|
|
|
|
/* Canonicalise a path by removing all `.' or `..' components and
|
2006-01-08 17:16:03 +00:00
|
|
|
|
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);
|
2003-07-07 09:25:26 +00:00
|
|
|
|
|
2004-03-27 17:58:04 +00: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 14:37:41 +00:00
|
|
|
|
Path dirOf(const Path & path);
|
2003-06-16 13:33:38 +00:00
|
|
|
|
|
2004-03-27 17:58:04 +00:00
|
|
|
|
/* Return the base name of the given canonical path, i.e., everything
|
|
|
|
|
following the final `/'. */
|
2003-10-07 14:37:41 +00:00
|
|
|
|
string baseNameOf(const Path & path);
|
2003-04-08 12:00:51 +00:00
|
|
|
|
|
2013-07-12 12:01:25 +00:00
|
|
|
|
/* Check whether a given path is a descendant of the given
|
|
|
|
|
directory. */
|
|
|
|
|
bool isInDir(const Path & path, const Path & dir);
|
|
|
|
|
|
2010-12-13 13:32:58 +00:00
|
|
|
|
/* Get status of `path'. */
|
|
|
|
|
struct stat lstat(const Path & path);
|
|
|
|
|
|
2003-07-08 13:22:08 +00:00
|
|
|
|
/* Return true iff the given path exists. */
|
2003-10-07 14:37:41 +00:00
|
|
|
|
bool pathExists(const Path & path);
|
2003-04-08 12:00:51 +00:00
|
|
|
|
|
2004-01-05 16:26:43 +00:00
|
|
|
|
/* Read the contents (target) of a symbolic link. The result is not
|
|
|
|
|
in any way canonicalised. */
|
|
|
|
|
Path readLink(const Path & path);
|
|
|
|
|
|
2005-02-01 13:48:46 +00:00
|
|
|
|
bool isLink(const Path & path);
|
|
|
|
|
|
2003-11-19 17:27:16 +00:00
|
|
|
|
/* Read the contents of a directory. The entries `.' and `..' are
|
|
|
|
|
removed. */
|
2014-08-01 14:37:47 +00:00
|
|
|
|
struct DirEntry
|
|
|
|
|
{
|
|
|
|
|
string name;
|
|
|
|
|
ino_t ino;
|
|
|
|
|
unsigned char type; // one of DT_*
|
2014-08-01 15:30:51 +00:00
|
|
|
|
DirEntry(const string & name, ino_t ino, unsigned char type)
|
|
|
|
|
: name(name), ino(ino), type(type) { }
|
2014-08-01 14:37:47 +00:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
typedef vector<DirEntry> DirEntries;
|
|
|
|
|
|
|
|
|
|
DirEntries readDirectory(const Path & path);
|
2003-11-19 17:27:16 +00:00
|
|
|
|
|
2014-10-03 20:37:51 +00:00
|
|
|
|
unsigned char getFileType(const Path & path);
|
|
|
|
|
|
2005-02-01 22:07:48 +00:00
|
|
|
|
/* Read the contents of a file into a string. */
|
|
|
|
|
string readFile(int fd);
|
2012-08-20 19:27:30 +00:00
|
|
|
|
string readFile(const Path & path, bool drain = false);
|
2005-02-01 22:07:48 +00:00
|
|
|
|
|
2005-02-09 09:50:29 +00:00
|
|
|
|
/* Write a string to a file. */
|
2010-02-24 12:16:50 +00:00
|
|
|
|
void writeFile(const Path & path, const string & s);
|
2005-02-09 09:50:29 +00:00
|
|
|
|
|
2009-03-28 19:29:55 +00:00
|
|
|
|
/* Read a line from a file descriptor. */
|
|
|
|
|
string readLine(int fd);
|
|
|
|
|
|
|
|
|
|
/* Write a line to a file descriptor. */
|
|
|
|
|
void writeLine(int fd, string s);
|
|
|
|
|
|
2003-06-23 14:40:49 +00:00
|
|
|
|
/* Delete a path; i.e., in the case of a directory, it is deleted
|
2016-02-24 16:44:12 +00:00
|
|
|
|
recursively. It's not an error if the path does not exist. The
|
|
|
|
|
second variant returns the number of bytes and blocks freed. */
|
2003-10-07 14:37:41 +00:00
|
|
|
|
void deletePath(const Path & path);
|
2003-08-22 20:12:44 +00:00
|
|
|
|
|
2012-08-02 02:34:46 +00:00
|
|
|
|
void deletePath(const Path & path, unsigned long long & bytesFreed);
|
2005-12-15 21:11:39 +00:00
|
|
|
|
|
2003-10-02 11:55:38 +00:00
|
|
|
|
/* Create a temporary directory. */
|
2008-03-27 13:45:17 +00:00
|
|
|
|
Path createTempDir(const Path & tmpRoot = "", const Path & prefix = "nix",
|
2012-07-26 19:04:40 +00:00
|
|
|
|
bool includePid = true, bool useGlobalCounter = true, mode_t mode = 0755);
|
2003-10-02 11:55:38 +00:00
|
|
|
|
|
2016-04-20 12:12:38 +00:00
|
|
|
|
/* Return the path to $XDG_CACHE_HOME/.cache. */
|
|
|
|
|
Path getCacheDir();
|
|
|
|
|
|
2007-10-27 00:46:59 +00:00
|
|
|
|
/* Create a directory and all its parents, if necessary. Returns the
|
|
|
|
|
list of created directories, in order of creation. */
|
|
|
|
|
Paths createDirs(const Path & path);
|
2005-03-24 17:46:38 +00:00
|
|
|
|
|
2014-02-27 22:17:53 +00:00
|
|
|
|
/* Create a symlink. */
|
|
|
|
|
void createSymlink(const Path & target, const Path & link);
|
|
|
|
|
|
2015-04-09 09:42:04 +00:00
|
|
|
|
/* Atomically create or replace a symlink. */
|
|
|
|
|
void replaceSymlink(const Path & target, const Path & link);
|
|
|
|
|
|
2003-11-22 15:58:34 +00:00
|
|
|
|
|
2003-07-20 21:11:43 +00:00
|
|
|
|
/* Wrappers arount read()/write() that read/write exactly the
|
|
|
|
|
requested number of bytes. */
|
|
|
|
|
void readFull(int fd, unsigned char * buf, size_t count);
|
2016-09-16 16:52:42 +00:00
|
|
|
|
void writeFull(int fd, const unsigned char * buf, size_t count, bool allowInterrupts = true);
|
|
|
|
|
void writeFull(int fd, const string & s, bool allowInterrupts = true);
|
2003-07-20 21:11:43 +00:00
|
|
|
|
|
2006-12-04 17:17:13 +00:00
|
|
|
|
MakeError(EndOfFile, Error)
|
|
|
|
|
|
2003-07-20 21:11:43 +00:00
|
|
|
|
|
2006-07-20 12:17:25 +00:00
|
|
|
|
/* Read a file descriptor until EOF occurs. */
|
|
|
|
|
string drainFD(int fd);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2003-10-22 10:48:22 +00:00
|
|
|
|
/* Automatic cleanup of resources. */
|
|
|
|
|
|
2006-12-04 17:17:13 +00:00
|
|
|
|
|
|
|
|
|
template <class T>
|
|
|
|
|
struct AutoDeleteArray
|
|
|
|
|
{
|
|
|
|
|
T * p;
|
|
|
|
|
AutoDeleteArray(T * p) : p(p) { }
|
2013-01-03 12:00:46 +00:00
|
|
|
|
~AutoDeleteArray()
|
2006-12-04 17:17:13 +00:00
|
|
|
|
{
|
|
|
|
|
delete [] p;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
2003-10-22 10:48:22 +00:00
|
|
|
|
class AutoDelete
|
|
|
|
|
{
|
2006-12-12 23:05:01 +00:00
|
|
|
|
Path path;
|
2003-10-22 10:48:22 +00:00
|
|
|
|
bool del;
|
2013-01-03 12:00:46 +00:00
|
|
|
|
bool recursive;
|
2003-10-22 10:48:22 +00:00
|
|
|
|
public:
|
2015-11-16 10:53:10 +00:00
|
|
|
|
AutoDelete();
|
2007-10-27 00:46:59 +00:00
|
|
|
|
AutoDelete(const Path & p, bool recursive = true);
|
2003-10-22 10:48:22 +00:00
|
|
|
|
~AutoDelete();
|
|
|
|
|
void cancel();
|
2015-11-16 10:53:10 +00:00
|
|
|
|
void reset(const Path & p, bool recursive = true);
|
2015-10-01 14:47:43 +00:00
|
|
|
|
operator Path() const { return path; }
|
2003-10-22 10:48:22 +00:00
|
|
|
|
};
|
|
|
|
|
|
2004-06-22 09:51:44 +00:00
|
|
|
|
|
2003-10-22 10:48:22 +00:00
|
|
|
|
class AutoCloseFD
|
|
|
|
|
{
|
|
|
|
|
int fd;
|
2016-07-11 19:44:44 +00:00
|
|
|
|
void close();
|
2003-10-22 10:48:22 +00:00
|
|
|
|
public:
|
|
|
|
|
AutoCloseFD();
|
|
|
|
|
AutoCloseFD(int fd);
|
2016-07-11 19:44:44 +00:00
|
|
|
|
AutoCloseFD(const AutoCloseFD & fd) = delete;
|
|
|
|
|
AutoCloseFD(AutoCloseFD&& fd);
|
2003-10-22 10:48:22 +00:00
|
|
|
|
~AutoCloseFD();
|
2016-07-11 19:44:44 +00:00
|
|
|
|
AutoCloseFD& operator =(const AutoCloseFD & fd) = delete;
|
|
|
|
|
AutoCloseFD& operator =(AutoCloseFD&& fd);
|
|
|
|
|
int get() const;
|
|
|
|
|
explicit operator bool() const;
|
|
|
|
|
int release();
|
2004-06-15 13:49:42 +00:00
|
|
|
|
};
|
|
|
|
|
|
2004-06-22 09:51:44 +00:00
|
|
|
|
|
2004-06-15 13:49:42 +00:00
|
|
|
|
class Pipe
|
|
|
|
|
{
|
|
|
|
|
public:
|
|
|
|
|
AutoCloseFD readSide, writeSide;
|
|
|
|
|
void create();
|
2003-10-22 10:48:22 +00:00
|
|
|
|
};
|
|
|
|
|
|
2004-06-22 09:51:44 +00:00
|
|
|
|
|
2003-10-22 10:48:22 +00:00
|
|
|
|
class AutoCloseDir
|
|
|
|
|
{
|
|
|
|
|
DIR * dir;
|
|
|
|
|
public:
|
|
|
|
|
AutoCloseDir();
|
|
|
|
|
AutoCloseDir(DIR * dir);
|
|
|
|
|
~AutoCloseDir();
|
|
|
|
|
void operator =(DIR * dir);
|
|
|
|
|
operator DIR *();
|
2011-12-22 15:55:53 +00:00
|
|
|
|
void close();
|
2003-10-22 10:48:22 +00:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
2004-06-22 09:51:44 +00:00
|
|
|
|
class Pid
|
|
|
|
|
{
|
|
|
|
|
pid_t pid;
|
|
|
|
|
bool separatePG;
|
2007-03-19 12:48:45 +00:00
|
|
|
|
int killSignal;
|
2004-06-22 09:51:44 +00:00
|
|
|
|
public:
|
|
|
|
|
Pid();
|
2014-07-10 14:50:51 +00:00
|
|
|
|
Pid(pid_t pid);
|
2004-06-22 09:51:44 +00:00
|
|
|
|
~Pid();
|
|
|
|
|
void operator =(pid_t pid);
|
|
|
|
|
operator pid_t();
|
2014-08-04 15:27:45 +00:00
|
|
|
|
void kill(bool quiet = false);
|
2004-06-22 09:51:44 +00:00
|
|
|
|
int wait(bool block);
|
|
|
|
|
void setSeparatePG(bool separatePG);
|
2007-03-19 12:48:45 +00:00
|
|
|
|
void setKillSignal(int signal);
|
2016-10-12 13:49:37 +00:00
|
|
|
|
pid_t release();
|
2004-06-22 09:51:44 +00:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
2006-12-07 00:16:07 +00:00
|
|
|
|
/* Kill all processes running under the specified uid by sending them
|
|
|
|
|
a SIGKILL. */
|
|
|
|
|
void killUser(uid_t uid);
|
|
|
|
|
|
|
|
|
|
|
2014-07-10 14:50:51 +00:00
|
|
|
|
/* Fork a process that runs the given function, and return the child
|
|
|
|
|
pid to the caller. */
|
2014-12-10 15:35:42 +00:00
|
|
|
|
struct ProcessOptions
|
|
|
|
|
{
|
|
|
|
|
string errorPrefix;
|
|
|
|
|
bool dieWithParent;
|
|
|
|
|
bool runExitHandlers;
|
|
|
|
|
bool allowVfork;
|
|
|
|
|
ProcessOptions() : errorPrefix("error: "), dieWithParent(true), runExitHandlers(false), allowVfork(true) { };
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
pid_t startProcess(std::function<void()> fun, const ProcessOptions & options = ProcessOptions());
|
2014-07-10 14:50:51 +00:00
|
|
|
|
|
|
|
|
|
|
2006-07-20 12:17:25 +00:00
|
|
|
|
/* Run a program and return its stdout in a string (i.e., like the
|
|
|
|
|
shell backtick operator). */
|
2007-02-21 14:31:42 +00:00
|
|
|
|
string runProgram(Path program, bool searchPath = false,
|
2015-02-04 15:43:32 +00:00
|
|
|
|
const Strings & args = Strings(), const string & input = "");
|
2006-07-20 12:17:25 +00:00
|
|
|
|
|
2016-09-21 14:21:47 +00:00
|
|
|
|
class ExecError : public Error
|
|
|
|
|
{
|
|
|
|
|
public:
|
|
|
|
|
int status;
|
|
|
|
|
|
|
|
|
|
template<typename... Args>
|
|
|
|
|
ExecError(int status, Args... args)
|
|
|
|
|
: Error(args...), status(status)
|
|
|
|
|
{ }
|
|
|
|
|
};
|
2014-05-21 15:19:36 +00:00
|
|
|
|
|
2014-12-12 14:01:16 +00:00
|
|
|
|
/* 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. */
|
2015-06-09 08:50:55 +00:00
|
|
|
|
std::vector<char *> stringsToCharPtrs(const Strings & ss);
|
2014-12-12 14:01:16 +00:00
|
|
|
|
|
2008-08-02 12:54:35 +00:00
|
|
|
|
/* 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);
|
|
|
|
|
|
2012-03-05 19:29:00 +00:00
|
|
|
|
/* Set the close-on-exec flag for the given file descriptor. */
|
|
|
|
|
void closeOnExec(int fd);
|
|
|
|
|
|
2014-07-31 08:31:17 +00:00
|
|
|
|
/* Restore default handling of SIGPIPE, otherwise some programs will
|
|
|
|
|
randomly say "Broken pipe". */
|
|
|
|
|
void restoreSIGPIPE();
|
|
|
|
|
|
2006-07-20 12:17:25 +00:00
|
|
|
|
|
2004-01-15 20:23:55 +00:00
|
|
|
|
/* User interruption. */
|
|
|
|
|
|
|
|
|
|
extern volatile sig_atomic_t _isInterrupted;
|
|
|
|
|
|
2016-03-29 13:08:24 +00:00
|
|
|
|
extern thread_local bool interruptThrown;
|
|
|
|
|
|
2004-01-15 20:23:55 +00:00
|
|
|
|
void _interrupted();
|
|
|
|
|
|
|
|
|
|
void inline checkInterrupt()
|
|
|
|
|
{
|
|
|
|
|
if (_isInterrupted) _interrupted();
|
|
|
|
|
}
|
|
|
|
|
|
2007-08-12 00:29:28 +00:00
|
|
|
|
MakeError(Interrupted, BaseError)
|
2006-12-04 17:17:13 +00:00
|
|
|
|
|
2004-01-15 20:23:55 +00:00
|
|
|
|
|
2005-09-22 15:43:22 +00:00
|
|
|
|
/* String tokenizer. */
|
2012-09-19 19:43:23 +00:00
|
|
|
|
template<class C> C tokenizeString(const string & s, const string & separators = " \t\n\r");
|
2005-09-22 15:43:22 +00:00
|
|
|
|
|
|
|
|
|
|
2010-08-27 13:18:13 +00:00
|
|
|
|
/* Concatenate the given strings with a separator between the
|
|
|
|
|
elements. */
|
|
|
|
|
string concatStringsSep(const string & sep, const Strings & ss);
|
2012-11-26 14:39:10 +00:00
|
|
|
|
string concatStringsSep(const string & sep, const StringSet & ss);
|
2010-08-27 13:18:13 +00:00
|
|
|
|
|
|
|
|
|
|
2012-08-01 15:19:24 +00:00
|
|
|
|
/* Remove trailing whitespace from a string. */
|
|
|
|
|
string chomp(const string & s);
|
|
|
|
|
|
|
|
|
|
|
2015-04-09 09:42:04 +00:00
|
|
|
|
/* Remove whitespace from the start and end of a string. */
|
|
|
|
|
string trim(const string & s, const string & whitespace = " \n\r\t");
|
|
|
|
|
|
|
|
|
|
|
2015-06-17 14:20:11 +00:00
|
|
|
|
/* Replace all occurrences of a string inside another string. */
|
|
|
|
|
string replaceStrings(const std::string & s,
|
|
|
|
|
const std::string & from, const std::string & to);
|
|
|
|
|
|
|
|
|
|
|
2004-06-22 08:50:25 +00:00
|
|
|
|
/* Convert the exit status of a child as returned by wait() into an
|
|
|
|
|
error string. */
|
|
|
|
|
string statusToString(int status);
|
|
|
|
|
|
2004-06-22 11:03:41 +00:00
|
|
|
|
bool statusOk(int status);
|
|
|
|
|
|
2004-06-22 08:50:25 +00:00
|
|
|
|
|
2004-09-10 13:32:08 +00:00
|
|
|
|
/* Parse a string into an integer. */
|
2009-11-24 12:26:25 +00:00
|
|
|
|
template<class N> bool string2Int(const string & s, N & n)
|
|
|
|
|
{
|
2016-01-19 12:00:02 +00:00
|
|
|
|
if (string(s, 0, 1) == "-" && !std::numeric_limits<N>::is_signed)
|
|
|
|
|
return false;
|
2009-11-24 12:26:25 +00:00
|
|
|
|
std::istringstream str(s);
|
|
|
|
|
str >> n;
|
|
|
|
|
return str && str.get() == EOF;
|
|
|
|
|
}
|
|
|
|
|
|
2016-01-04 23:40:40 +00:00
|
|
|
|
/* Parse a string into a float. */
|
|
|
|
|
template<class N> bool string2Float(const string & s, N & n)
|
|
|
|
|
{
|
|
|
|
|
std::istringstream str(s);
|
|
|
|
|
str >> n;
|
|
|
|
|
return str && str.get() == EOF;
|
|
|
|
|
}
|
|
|
|
|
|
2004-09-10 13:32:08 +00:00
|
|
|
|
|
2016-04-29 19:04:40 +00:00
|
|
|
|
/* Return true iff `s' starts with `prefix'. */
|
|
|
|
|
bool hasPrefix(const string & s, const string & prefix);
|
|
|
|
|
|
|
|
|
|
|
2008-08-25 13:31:57 +00:00
|
|
|
|
/* Return true iff `s' ends in `suffix'. */
|
|
|
|
|
bool hasSuffix(const string & s, const string & suffix);
|
|
|
|
|
|
|
|
|
|
|
2016-09-14 12:42:15 +00:00
|
|
|
|
/* Convert a string to lower case. */
|
|
|
|
|
std::string toLower(const std::string & s);
|
|
|
|
|
|
|
|
|
|
|
2012-09-29 01:39:30 +00:00
|
|
|
|
/* Escape a string that contains octal-encoded escape codes such as
|
|
|
|
|
used in /etc/fstab and /proc/mounts (e.g. "foo\040bar" decodes to
|
|
|
|
|
"foo bar"). */
|
|
|
|
|
string decodeOctalEscaped(const string & s);
|
|
|
|
|
|
|
|
|
|
|
2010-04-19 13:46:58 +00:00
|
|
|
|
/* Exception handling in destructors: print an error message, then
|
|
|
|
|
ignore the exception. */
|
|
|
|
|
void ignoreException();
|
2008-09-17 10:02:55 +00:00
|
|
|
|
|
|
|
|
|
|
2014-08-20 14:01:16 +00:00
|
|
|
|
/* Some ANSI escape sequences. */
|
|
|
|
|
#define ANSI_NORMAL "\e[0m"
|
|
|
|
|
#define ANSI_BOLD "\e[1m"
|
|
|
|
|
#define ANSI_RED "\e[31;1m"
|
|
|
|
|
|
|
|
|
|
|
2016-11-25 23:37:43 +00:00
|
|
|
|
/* Filter out ANSI escape codes from the given string. If ‘nixOnly’ is
|
2014-08-20 14:01:16 +00:00
|
|
|
|
set, only filter escape codes generated by Nixpkgs' stdenv (used to
|
|
|
|
|
denote nesting etc.). */
|
|
|
|
|
string filterANSIEscapes(const string & s, bool nixOnly = false);
|
|
|
|
|
|
|
|
|
|
|
2015-02-09 14:09:39 +00:00
|
|
|
|
/* Base64 encoding/decoding. */
|
|
|
|
|
string base64Encode(const string & s);
|
|
|
|
|
string base64Decode(const string & s);
|
|
|
|
|
|
|
|
|
|
|
Eliminate the "store" global variable
Also, move a few free-standing functions into StoreAPI and Derivation.
Also, introduce a non-nullable smart pointer, ref<T>, which is just a
wrapper around std::shared_ptr ensuring that the pointer is never
null. (For reference-counted values, this is better than passing a
"T&", because the latter doesn't maintain the refcount. Usually, the
caller will have a shared_ptr keeping the value alive, but that's not
always the case, e.g., when passing a reference to a std::thread via
std::bind.)
2016-02-04 13:28:26 +00:00
|
|
|
|
/* Get a value for the specified key from an associate container, or a
|
|
|
|
|
default value if the key doesn't exist. */
|
|
|
|
|
template <class T>
|
|
|
|
|
string get(const T & map, const string & key, const string & def = "")
|
|
|
|
|
{
|
|
|
|
|
auto i = map.find(key);
|
|
|
|
|
return i == map.end() ? def : i->second;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2016-11-25 23:37:43 +00:00
|
|
|
|
/* Call ‘failure’ with the current exception as argument. If ‘failure’
|
2016-09-16 16:54:14 +00:00
|
|
|
|
throws an exception, abort the program. */
|
2016-09-20 15:49:31 +00:00
|
|
|
|
void callFailure(const std::function<void(std::exception_ptr exc)> & failure,
|
|
|
|
|
std::exception_ptr exc = std::current_exception());
|
2016-09-16 16:54:14 +00:00
|
|
|
|
|
|
|
|
|
|
2016-11-25 23:37:43 +00:00
|
|
|
|
/* Evaluate the function ‘f’. If it returns a value, call ‘success’
|
|
|
|
|
with that value as its argument. If it or ‘success’ throws an
|
|
|
|
|
exception, call ‘failure’. If ‘failure’ throws an exception, abort
|
2016-09-16 16:54:14 +00:00
|
|
|
|
the program. */
|
|
|
|
|
template<class T>
|
|
|
|
|
void sync2async(
|
|
|
|
|
const std::function<void(T)> & success,
|
|
|
|
|
const std::function<void(std::exception_ptr exc)> & failure,
|
|
|
|
|
const std::function<T()> & f)
|
|
|
|
|
{
|
|
|
|
|
try {
|
|
|
|
|
success(f());
|
|
|
|
|
} catch (...) {
|
|
|
|
|
callFailure(failure);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2016-11-25 23:37:43 +00:00
|
|
|
|
/* Call the function ‘success’. If it throws an exception, call
|
|
|
|
|
‘failure’. If that throws an exception, abort the program. */
|
2016-09-16 16:54:14 +00:00
|
|
|
|
template<class T>
|
|
|
|
|
void callSuccess(
|
|
|
|
|
const std::function<void(T)> & success,
|
|
|
|
|
const std::function<void(std::exception_ptr exc)> & failure,
|
|
|
|
|
T && arg)
|
|
|
|
|
{
|
|
|
|
|
try {
|
|
|
|
|
success(arg);
|
|
|
|
|
} catch (...) {
|
|
|
|
|
callFailure(failure);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2016-11-17 16:55:41 +00:00
|
|
|
|
/* A variant of std::istringstream that doesn't copy its string
|
2016-11-16 15:21:30 +00:00
|
|
|
|
argument. This is useful for large strings. The caller must ensure
|
|
|
|
|
that the string object is not destroyed while it's referenced by
|
|
|
|
|
this object. */
|
2016-11-17 15:10:32 +00:00
|
|
|
|
class istringbuf_nocopy : public std::streambuf
|
2016-11-16 15:21:30 +00:00
|
|
|
|
{
|
2016-11-17 15:10:32 +00:00
|
|
|
|
const std::string & s;
|
2016-11-17 16:48:10 +00:00
|
|
|
|
decltype(s.size()) off;
|
|
|
|
|
decltype(s.size()) size;
|
2016-11-17 15:10:32 +00:00
|
|
|
|
public:
|
2016-11-17 16:48:10 +00:00
|
|
|
|
istringbuf_nocopy(const std::string & s) : s{s}, off{0}, size{s.size()}
|
2016-11-17 15:10:32 +00:00
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
int_type underflow()
|
|
|
|
|
{
|
2016-11-17 16:48:10 +00:00
|
|
|
|
if (off == size)
|
2016-11-17 15:10:32 +00:00
|
|
|
|
return traits_type::eof();
|
|
|
|
|
return traits_type::to_int_type(s[off]);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int_type uflow()
|
|
|
|
|
{
|
2016-11-17 16:48:10 +00:00
|
|
|
|
if (off == size)
|
2016-11-17 15:10:32 +00:00
|
|
|
|
return traits_type::eof();
|
|
|
|
|
return traits_type::to_int_type(s[off++]);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int_type pbackfail(int_type ch)
|
|
|
|
|
{
|
|
|
|
|
if (off == 0 || (ch != traits_type::eof() && ch != s[off - 1]))
|
|
|
|
|
return traits_type::eof();
|
|
|
|
|
|
|
|
|
|
return traits_type::to_int_type(s[--off]);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
std::streamsize showmanyc()
|
2016-11-16 15:21:30 +00:00
|
|
|
|
{
|
2016-11-17 16:48:10 +00:00
|
|
|
|
return size - off;
|
2016-11-16 15:21:30 +00:00
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
2016-11-17 16:48:10 +00:00
|
|
|
|
struct istringstream_nocopy : public std::iostream
|
2016-11-17 15:10:32 +00:00
|
|
|
|
{
|
|
|
|
|
istringbuf_nocopy buf;
|
2016-11-17 16:48:10 +00:00
|
|
|
|
istringstream_nocopy(const std::string & s) : std::iostream(&buf), buf(s) {};
|
2016-11-17 15:10:32 +00:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
2006-09-04 21:06:23 +00:00
|
|
|
|
}
|