2012-07-18 21:59:03 +03:00
|
|
|
#pragma once
|
2003-04-04 19:14:56 +03:00
|
|
|
|
2006-09-05 00:06:23 +03:00
|
|
|
#include "types.hh"
|
2020-04-24 23:57:51 +03:00
|
|
|
#include "error.hh"
|
2016-04-25 16:26:07 +03:00
|
|
|
#include "logging.hh"
|
2020-03-27 18:55:09 +02:00
|
|
|
#include "ansicolor.hh"
|
2003-04-08 15:00:51 +03:00
|
|
|
|
2003-10-22 13:48:22 +03:00
|
|
|
#include <sys/types.h>
|
2010-12-13 15:32:58 +02:00
|
|
|
#include <sys/stat.h>
|
2003-10-22 13:48:22 +03:00
|
|
|
#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>
|
2016-09-20 16:39:08 +03:00
|
|
|
|
2021-12-31 04:16:59 +02:00
|
|
|
#include <boost/lexical_cast.hpp>
|
|
|
|
|
2021-12-02 00:06:15 +02:00
|
|
|
#include <atomic>
|
2014-07-10 17:50:51 +03:00
|
|
|
#include <functional>
|
2016-09-20 16:39:08 +03:00
|
|
|
#include <map>
|
2017-03-08 23:24:10 +02:00
|
|
|
#include <sstream>
|
2019-02-12 14:43:32 +02:00
|
|
|
#include <optional>
|
2009-11-24 14:56:26 +02:00
|
|
|
|
2015-11-07 05:51:33 +02: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 16:55:12 +03:00
|
|
|
|
2006-09-05 00:06:23 +03:00
|
|
|
namespace nix {
|
2003-10-07 17:37:41 +03:00
|
|
|
|
2018-03-16 17:59:31 +02:00
|
|
|
struct Sink;
|
|
|
|
struct Source;
|
|
|
|
|
2003-10-07 17:37:41 +03:00
|
|
|
|
2019-05-03 11:44:32 +03:00
|
|
|
/* The system for which Nix is compiled. */
|
|
|
|
extern const std::string nativeSystem;
|
|
|
|
|
|
|
|
|
2004-05-12 12:35:51 +03:00
|
|
|
/* Return an environment variable. */
|
2019-11-22 17:06:44 +02:00
|
|
|
std::optional<std::string> getEnv(const std::string & key);
|
2004-05-12 12:35:51 +03:00
|
|
|
|
2023-03-01 21:01:36 +02:00
|
|
|
/* Return a non empty environment variable. Returns nullopt if the env
|
|
|
|
variable is set to "" */
|
|
|
|
std::optional<std::string> getEnvNonEmpty(const std::string & key);
|
|
|
|
|
2016-09-20 16:39:08 +03:00
|
|
|
/* Get the entire environment. */
|
|
|
|
std::map<std::string, std::string> getEnv();
|
|
|
|
|
2018-02-26 19:29:40 +02:00
|
|
|
/* Clear the environment. */
|
|
|
|
void clearEnv();
|
|
|
|
|
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. */
|
2020-01-21 17:27:53 +02:00
|
|
|
Path absPath(Path path,
|
2022-01-21 18:55:51 +02:00
|
|
|
std::optional<PathView> dir = {},
|
2020-01-21 17:27:53 +02:00
|
|
|
bool resolveSymlinks = false);
|
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
|
2006-01-08 19:16:03 +02:00
|
|
|
double or trailing slashes. Optionally resolves all symlink
|
|
|
|
components such that each component of the resulting path is *not*
|
|
|
|
a symbolic link. */
|
2022-01-12 17:02:29 +02:00
|
|
|
Path canonPath(PathView path, bool resolveSymlinks = false);
|
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
|
2020-05-08 16:03:44 +03:00
|
|
|
immediate child thereof (e.g., `/foo'), this means `/'
|
|
|
|
is returned.*/
|
2022-01-21 18:55:51 +02:00
|
|
|
Path dirOf(const PathView 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
|
2020-05-08 16:07:40 +03:00
|
|
|
following the final `/' (trailing slashes are removed). */
|
2019-12-05 20:11:09 +02:00
|
|
|
std::string_view baseNameOf(std::string_view path);
|
2003-04-08 15:00:51 +03:00
|
|
|
|
2022-02-19 15:26:34 +02:00
|
|
|
/* Perform tilde expansion on a path. */
|
|
|
|
std::string expandTilde(std::string_view path);
|
|
|
|
|
2021-12-02 15:16:05 +02:00
|
|
|
/* Check whether 'path' is a descendant of 'dir'. Both paths must be
|
|
|
|
canonicalized. */
|
|
|
|
bool isInDir(std::string_view path, std::string_view dir);
|
2013-07-12 15:01:25 +03:00
|
|
|
|
2021-12-02 15:16:05 +02:00
|
|
|
/* Check whether 'path' is equal to 'dir' or a descendant of
|
|
|
|
'dir'. Both paths must be canonicalized. */
|
|
|
|
bool isDirOrInDir(std::string_view path, std::string_view dir);
|
2018-01-16 19:50:38 +02:00
|
|
|
|
2010-12-13 15:32:58 +02:00
|
|
|
/* Get status of `path'. */
|
2022-02-18 14:26:40 +02:00
|
|
|
struct stat stat(const Path & path);
|
2010-12-13 15:32:58 +02:00
|
|
|
struct stat lstat(const Path & path);
|
|
|
|
|
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);
|
|
|
|
|
2005-02-01 15:48:46 +02:00
|
|
|
bool isLink(const Path & path);
|
|
|
|
|
2003-11-19 19:27:16 +02:00
|
|
|
/* Read the contents of a directory. The entries `.' and `..' are
|
|
|
|
removed. */
|
2014-08-01 17:37:47 +03:00
|
|
|
struct DirEntry
|
|
|
|
{
|
2022-02-25 17:00:00 +02:00
|
|
|
std::string name;
|
2014-08-01 17:37:47 +03:00
|
|
|
ino_t ino;
|
|
|
|
unsigned char type; // one of DT_*
|
2022-02-25 17:00:00 +02:00
|
|
|
DirEntry(std::string name, ino_t ino, unsigned char type)
|
|
|
|
: name(std::move(name)), ino(ino), type(type) { }
|
2014-08-01 17:37:47 +03:00
|
|
|
};
|
|
|
|
|
2022-02-21 17:32:34 +02:00
|
|
|
typedef std::vector<DirEntry> DirEntries;
|
2014-08-01 17:37:47 +03:00
|
|
|
|
|
|
|
DirEntries readDirectory(const Path & path);
|
2003-11-19 19:27:16 +02:00
|
|
|
|
2014-10-03 23:37:51 +03:00
|
|
|
unsigned char getFileType(const Path & path);
|
|
|
|
|
2005-02-02 00:07:48 +02:00
|
|
|
/* Read the contents of a file into a string. */
|
2022-02-25 17:00:00 +02:00
|
|
|
std::string readFile(int fd);
|
|
|
|
std::string readFile(const Path & path);
|
2018-03-28 00:12:31 +03:00
|
|
|
void readFile(const Path & path, Sink & sink);
|
2005-02-02 00:07:48 +02:00
|
|
|
|
2005-02-09 11:50:29 +02:00
|
|
|
/* Write a string to a file. */
|
2022-09-19 21:15:31 +03:00
|
|
|
void writeFile(const Path & path, std::string_view s, mode_t mode = 0666, bool sync = false);
|
2005-02-09 11:50:29 +02:00
|
|
|
|
2022-09-19 21:15:31 +03:00
|
|
|
void writeFile(const Path & path, Source & source, mode_t mode = 0666, bool sync = false);
|
|
|
|
|
|
|
|
/* Flush a file's parent directory to disk */
|
|
|
|
void syncParent(const Path & path);
|
2018-03-28 14:32:44 +03:00
|
|
|
|
2009-03-28 21:29:55 +02:00
|
|
|
/* Read a line from a file descriptor. */
|
2022-02-25 17:00:00 +02:00
|
|
|
std::string readLine(int fd);
|
2009-03-28 21:29:55 +02:00
|
|
|
|
|
|
|
/* Write a line to a file descriptor. */
|
2022-02-25 17:00:00 +02:00
|
|
|
void writeLine(int fd, std::string s);
|
2009-03-28 21:29:55 +02:00
|
|
|
|
2003-06-23 17:40:49 +03:00
|
|
|
/* Delete a path; i.e., in the case of a directory, it is deleted
|
2016-02-24 18:44:12 +02: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 17:37:41 +03:00
|
|
|
void deletePath(const Path & path);
|
2003-08-22 23:12:44 +03:00
|
|
|
|
2020-07-30 14:10:49 +03:00
|
|
|
void deletePath(const Path & path, uint64_t & bytesFreed);
|
2005-12-15 23:11:39 +02:00
|
|
|
|
2019-10-09 20:21:07 +03:00
|
|
|
std::string getUserName();
|
|
|
|
|
2022-04-13 11:26:50 +03:00
|
|
|
/* Return the given user's home directory from /etc/passwd. */
|
|
|
|
Path getHomeOf(uid_t userId);
|
|
|
|
|
2017-05-05 17:40:12 +03:00
|
|
|
/* Return $HOME or the user's home directory from /etc/passwd. */
|
|
|
|
Path getHome();
|
|
|
|
|
2017-04-20 15:58:16 +03:00
|
|
|
/* Return $XDG_CACHE_HOME or $HOME/.cache. */
|
2016-04-20 15:12:38 +03:00
|
|
|
Path getCacheDir();
|
|
|
|
|
2017-04-20 15:58:16 +03:00
|
|
|
/* Return $XDG_CONFIG_HOME or $HOME/.config. */
|
|
|
|
Path getConfigDir();
|
|
|
|
|
2018-10-25 14:00:21 +03:00
|
|
|
/* Return the directories to search for user configuration files */
|
|
|
|
std::vector<Path> getConfigDirs();
|
|
|
|
|
2017-04-25 19:56:29 +03:00
|
|
|
/* Return $XDG_DATA_HOME or $HOME/.local/share. */
|
|
|
|
Path getDataDir();
|
|
|
|
|
2022-06-22 23:43:53 +03:00
|
|
|
/* Return the path of the current executable. */
|
|
|
|
std::optional<Path> getSelfExe();
|
|
|
|
|
2021-11-17 22:35:21 +02:00
|
|
|
/* Return $XDG_STATE_HOME or $HOME/.local/state. */
|
|
|
|
Path getStateDir();
|
|
|
|
|
|
|
|
/* Create the Nix state directory and return the path to it. */
|
|
|
|
Path createNixStateDir();
|
|
|
|
|
2007-10-27 03:46:59 +03: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);
|
2022-06-22 23:43:53 +03:00
|
|
|
inline Paths createDirs(PathView path)
|
|
|
|
{
|
2022-01-21 18:55:51 +02:00
|
|
|
return createDirs(Path(path));
|
|
|
|
}
|
2005-03-24 19:46:38 +02:00
|
|
|
|
2014-02-28 00:17:53 +02:00
|
|
|
/* Create a symlink. */
|
2019-05-28 23:35:41 +03:00
|
|
|
void createSymlink(const Path & target, const Path & link,
|
|
|
|
std::optional<time_t> mtime = {});
|
2014-02-28 00:17:53 +02:00
|
|
|
|
2015-04-09 12:42:04 +03:00
|
|
|
/* Atomically create or replace a symlink. */
|
2019-05-28 23:35:41 +03:00
|
|
|
void replaceSymlink(const Path & target, const Path & link,
|
|
|
|
std::optional<time_t> mtime = {});
|
2015-04-09 12:42:04 +03:00
|
|
|
|
2022-04-13 15:10:36 +03:00
|
|
|
void renameFile(const Path & src, const Path & dst);
|
2022-03-17 16:28:46 +02:00
|
|
|
|
2022-04-13 15:19:42 +03:00
|
|
|
/**
|
|
|
|
* Similar to 'renameFile', but fallback to a copy+remove if `src` and `dst`
|
|
|
|
* are on a different filesystem.
|
|
|
|
*
|
|
|
|
* Beware that this might not be atomic because of the copy that happens behind
|
|
|
|
* the scenes
|
|
|
|
*/
|
|
|
|
void moveFile(const Path & src, const Path & dst);
|
|
|
|
|
2003-11-22 17:58:34 +02:00
|
|
|
|
2003-07-21 00:11:43 +03:00
|
|
|
/* Wrappers arount read()/write() that read/write exactly the
|
|
|
|
requested number of bytes. */
|
2020-12-02 15:10:56 +02:00
|
|
|
void readFull(int fd, char * buf, size_t count);
|
2020-12-02 13:43:52 +02:00
|
|
|
void writeFull(int fd, std::string_view s, bool allowInterrupts = true);
|
2003-07-21 00:11:43 +03:00
|
|
|
|
2019-11-10 18:14:26 +02:00
|
|
|
MakeError(EndOfFile, Error);
|
2006-12-04 19:17:13 +02:00
|
|
|
|
2003-07-21 00:11:43 +03:00
|
|
|
|
2006-07-20 15:17:25 +03:00
|
|
|
/* Read a file descriptor until EOF occurs. */
|
2022-02-25 17:00:00 +02:00
|
|
|
std::string drainFD(int fd, bool block = true, const size_t reserveSize=0);
|
2006-07-20 15:17:25 +03:00
|
|
|
|
2018-03-20 16:17:59 +02:00
|
|
|
void drainFD(int fd, Sink & sink, bool block = true);
|
2006-07-20 15:17:25 +03:00
|
|
|
|
2022-07-19 09:09:46 +03:00
|
|
|
/* If cgroups are active, attempt to calculate the number of CPUs available.
|
|
|
|
If cgroups are unavailable or if cpu.max is set to "max", return 0. */
|
|
|
|
unsigned int getMaxCPU();
|
2006-07-20 15:17:25 +03:00
|
|
|
|
2003-10-22 13:48:22 +03:00
|
|
|
/* Automatic cleanup of resources. */
|
|
|
|
|
2006-12-04 19:17:13 +02:00
|
|
|
|
2003-10-22 13:48:22 +03:00
|
|
|
class AutoDelete
|
|
|
|
{
|
2006-12-13 01:05:01 +02:00
|
|
|
Path path;
|
2003-10-22 13:48:22 +03:00
|
|
|
bool del;
|
2013-01-03 14:00:46 +02:00
|
|
|
bool recursive;
|
2003-10-22 13:48:22 +03:00
|
|
|
public:
|
2015-11-16 12:53:10 +02:00
|
|
|
AutoDelete();
|
2007-10-27 03:46:59 +03:00
|
|
|
AutoDelete(const Path & p, bool recursive = true);
|
2003-10-22 13:48:22 +03:00
|
|
|
~AutoDelete();
|
|
|
|
void cancel();
|
2015-11-16 12:53:10 +02:00
|
|
|
void reset(const Path & p, bool recursive = true);
|
2015-10-01 17:47:43 +03:00
|
|
|
operator Path() const { return path; }
|
2022-01-21 18:55:51 +02:00
|
|
|
operator PathView() const { return path; }
|
2003-10-22 13:48:22 +03:00
|
|
|
};
|
|
|
|
|
2004-06-22 12:51:44 +03:00
|
|
|
|
2003-10-22 13:48:22 +03:00
|
|
|
class AutoCloseFD
|
|
|
|
{
|
|
|
|
int fd;
|
|
|
|
public:
|
|
|
|
AutoCloseFD();
|
|
|
|
AutoCloseFD(int fd);
|
2016-07-11 22:44:44 +03:00
|
|
|
AutoCloseFD(const AutoCloseFD & fd) = delete;
|
|
|
|
AutoCloseFD(AutoCloseFD&& fd);
|
2003-10-22 13:48:22 +03:00
|
|
|
~AutoCloseFD();
|
2016-07-11 22:44:44 +03:00
|
|
|
AutoCloseFD& operator =(const AutoCloseFD & fd) = delete;
|
|
|
|
AutoCloseFD& operator =(AutoCloseFD&& fd);
|
|
|
|
int get() const;
|
|
|
|
explicit operator bool() const;
|
|
|
|
int release();
|
2021-04-07 13:21:31 +03:00
|
|
|
void close();
|
2022-09-19 21:15:31 +03:00
|
|
|
void fsync();
|
2004-06-15 16:49:42 +03:00
|
|
|
};
|
|
|
|
|
2004-06-22 12:51:44 +03:00
|
|
|
|
2020-03-30 20:14:17 +03:00
|
|
|
/* 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 temporary file, returning a file handle and its path. */
|
|
|
|
std::pair<AutoCloseFD, Path> createTempFile(const Path & prefix = "nix");
|
|
|
|
|
|
|
|
|
2004-06-15 16:49:42 +03:00
|
|
|
class Pipe
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
AutoCloseFD readSide, writeSide;
|
|
|
|
void create();
|
2021-04-07 13:21:31 +03:00
|
|
|
void close();
|
2003-10-22 13:48:22 +03:00
|
|
|
};
|
|
|
|
|
2004-06-22 12:51:44 +03:00
|
|
|
|
2017-01-16 23:39:27 +02:00
|
|
|
struct DIRDeleter
|
2003-10-22 13:48:22 +03:00
|
|
|
{
|
2017-01-16 23:39:27 +02:00
|
|
|
void operator()(DIR * dir) const {
|
|
|
|
closedir(dir);
|
|
|
|
}
|
2003-10-22 13:48:22 +03:00
|
|
|
};
|
|
|
|
|
2017-01-16 23:39:27 +02:00
|
|
|
typedef std::unique_ptr<DIR, DIRDeleter> AutoCloseDir;
|
|
|
|
|
2003-10-22 13:48:22 +03:00
|
|
|
|
2004-06-22 12:51:44 +03:00
|
|
|
class Pid
|
|
|
|
{
|
2017-01-19 17:58:39 +02:00
|
|
|
pid_t pid = -1;
|
|
|
|
bool separatePG = false;
|
|
|
|
int killSignal = SIGKILL;
|
2004-06-22 12:51:44 +03:00
|
|
|
public:
|
|
|
|
Pid();
|
2014-07-10 17:50:51 +03:00
|
|
|
Pid(pid_t pid);
|
2004-06-22 12:51:44 +03:00
|
|
|
~Pid();
|
|
|
|
void operator =(pid_t pid);
|
|
|
|
operator pid_t();
|
2017-03-16 11:52:28 +02:00
|
|
|
int kill();
|
2017-01-19 17:58:39 +02:00
|
|
|
int wait();
|
|
|
|
|
2004-06-22 12:51:44 +03:00
|
|
|
void setSeparatePG(bool separatePG);
|
2007-03-19 14:48:45 +02:00
|
|
|
void setKillSignal(int signal);
|
2016-10-12 16:49:37 +03:00
|
|
|
pid_t release();
|
2004-06-22 12:51:44 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2006-12-07 02:16:07 +02:00
|
|
|
/* Kill all processes running under the specified uid by sending them
|
|
|
|
a SIGKILL. */
|
|
|
|
void killUser(uid_t uid);
|
|
|
|
|
|
|
|
|
2014-07-10 17:50:51 +03:00
|
|
|
/* Fork a process that runs the given function, and return the child
|
|
|
|
pid to the caller. */
|
2014-12-10 17:35:42 +02:00
|
|
|
struct ProcessOptions
|
|
|
|
{
|
2022-02-25 17:00:00 +02:00
|
|
|
std::string errorPrefix = "";
|
2017-01-19 16:15:45 +02:00
|
|
|
bool dieWithParent = true;
|
|
|
|
bool runExitHandlers = false;
|
2021-10-06 14:54:59 +03:00
|
|
|
bool allowVfork = false;
|
2023-02-10 15:38:14 +02:00
|
|
|
int cloneFlags = 0; // use clone() with the specified flags (Linux only)
|
2014-12-10 17:35:42 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
pid_t startProcess(std::function<void()> fun, const ProcessOptions & options = ProcessOptions());
|
2014-07-10 17:50:51 +03:00
|
|
|
|
|
|
|
|
2006-07-20 15:17:25 +03:00
|
|
|
/* Run a program and return its stdout in a string (i.e., like the
|
|
|
|
shell backtick operator). */
|
2022-02-25 17:00:00 +02:00
|
|
|
std::string runProgram(Path program, bool searchPath = false,
|
2017-03-15 15:40:47 +02:00
|
|
|
const Strings & args = Strings(),
|
2019-02-12 14:43:32 +02:00
|
|
|
const std::optional<std::string> & input = {});
|
2006-07-20 15:17:25 +03:00
|
|
|
|
2017-11-01 19:43:11 +02:00
|
|
|
struct RunOptions
|
|
|
|
{
|
2021-09-14 00:22:09 +03:00
|
|
|
Path program;
|
|
|
|
bool searchPath = true;
|
|
|
|
Strings args;
|
2019-05-11 23:35:53 +03:00
|
|
|
std::optional<uid_t> uid;
|
|
|
|
std::optional<uid_t> gid;
|
|
|
|
std::optional<Path> chdir;
|
2019-07-11 21:23:03 +03:00
|
|
|
std::optional<std::map<std::string, std::string>> environment;
|
2019-02-12 14:43:32 +02:00
|
|
|
std::optional<std::string> input;
|
2018-03-19 18:09:52 +02:00
|
|
|
Source * standardIn = nullptr;
|
|
|
|
Sink * standardOut = nullptr;
|
2019-07-11 21:23:03 +03:00
|
|
|
bool mergeStderrToStdout = false;
|
2017-11-01 19:43:11 +02:00
|
|
|
};
|
|
|
|
|
2021-09-14 00:22:09 +03:00
|
|
|
std::pair<int, std::string> runProgram(RunOptions && options);
|
2017-11-01 19:43:11 +02:00
|
|
|
|
2018-03-16 17:59:31 +02:00
|
|
|
void runProgram2(const RunOptions & options);
|
|
|
|
|
2017-11-01 19:43:11 +02:00
|
|
|
|
2021-04-07 14:40:13 +03:00
|
|
|
/* Change the stack size. */
|
|
|
|
void setStackSize(size_t stackSize);
|
|
|
|
|
|
|
|
|
2021-04-07 14:10:02 +03:00
|
|
|
/* Restore the original inherited Unix process context (such as signal
|
2021-12-22 16:49:51 +02:00
|
|
|
masks, stack size). */
|
2021-10-15 17:25:49 +03:00
|
|
|
void restoreProcessContext(bool restoreMounts = true);
|
|
|
|
|
|
|
|
/* Save the current mount namespace. Ignored if called more than
|
|
|
|
once. */
|
|
|
|
void saveMountNamespace();
|
|
|
|
|
|
|
|
/* Restore the mount namespace saved by saveMountNamespace(). Ignored
|
|
|
|
if saveMountNamespace() was never called. */
|
|
|
|
void restoreMountNamespace();
|
2021-04-07 14:10:02 +03:00
|
|
|
|
2021-12-16 22:26:22 +02:00
|
|
|
/* Cause this thread to not share any FS attributes with the main
|
|
|
|
thread, because this causes setns() in restoreMountNamespace() to
|
|
|
|
fail. */
|
|
|
|
void unshareFilesystem();
|
|
|
|
|
2021-04-07 14:10:02 +03:00
|
|
|
|
2016-09-21 17:21:47 +03:00
|
|
|
class ExecError : public Error
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
int status;
|
|
|
|
|
|
|
|
template<typename... Args>
|
2019-12-05 19:23:32 +02:00
|
|
|
ExecError(int status, const Args & ... args)
|
2016-09-21 17:21:47 +03:00
|
|
|
: Error(args...), status(status)
|
|
|
|
{ }
|
|
|
|
};
|
2014-05-21 18:19:36 +03:00
|
|
|
|
2014-12-12 16:01:16 +02: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 11:50:55 +03:00
|
|
|
std::vector<char *> stringsToCharPtrs(const Strings & ss);
|
2014-12-12 16:01:16 +02:00
|
|
|
|
2017-08-09 17:22:05 +03:00
|
|
|
/* Close all file descriptors except those listed in the given set.
|
|
|
|
Good practice in child processes. */
|
2022-02-21 17:28:23 +02:00
|
|
|
void closeMostFDs(const std::set<int> & exceptions);
|
2008-08-02 15:54:35 +03:00
|
|
|
|
2012-03-05 21:29:00 +02:00
|
|
|
/* Set the close-on-exec flag for the given file descriptor. */
|
|
|
|
void closeOnExec(int fd);
|
|
|
|
|
2006-07-20 15:17:25 +03:00
|
|
|
|
2004-01-15 22:23:55 +02:00
|
|
|
/* User interruption. */
|
|
|
|
|
2021-11-24 15:50:08 +02:00
|
|
|
extern std::atomic<bool> _isInterrupted;
|
2004-01-15 22:23:55 +02:00
|
|
|
|
2017-09-08 16:31:24 +03:00
|
|
|
extern thread_local std::function<bool()> interruptCheck;
|
|
|
|
|
2017-04-21 17:28:10 +03:00
|
|
|
void setInterruptThrown();
|
2016-03-29 16:08:24 +03:00
|
|
|
|
2004-01-15 22:23:55 +02:00
|
|
|
void _interrupted();
|
|
|
|
|
|
|
|
void inline checkInterrupt()
|
|
|
|
{
|
2017-09-08 16:31:24 +03:00
|
|
|
if (_isInterrupted || (interruptCheck && interruptCheck()))
|
|
|
|
_interrupted();
|
2004-01-15 22:23:55 +02:00
|
|
|
}
|
|
|
|
|
2019-11-10 18:14:26 +02:00
|
|
|
MakeError(Interrupted, BaseError);
|
2006-12-04 19:17:13 +02:00
|
|
|
|
2004-01-15 22:23:55 +02:00
|
|
|
|
2019-11-10 18:14:26 +02:00
|
|
|
MakeError(FormatError, Error);
|
2016-07-19 01:50:27 +03:00
|
|
|
|
|
|
|
|
2005-09-22 18:43:22 +03:00
|
|
|
/* String tokenizer. */
|
2022-01-12 17:02:29 +02:00
|
|
|
template<class C> C tokenizeString(std::string_view s, std::string_view separators = " \t\n\r");
|
2005-09-22 18:43:22 +03:00
|
|
|
|
|
|
|
|
2010-08-27 16:18:13 +03:00
|
|
|
/* Concatenate the given strings with a separator between the
|
|
|
|
elements. */
|
2019-05-02 22:09:52 +03:00
|
|
|
template<class C>
|
2022-02-25 17:00:00 +02:00
|
|
|
std::string concatStringsSep(const std::string_view sep, const C & ss)
|
2019-05-02 22:09:52 +03:00
|
|
|
{
|
2022-01-12 17:02:29 +02:00
|
|
|
size_t size = 0;
|
|
|
|
// need a cast to string_view since this is also called with Symbols
|
|
|
|
for (const auto & s : ss) size += sep.size() + std::string_view(s).size();
|
2022-02-25 17:00:00 +02:00
|
|
|
std::string s;
|
2022-01-12 17:02:29 +02:00
|
|
|
s.reserve(size);
|
2019-05-02 22:09:52 +03:00
|
|
|
for (auto & i : ss) {
|
|
|
|
if (s.size() != 0) s += sep;
|
|
|
|
s += i;
|
|
|
|
}
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
2022-01-12 17:02:29 +02:00
|
|
|
template<class ... Parts>
|
|
|
|
auto concatStrings(Parts && ... parts)
|
2022-02-25 17:00:00 +02:00
|
|
|
-> std::enable_if_t<(... && std::is_convertible_v<Parts, std::string_view>), std::string>
|
2022-01-12 17:02:29 +02:00
|
|
|
{
|
|
|
|
std::string_view views[sizeof...(parts)] = { parts... };
|
|
|
|
return concatStringsSep({}, views);
|
|
|
|
}
|
|
|
|
|
2019-05-02 22:09:52 +03:00
|
|
|
|
|
|
|
/* Add quotes around a collection of strings. */
|
|
|
|
template<class C> Strings quoteStrings(const C & c)
|
|
|
|
{
|
|
|
|
Strings res;
|
|
|
|
for (auto & s : c)
|
|
|
|
res.push_back("'" + s + "'");
|
|
|
|
return res;
|
|
|
|
}
|
2010-08-27 16:18:13 +03:00
|
|
|
|
|
|
|
|
2021-01-21 01:49:29 +02:00
|
|
|
/* Remove trailing whitespace from a string. FIXME: return
|
|
|
|
std::string_view. */
|
2022-02-25 17:00:00 +02:00
|
|
|
std::string chomp(std::string_view s);
|
2012-08-01 18:19:24 +03:00
|
|
|
|
|
|
|
|
2015-04-09 12:42:04 +03:00
|
|
|
/* Remove whitespace from the start and end of a string. */
|
2022-02-25 17:00:00 +02:00
|
|
|
std::string trim(std::string_view s, std::string_view whitespace = " \n\r\t");
|
2015-04-09 12:42:04 +03:00
|
|
|
|
|
|
|
|
2015-06-17 17:20:11 +03:00
|
|
|
/* Replace all occurrences of a string inside another string. */
|
2022-02-25 17:00:00 +02:00
|
|
|
std::string replaceStrings(
|
|
|
|
std::string s,
|
|
|
|
std::string_view from,
|
|
|
|
std::string_view to);
|
2015-06-17 17:20:11 +03:00
|
|
|
|
|
|
|
|
2022-02-25 17:00:00 +02:00
|
|
|
std::string rewriteStrings(std::string s, const StringMap & rewrites);
|
2018-03-30 01:56:13 +03:00
|
|
|
|
|
|
|
|
2004-06-22 11:50:25 +03:00
|
|
|
/* Convert the exit status of a child as returned by wait() into an
|
|
|
|
error string. */
|
2022-02-25 17:00:00 +02:00
|
|
|
std::string statusToString(int status);
|
2004-06-22 11:50:25 +03:00
|
|
|
|
2004-06-22 14:03:41 +03:00
|
|
|
bool statusOk(int status);
|
|
|
|
|
2004-06-22 11:50:25 +03:00
|
|
|
|
2004-09-10 16:32:08 +03:00
|
|
|
/* Parse a string into an integer. */
|
2021-01-08 13:22:21 +02:00
|
|
|
template<class N>
|
2021-12-31 04:16:59 +02:00
|
|
|
std::optional<N> string2Int(const std::string_view s)
|
2009-11-24 14:26:25 +02:00
|
|
|
{
|
2021-01-08 13:22:21 +02:00
|
|
|
if (s.substr(0, 1) == "-" && !std::numeric_limits<N>::is_signed)
|
2021-01-08 13:51:19 +02:00
|
|
|
return std::nullopt;
|
2021-12-31 04:16:59 +02:00
|
|
|
try {
|
|
|
|
return boost::lexical_cast<N>(s.data(), s.size());
|
|
|
|
} catch (const boost::bad_lexical_cast &) {
|
|
|
|
return std::nullopt;
|
|
|
|
}
|
2021-01-08 13:51:19 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Like string2Int(), but support an optional suffix 'K', 'M', 'G' or
|
|
|
|
'T' denoting a binary unit prefix. */
|
|
|
|
template<class N>
|
2021-12-31 04:16:59 +02:00
|
|
|
N string2IntWithUnitPrefix(std::string_view s)
|
2021-01-08 13:51:19 +02:00
|
|
|
{
|
|
|
|
N multiplier = 1;
|
|
|
|
if (!s.empty()) {
|
|
|
|
char u = std::toupper(*s.rbegin());
|
|
|
|
if (std::isalpha(u)) {
|
|
|
|
if (u == 'K') multiplier = 1ULL << 10;
|
|
|
|
else if (u == 'M') multiplier = 1ULL << 20;
|
|
|
|
else if (u == 'G') multiplier = 1ULL << 30;
|
|
|
|
else if (u == 'T') multiplier = 1ULL << 40;
|
|
|
|
else throw UsageError("invalid unit specifier '%1%'", u);
|
2021-12-31 04:16:59 +02:00
|
|
|
s.remove_suffix(1);
|
2021-01-08 13:51:19 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (auto n = string2Int<N>(s))
|
|
|
|
return *n * multiplier;
|
|
|
|
throw UsageError("'%s' is not an integer", s);
|
2009-11-24 14:26:25 +02:00
|
|
|
}
|
|
|
|
|
2016-01-05 01:40:40 +02:00
|
|
|
/* Parse a string into a float. */
|
2021-01-08 13:22:21 +02:00
|
|
|
template<class N>
|
2021-12-31 04:16:59 +02:00
|
|
|
std::optional<N> string2Float(const std::string_view s)
|
2016-01-05 01:40:40 +02:00
|
|
|
{
|
2021-12-31 04:16:59 +02:00
|
|
|
try {
|
|
|
|
return boost::lexical_cast<N>(s.data(), s.size());
|
|
|
|
} catch (const boost::bad_lexical_cast &) {
|
|
|
|
return std::nullopt;
|
|
|
|
}
|
2016-01-05 01:40:40 +02:00
|
|
|
}
|
|
|
|
|
2004-09-10 16:32:08 +03:00
|
|
|
|
2022-12-07 13:58:58 +02:00
|
|
|
/* Convert a little-endian integer to host order. */
|
|
|
|
template<typename T>
|
|
|
|
T readLittleEndian(unsigned char * p)
|
|
|
|
{
|
|
|
|
T x = 0;
|
2022-12-12 13:40:51 +02:00
|
|
|
for (size_t i = 0; i < sizeof(x); ++i, ++p) {
|
|
|
|
x |= ((T) *p) << (i * 8);
|
|
|
|
}
|
2022-12-07 13:58:58 +02:00
|
|
|
return x;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-04-29 22:04:40 +03:00
|
|
|
/* Return true iff `s' starts with `prefix'. */
|
2020-06-13 00:12:36 +03:00
|
|
|
bool hasPrefix(std::string_view s, std::string_view prefix);
|
2016-04-29 22:04:40 +03:00
|
|
|
|
|
|
|
|
2008-08-25 16:31:57 +03:00
|
|
|
/* Return true iff `s' ends in `suffix'. */
|
2019-12-05 20:11:09 +02:00
|
|
|
bool hasSuffix(std::string_view s, std::string_view suffix);
|
2008-08-25 16:31:57 +03:00
|
|
|
|
|
|
|
|
2016-09-14 15:42:15 +03:00
|
|
|
/* Convert a string to lower case. */
|
|
|
|
std::string toLower(const std::string & s);
|
|
|
|
|
|
|
|
|
2017-10-25 14:01:50 +03:00
|
|
|
/* Escape a string as a shell word. */
|
2022-01-21 18:55:51 +02:00
|
|
|
std::string shellEscape(const std::string_view s);
|
2017-08-25 16:57:49 +03:00
|
|
|
|
|
|
|
|
2010-04-19 16:46:58 +03:00
|
|
|
/* Exception handling in destructors: print an error message, then
|
|
|
|
ignore the exception. */
|
2022-12-02 16:03:40 +02:00
|
|
|
void ignoreException(Verbosity lvl = lvlError);
|
2008-09-17 13:02:55 +03:00
|
|
|
|
|
|
|
|
2014-08-20 17:01:16 +03:00
|
|
|
|
2020-03-24 15:17:10 +02:00
|
|
|
/* Tree formatting. */
|
|
|
|
constexpr char treeConn[] = "├───";
|
|
|
|
constexpr char treeLast[] = "└───";
|
|
|
|
constexpr char treeLine[] = "│ ";
|
|
|
|
constexpr char treeNull[] = " ";
|
|
|
|
|
2021-07-02 03:19:01 +03:00
|
|
|
/* Determine whether ANSI escape sequences are appropriate for the
|
|
|
|
present output. */
|
|
|
|
bool shouldANSI();
|
2020-03-24 15:17:10 +02:00
|
|
|
|
2018-03-15 17:08:07 +02:00
|
|
|
/* Truncate a string to 'width' printable characters. If 'filterAll'
|
|
|
|
is true, all ANSI escape sequences are filtered out. Otherwise,
|
|
|
|
some escape sequences (such as colour setting) are copied but not
|
|
|
|
included in the character count. Also, tabs are expanded to
|
|
|
|
spaces. */
|
2023-03-02 16:44:19 +02:00
|
|
|
std::string filterANSIEscapes(std::string_view s,
|
2018-03-15 17:08:07 +02:00
|
|
|
bool filterAll = false,
|
2018-02-07 16:19:10 +02:00
|
|
|
unsigned int width = std::numeric_limits<unsigned int>::max());
|
2014-08-20 17:01:16 +03:00
|
|
|
|
|
|
|
|
2015-02-09 16:09:39 +02:00
|
|
|
/* Base64 encoding/decoding. */
|
2022-02-25 17:00:00 +02:00
|
|
|
std::string base64Encode(std::string_view s);
|
|
|
|
std::string base64Decode(std::string_view s);
|
2015-02-09 16:09:39 +02:00
|
|
|
|
|
|
|
|
2020-08-20 13:21:46 +03:00
|
|
|
/* Remove common leading whitespace from the lines in the string
|
|
|
|
's'. For example, if every line is indented by at least 3 spaces,
|
|
|
|
then we remove 3 spaces from the start of every line. */
|
|
|
|
std::string stripIndentation(std::string_view s);
|
|
|
|
|
|
|
|
|
2022-12-07 13:58:58 +02:00
|
|
|
/* Get the prefix of 's' up to and excluding the next line break (LF
|
|
|
|
optionally preceded by CR), and the remainder following the line
|
|
|
|
break. */
|
|
|
|
std::pair<std::string_view, std::string_view> getLine(std::string_view s);
|
|
|
|
|
|
|
|
|
2020-06-04 12:14:19 +03:00
|
|
|
/* Get a value for the specified key from an associate container. */
|
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 15:28:26 +02:00
|
|
|
template <class T>
|
2022-05-04 08:44:32 +03:00
|
|
|
const typename T::mapped_type * get(const T & map, const typename T::key_type & key)
|
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 15:28:26 +02:00
|
|
|
{
|
|
|
|
auto i = map.find(key);
|
2022-05-04 08:44:32 +03:00
|
|
|
if (i == map.end()) return nullptr;
|
|
|
|
return &i->second;
|
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 15:28:26 +02:00
|
|
|
}
|
|
|
|
|
2022-05-04 08:44:32 +03:00
|
|
|
template <class T>
|
|
|
|
typename T::mapped_type * get(T & map, const typename T::key_type & key)
|
|
|
|
{
|
|
|
|
auto i = map.find(key);
|
|
|
|
if (i == map.end()) return nullptr;
|
|
|
|
return &i->second;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get a value for the specified key from an associate container, or a default value if the key isn't present. */
|
|
|
|
template <class T>
|
|
|
|
const typename T::mapped_type & getOr(T & map,
|
|
|
|
const typename T::key_type & key,
|
|
|
|
const typename T::mapped_type & defaultValue)
|
|
|
|
{
|
|
|
|
auto i = map.find(key);
|
|
|
|
if (i == map.end()) return defaultValue;
|
|
|
|
return i->second;
|
|
|
|
}
|
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 15:28:26 +02:00
|
|
|
|
2021-08-20 12:18:35 +03:00
|
|
|
/* Remove and return the first item from a container. */
|
|
|
|
template <class T>
|
|
|
|
std::optional<typename T::value_type> remove_begin(T & c)
|
|
|
|
{
|
|
|
|
auto i = c.begin();
|
|
|
|
if (i == c.end()) return {};
|
|
|
|
auto v = std::move(*i);
|
|
|
|
c.erase(i);
|
|
|
|
return v;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-10-14 13:31:21 +03:00
|
|
|
/* Remove and return the first item from a container. */
|
|
|
|
template <class T>
|
|
|
|
std::optional<typename T::value_type> pop(T & c)
|
|
|
|
{
|
|
|
|
if (c.empty()) return {};
|
|
|
|
auto v = std::move(c.front());
|
|
|
|
c.pop();
|
|
|
|
return v;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-03-27 23:16:01 +03:00
|
|
|
template<typename T>
|
2020-09-21 19:40:11 +03:00
|
|
|
class Callback;
|
2016-09-16 19:54:14 +03:00
|
|
|
|
|
|
|
|
2017-01-17 19:21:02 +02:00
|
|
|
/* Start a thread that handles various signals. Also block those signals
|
|
|
|
on the current thread (and thus any threads created by it). */
|
|
|
|
void startSignalHandlerThread();
|
|
|
|
|
|
|
|
struct InterruptCallback
|
|
|
|
{
|
|
|
|
virtual ~InterruptCallback() { };
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Register a function that gets called on SIGINT (in a non-signal
|
|
|
|
context). */
|
|
|
|
std::unique_ptr<InterruptCallback> createInterruptCallback(
|
|
|
|
std::function<void()> callback);
|
|
|
|
|
2017-01-25 14:37:02 +02:00
|
|
|
void triggerInterrupt();
|
|
|
|
|
|
|
|
/* A RAII class that causes the current thread to receive SIGUSR1 when
|
|
|
|
the signal handler thread receives SIGINT. That is, this allows
|
|
|
|
SIGINT to be multiplexed to multiple threads. */
|
|
|
|
struct ReceiveInterrupts
|
|
|
|
{
|
|
|
|
pthread_t target;
|
|
|
|
std::unique_ptr<InterruptCallback> callback;
|
|
|
|
|
|
|
|
ReceiveInterrupts()
|
|
|
|
: target(pthread_self())
|
|
|
|
, callback(createInterruptCallback([&]() { pthread_kill(target, SIGUSR1); }))
|
|
|
|
{ }
|
|
|
|
};
|
|
|
|
|
2017-08-14 23:42:17 +03:00
|
|
|
|
|
|
|
|
|
|
|
/* A RAII helper that increments a counter on construction and
|
|
|
|
decrements it on destruction. */
|
|
|
|
template<typename T>
|
|
|
|
struct MaintainCount
|
|
|
|
{
|
|
|
|
T & counter;
|
|
|
|
long delta;
|
|
|
|
MaintainCount(T & counter, long delta = 1) : counter(counter), delta(delta) { counter += delta; }
|
|
|
|
~MaintainCount() { counter -= delta; }
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2017-08-25 16:57:49 +03:00
|
|
|
/* Return the number of rows and columns of the terminal. */
|
|
|
|
std::pair<unsigned short, unsigned short> getWindowSize();
|
|
|
|
|
|
|
|
|
2017-10-30 20:57:40 +02:00
|
|
|
/* Used in various places. */
|
|
|
|
typedef std::function<bool(const Path & path)> PathFilter;
|
|
|
|
|
|
|
|
extern PathFilter defaultPathFilter;
|
|
|
|
|
2020-10-11 19:38:46 +03:00
|
|
|
/* Common initialisation performed in child processes. */
|
|
|
|
void commonChildInit(Pipe & logPipe);
|
2017-10-30 20:57:40 +02:00
|
|
|
|
2021-08-16 21:03:32 +03:00
|
|
|
/* Create a Unix domain socket. */
|
|
|
|
AutoCloseFD createUnixDomainSocket();
|
|
|
|
|
2018-09-25 13:36:11 +03:00
|
|
|
/* Create a Unix domain socket in listen mode. */
|
|
|
|
AutoCloseFD createUnixDomainSocket(const Path & path, mode_t mode);
|
|
|
|
|
2021-08-24 14:52:55 +03:00
|
|
|
/* Bind a Unix domain socket to a path. */
|
|
|
|
void bind(int fd, const std::string & path);
|
|
|
|
|
|
|
|
/* Connect to a Unix domain socket. */
|
|
|
|
void connect(int fd, const std::string & path);
|
|
|
|
|
2018-09-25 13:36:11 +03:00
|
|
|
|
2020-03-24 15:17:10 +02:00
|
|
|
// A Rust/Python-like enumerate() iterator adapter.
|
|
|
|
// Borrowed from http://reedbeta.com/blog/python-like-enumerate-in-cpp17.
|
|
|
|
template <typename T,
|
|
|
|
typename TIter = decltype(std::begin(std::declval<T>())),
|
|
|
|
typename = decltype(std::end(std::declval<T>()))>
|
|
|
|
constexpr auto enumerate(T && iterable)
|
|
|
|
{
|
|
|
|
struct iterator
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
TIter iter;
|
|
|
|
bool operator != (const iterator & other) const { return iter != other.iter; }
|
|
|
|
void operator ++ () { ++i; ++iter; }
|
|
|
|
auto operator * () const { return std::tie(i, *iter); }
|
|
|
|
};
|
|
|
|
|
|
|
|
struct iterable_wrapper
|
|
|
|
{
|
|
|
|
T iterable;
|
|
|
|
auto begin() { return iterator{ 0, std::begin(iterable) }; }
|
|
|
|
auto end() { return iterator{ 0, std::end(iterable) }; }
|
|
|
|
};
|
|
|
|
|
|
|
|
return iterable_wrapper{ std::forward<T>(iterable) };
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-07-13 01:15:14 +03:00
|
|
|
// C++17 std::visit boilerplate
|
|
|
|
template<class... Ts> struct overloaded : Ts... { using Ts::operator()...; };
|
|
|
|
template<class... Ts> overloaded(Ts...) -> overloaded<Ts...>;
|
|
|
|
|
|
|
|
|
2020-10-06 11:40:49 +03:00
|
|
|
std::string showBytes(uint64_t bytes);
|
|
|
|
|
|
|
|
|
2022-05-09 15:28:27 +03:00
|
|
|
/* Provide an addition operator between strings and string_views
|
|
|
|
inexplicably omitted from the standard library. */
|
|
|
|
inline std::string operator + (const std::string & s1, std::string_view s2)
|
|
|
|
{
|
|
|
|
auto s = s1;
|
|
|
|
s.append(s2);
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline std::string operator + (std::string && s, std::string_view s2)
|
|
|
|
{
|
|
|
|
s.append(s2);
|
2022-06-02 17:48:53 +03:00
|
|
|
return std::move(s);
|
2022-05-09 15:28:27 +03:00
|
|
|
}
|
|
|
|
|
2022-12-07 13:58:58 +02:00
|
|
|
inline std::string operator + (std::string_view s1, const char * s2)
|
|
|
|
{
|
2022-12-12 13:36:19 +02:00
|
|
|
std::string s;
|
|
|
|
s.reserve(s1.size() + strlen(s2));
|
|
|
|
s.append(s1);
|
2022-12-07 13:58:58 +02:00
|
|
|
s.append(s2);
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
2006-09-05 00:06:23 +03:00
|
|
|
}
|