2007-03-30 16:24:35 +03:00
|
|
|
#include "config.h"
|
2006-11-30 19:43:04 +02:00
|
|
|
#include "local-store.hh"
|
2006-09-05 00:06:23 +03:00
|
|
|
#include "util.hh"
|
|
|
|
#include "globals.hh"
|
|
|
|
#include "db.hh"
|
|
|
|
#include "archive.hh"
|
|
|
|
#include "pathlocks.hh"
|
|
|
|
#include "aterm.hh"
|
|
|
|
#include "derivations-ast.hh"
|
2007-02-21 17:45:32 +02:00
|
|
|
#include "worker-protocol.hh"
|
2006-09-05 00:06:23 +03:00
|
|
|
|
2003-06-23 16:27:59 +03:00
|
|
|
#include <iostream>
|
2003-12-22 18:40:46 +02:00
|
|
|
#include <algorithm>
|
2003-06-23 16:27:59 +03:00
|
|
|
|
2005-01-19 18:39:47 +02:00
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/stat.h>
|
2003-10-15 15:42:39 +03:00
|
|
|
#include <unistd.h>
|
2005-01-19 18:39:47 +02:00
|
|
|
#include <utime.h>
|
2003-06-23 16:27:59 +03:00
|
|
|
|
2006-11-30 20:35:36 +02:00
|
|
|
|
2006-09-05 00:06:23 +03:00
|
|
|
namespace nix {
|
2003-06-23 16:27:59 +03:00
|
|
|
|
2006-09-05 00:06:23 +03:00
|
|
|
|
2003-10-15 15:42:39 +03:00
|
|
|
/* Nix database. */
|
|
|
|
static Database nixDB;
|
|
|
|
|
|
|
|
|
|
|
|
/* Database tables. */
|
|
|
|
|
|
|
|
/* dbValidPaths :: Path -> ()
|
|
|
|
|
|
|
|
The existence of a key $p$ indicates that path $p$ is valid (that
|
|
|
|
is, produced by a succesful build). */
|
2004-10-25 17:38:23 +03:00
|
|
|
static TableId dbValidPaths = 0;
|
2003-10-15 15:42:39 +03:00
|
|
|
|
2005-01-19 13:16:11 +02:00
|
|
|
/* dbReferences :: Path -> [Path]
|
2003-10-15 15:42:39 +03:00
|
|
|
|
2005-01-19 13:16:11 +02:00
|
|
|
This table lists the outgoing file system references for each
|
|
|
|
output path that has been built by a Nix derivation. These are
|
|
|
|
found by scanning the path for the hash components of input
|
|
|
|
paths. */
|
|
|
|
static TableId dbReferences = 0;
|
2003-10-15 15:42:39 +03:00
|
|
|
|
2005-12-12 20:24:42 +02:00
|
|
|
/* dbReferrers :: Path -> Path
|
2003-10-15 15:42:39 +03:00
|
|
|
|
2005-12-12 20:24:42 +02:00
|
|
|
This table is just the reverse mapping of dbReferences. This table
|
|
|
|
can have duplicate keys, each corresponding value denoting a single
|
|
|
|
referrer. */
|
|
|
|
static TableId dbReferrers = 0;
|
2003-10-15 15:42:39 +03:00
|
|
|
|
2005-02-07 15:40:40 +02:00
|
|
|
/* dbDerivers :: Path -> [Path]
|
|
|
|
|
|
|
|
This table lists the derivation used to build a path. There can
|
|
|
|
only be multiple such paths for fixed-output derivations (i.e.,
|
|
|
|
derivations specifying an expected hash). */
|
|
|
|
static TableId dbDerivers = 0;
|
|
|
|
|
2003-10-15 15:42:39 +03:00
|
|
|
|
2005-12-12 20:24:42 +02:00
|
|
|
static void upgradeStore07();
|
|
|
|
static void upgradeStore09();
|
2007-08-13 14:37:39 +03:00
|
|
|
static void upgradeStore11();
|
2005-02-09 11:50:29 +02:00
|
|
|
|
|
|
|
|
2006-03-11 00:27:26 +02:00
|
|
|
void checkStoreNotSymlink()
|
|
|
|
{
|
|
|
|
if (getEnv("NIX_IGNORE_SYMLINK_STORE") == "1") return;
|
|
|
|
Path path = nixStore;
|
|
|
|
struct stat st;
|
|
|
|
while (path != "/") {
|
|
|
|
if (lstat(path.c_str(), &st))
|
|
|
|
throw SysError(format("getting status of `%1%'") % path);
|
|
|
|
if (S_ISLNK(st.st_mode))
|
|
|
|
throw Error(format(
|
|
|
|
"the path `%1%' is a symlink; "
|
|
|
|
"this is not allowed for the Nix store and its parent directories")
|
|
|
|
% path);
|
|
|
|
path = dirOf(path);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-11-30 19:43:04 +02:00
|
|
|
LocalStore::LocalStore(bool reserveSpace)
|
2003-10-15 15:42:39 +03:00
|
|
|
{
|
2007-08-12 03:29:28 +03:00
|
|
|
substitutablePathsLoaded = false;
|
|
|
|
|
2004-10-25 17:38:23 +03:00
|
|
|
if (readOnlyMode) return;
|
2005-02-09 11:50:29 +02:00
|
|
|
|
2006-03-11 00:27:26 +02:00
|
|
|
checkStoreNotSymlink();
|
|
|
|
|
2006-02-16 15:19:15 +02:00
|
|
|
try {
|
|
|
|
Path reservedPath = nixDBPath + "/reserved";
|
2006-02-16 15:58:10 +02:00
|
|
|
string s = querySetting("gc-reserved-space", "");
|
|
|
|
int reservedSize;
|
|
|
|
if (!string2Int(s, reservedSize)) reservedSize = 1024 * 1024;
|
2006-02-16 15:19:15 +02:00
|
|
|
if (reserveSpace) {
|
|
|
|
struct stat st;
|
|
|
|
if (stat(reservedPath.c_str(), &st) == -1 ||
|
|
|
|
st.st_size != reservedSize)
|
2006-02-16 15:58:10 +02:00
|
|
|
writeFile(reservedPath, string(reservedSize, 'X'));
|
2006-02-16 15:19:15 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
deletePath(reservedPath);
|
|
|
|
} catch (SysError & e) { /* don't care about errors */
|
|
|
|
}
|
|
|
|
|
2004-10-25 17:38:23 +03:00
|
|
|
try {
|
|
|
|
nixDB.open(nixDBPath);
|
|
|
|
} catch (DbNoPermission & e) {
|
|
|
|
printMsg(lvlTalkative, "cannot access Nix database; continuing anyway");
|
|
|
|
readOnlyMode = true;
|
|
|
|
return;
|
|
|
|
}
|
2003-10-15 15:42:39 +03:00
|
|
|
dbValidPaths = nixDB.openTable("validpaths");
|
2005-01-19 13:16:11 +02:00
|
|
|
dbReferences = nixDB.openTable("references");
|
2005-12-12 20:24:42 +02:00
|
|
|
dbReferrers = nixDB.openTable("referrers", true); /* must be sorted */
|
2005-02-07 15:40:40 +02:00
|
|
|
dbDerivers = nixDB.openTable("derivers");
|
2005-02-09 11:50:29 +02:00
|
|
|
|
|
|
|
int curSchema = 0;
|
|
|
|
Path schemaFN = nixDBPath + "/schema";
|
|
|
|
if (pathExists(schemaFN)) {
|
|
|
|
string s = readFile(schemaFN);
|
|
|
|
if (!string2Int(s, curSchema))
|
|
|
|
throw Error(format("`%1%' is corrupt") % schemaFN);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (curSchema > nixSchemaVersion)
|
|
|
|
throw Error(format("current Nix store schema is version %1%, but I only support %2%")
|
|
|
|
% curSchema % nixSchemaVersion);
|
|
|
|
|
|
|
|
if (curSchema < nixSchemaVersion) {
|
2005-12-12 20:24:42 +02:00
|
|
|
if (curSchema <= 1)
|
|
|
|
upgradeStore07();
|
|
|
|
if (curSchema == 2)
|
|
|
|
upgradeStore09();
|
2007-08-13 14:37:39 +03:00
|
|
|
if (curSchema == 3)
|
|
|
|
upgradeStore11();
|
2005-02-09 11:50:29 +02:00
|
|
|
writeFile(schemaFN, (format("%1%") % nixSchemaVersion).str());
|
|
|
|
}
|
2003-10-15 15:42:39 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-11-30 19:43:04 +02:00
|
|
|
LocalStore::~LocalStore()
|
2006-03-01 18:36:35 +02:00
|
|
|
{
|
|
|
|
/* If the database isn't open, this is a NOP. */
|
2007-05-01 18:16:17 +03:00
|
|
|
try {
|
|
|
|
nixDB.close();
|
|
|
|
} catch (...) {
|
|
|
|
ignoreException();
|
|
|
|
}
|
2006-03-01 18:36:35 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-10-15 15:42:39 +03:00
|
|
|
void createStoreTransaction(Transaction & txn)
|
|
|
|
{
|
|
|
|
Transaction txn2(nixDB);
|
|
|
|
txn2.moveTo(txn);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-12-13 01:05:01 +02:00
|
|
|
void copyPath(const Path & src, const Path & dst, PathFilter & filter)
|
2003-06-16 16:33:38 +03:00
|
|
|
{
|
2003-07-31 19:05:35 +03:00
|
|
|
debug(format("copying `%1%' to `%2%'") % src % dst);
|
|
|
|
|
2005-03-03 15:58:02 +02:00
|
|
|
/* Dump an archive of the path `src' into a string buffer, then
|
|
|
|
restore the archive to `dst'. This is not a very good method
|
|
|
|
for very large paths, but `copyPath' is mainly used for small
|
|
|
|
files. */
|
2003-06-23 16:27:59 +03:00
|
|
|
|
2006-12-13 01:05:01 +02:00
|
|
|
StringSink sink;
|
|
|
|
dumpPath(src, sink, filter);
|
2003-06-23 16:27:59 +03:00
|
|
|
|
2006-12-13 01:05:01 +02:00
|
|
|
StringSource source(sink.s);
|
2005-03-03 15:58:02 +02:00
|
|
|
restorePath(dst, source);
|
2003-06-16 16:33:38 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-12-09 22:02:27 +02:00
|
|
|
static void _canonicalisePathMetaData(const Path & path)
|
2005-01-19 18:39:47 +02:00
|
|
|
{
|
|
|
|
checkInterrupt();
|
|
|
|
|
|
|
|
struct stat st;
|
|
|
|
if (lstat(path.c_str(), &st))
|
|
|
|
throw SysError(format("getting attributes of path `%1%'") % path);
|
|
|
|
|
2006-12-09 22:02:27 +02:00
|
|
|
/* Change ownership to the current uid. If its a symlink, use
|
|
|
|
lchown if available, otherwise don't bother. Wrong ownership
|
|
|
|
of a symlink doesn't matter, since the owning user can't change
|
|
|
|
the symlink and can't delete it because the directory is not
|
|
|
|
writable. The only exception is top-level paths in the Nix
|
|
|
|
store (since that directory is group-writable for the Nix build
|
|
|
|
users group); we check for this case below. */
|
|
|
|
if (st.st_uid != geteuid()) {
|
|
|
|
#if HAVE_LCHOWN
|
2007-02-06 22:03:53 +02:00
|
|
|
if (lchown(path.c_str(), geteuid(), (gid_t) -1) == -1)
|
2006-12-09 22:02:27 +02:00
|
|
|
#else
|
|
|
|
if (!S_ISLNK(st.st_mode) &&
|
2007-02-06 22:03:53 +02:00
|
|
|
chown(path.c_str(), geteuid(), (gid_t) -1) == -1)
|
2006-12-09 22:02:27 +02:00
|
|
|
#endif
|
|
|
|
throw SysError(format("changing owner of `%1%' to %2%")
|
|
|
|
% path % geteuid());
|
|
|
|
}
|
|
|
|
|
2005-01-19 18:39:47 +02:00
|
|
|
if (!S_ISLNK(st.st_mode)) {
|
|
|
|
|
|
|
|
/* Mask out all type related bits. */
|
|
|
|
mode_t mode = st.st_mode & ~S_IFMT;
|
|
|
|
|
|
|
|
if (mode != 0444 && mode != 0555) {
|
|
|
|
mode = (st.st_mode & S_IFMT)
|
|
|
|
| 0444
|
|
|
|
| (st.st_mode & S_IXUSR ? 0111 : 0);
|
|
|
|
if (chmod(path.c_str(), mode) == -1)
|
|
|
|
throw SysError(format("changing mode of `%1%' to %2$o") % path % mode);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (st.st_mtime != 0) {
|
|
|
|
struct utimbuf utimbuf;
|
|
|
|
utimbuf.actime = st.st_atime;
|
|
|
|
utimbuf.modtime = 0;
|
|
|
|
if (utime(path.c_str(), &utimbuf) == -1)
|
|
|
|
throw SysError(format("changing modification time of `%1%'") % path);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (S_ISDIR(st.st_mode)) {
|
|
|
|
Strings names = readDirectory(path);
|
|
|
|
for (Strings::iterator i = names.begin(); i != names.end(); ++i)
|
2006-12-09 22:02:27 +02:00
|
|
|
_canonicalisePathMetaData(path + "/" + *i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void canonicalisePathMetaData(const Path & path)
|
|
|
|
{
|
|
|
|
_canonicalisePathMetaData(path);
|
|
|
|
|
|
|
|
/* On platforms that don't have lchown(), the top-level path can't
|
|
|
|
be a symlink, since we can't change its ownership. */
|
|
|
|
struct stat st;
|
|
|
|
if (lstat(path.c_str(), &st))
|
|
|
|
throw SysError(format("getting attributes of path `%1%'") % path);
|
|
|
|
|
|
|
|
if (st.st_uid != geteuid()) {
|
|
|
|
assert(S_ISLNK(st.st_mode));
|
|
|
|
throw Error(format("wrong ownership of top-level store path `%1%'") % path);
|
2005-01-19 18:39:47 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-03-23 13:25:20 +02:00
|
|
|
bool isValidPathTxn(const Transaction & txn, const Path & path)
|
2003-12-05 13:05:19 +02:00
|
|
|
{
|
|
|
|
string s;
|
|
|
|
return nixDB.queryString(txn, dbValidPaths, path, s);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-11-30 19:43:04 +02:00
|
|
|
bool LocalStore::isValidPath(const Path & path)
|
2003-12-05 13:05:19 +02:00
|
|
|
{
|
2005-01-25 23:28:25 +02:00
|
|
|
return isValidPathTxn(noTxn, path);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-12-12 20:24:42 +02:00
|
|
|
static string addPrefix(const string & prefix, const string & s)
|
|
|
|
{
|
2005-12-25 04:02:29 +02:00
|
|
|
return prefix + string(1, (char) 0) + s;
|
2005-12-12 20:24:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static string stripPrefix(const string & prefix, const string & s)
|
|
|
|
{
|
|
|
|
if (s.size() <= prefix.size() ||
|
2005-12-25 01:32:59 +02:00
|
|
|
string(s, 0, prefix.size()) != prefix ||
|
2005-12-12 20:24:42 +02:00
|
|
|
s[prefix.size()] != 0)
|
|
|
|
throw Error(format("string `%1%' is missing prefix `%2%'")
|
|
|
|
% s % prefix);
|
|
|
|
return string(s, prefix.size() + 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-12-13 23:04:48 +02:00
|
|
|
static PathSet getReferrers(const Transaction & txn, const Path & storePath)
|
2005-01-27 18:18:39 +02:00
|
|
|
{
|
2005-12-12 20:24:42 +02:00
|
|
|
PathSet referrers;
|
|
|
|
Strings keys;
|
2005-12-25 04:02:29 +02:00
|
|
|
nixDB.enumTable(txn, dbReferrers, keys, storePath + string(1, (char) 0));
|
2005-12-12 20:24:42 +02:00
|
|
|
for (Strings::iterator i = keys.begin(); i != keys.end(); ++i)
|
|
|
|
referrers.insert(stripPrefix(storePath, *i));
|
|
|
|
return referrers;
|
2005-01-27 18:18:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-01-19 13:16:11 +02:00
|
|
|
void setReferences(const Transaction & txn, const Path & storePath,
|
|
|
|
const PathSet & references)
|
2003-10-10 17:46:28 +03:00
|
|
|
{
|
2007-08-12 03:29:28 +03:00
|
|
|
/* For invalid paths, we can only clear the references. */
|
|
|
|
if (references.size() > 0 && !isValidPathTxn(txn, storePath))
|
2005-01-25 23:28:25 +02:00
|
|
|
throw Error(
|
2007-08-12 03:29:28 +03:00
|
|
|
format("cannot set references for invalid path `%1%'") % storePath);
|
2005-01-27 18:18:39 +02:00
|
|
|
|
|
|
|
Paths oldReferences;
|
2005-01-27 19:48:14 +02:00
|
|
|
nixDB.queryStrings(txn, dbReferences, storePath, oldReferences);
|
2005-03-03 15:10:44 +02:00
|
|
|
|
|
|
|
PathSet oldReferences2(oldReferences.begin(), oldReferences.end());
|
|
|
|
if (oldReferences2 == references) return;
|
2005-01-25 23:28:25 +02:00
|
|
|
|
2005-01-19 13:16:11 +02:00
|
|
|
nixDB.setStrings(txn, dbReferences, storePath,
|
|
|
|
Paths(references.begin(), references.end()));
|
2003-12-05 13:05:19 +02:00
|
|
|
|
2005-12-13 23:04:48 +02:00
|
|
|
/* Update the referrers mappings of all new referenced paths. */
|
2005-01-19 13:16:11 +02:00
|
|
|
for (PathSet::const_iterator i = references.begin();
|
|
|
|
i != references.end(); ++i)
|
2005-12-12 20:24:42 +02:00
|
|
|
if (oldReferences2.find(*i) == oldReferences2.end())
|
|
|
|
nixDB.setString(txn, dbReferrers, addPrefix(*i, storePath), "");
|
2005-01-27 18:18:39 +02:00
|
|
|
|
2005-12-13 23:04:48 +02:00
|
|
|
/* Remove referrer mappings from paths that are no longer
|
2005-01-27 18:18:39 +02:00
|
|
|
references. */
|
|
|
|
for (Paths::iterator i = oldReferences.begin();
|
|
|
|
i != oldReferences.end(); ++i)
|
2005-12-12 20:24:42 +02:00
|
|
|
if (references.find(*i) == references.end())
|
|
|
|
nixDB.delPair(txn, dbReferrers, addPrefix(*i, storePath));
|
2003-10-15 15:42:39 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-02-08 15:23:55 +02:00
|
|
|
void queryReferences(const Transaction & txn,
|
|
|
|
const Path & storePath, PathSet & references)
|
2003-10-10 18:25:21 +03:00
|
|
|
{
|
2005-01-19 13:16:11 +02:00
|
|
|
Paths references2;
|
2007-08-12 03:29:28 +03:00
|
|
|
if (!isValidPathTxn(txn, storePath))
|
2005-01-25 23:28:25 +02:00
|
|
|
throw Error(format("path `%1%' is not valid") % storePath);
|
2005-02-08 15:23:55 +02:00
|
|
|
nixDB.queryStrings(txn, dbReferences, storePath, references2);
|
2005-01-19 13:16:11 +02:00
|
|
|
references.insert(references2.begin(), references2.end());
|
2003-10-10 18:25:21 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-11-30 19:43:04 +02:00
|
|
|
void LocalStore::queryReferences(const Path & storePath,
|
|
|
|
PathSet & references)
|
|
|
|
{
|
|
|
|
nix::queryReferences(noTxn, storePath, references);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-12-13 23:04:48 +02:00
|
|
|
void queryReferrers(const Transaction & txn,
|
|
|
|
const Path & storePath, PathSet & referrers)
|
2005-01-19 18:59:56 +02:00
|
|
|
{
|
2007-08-12 03:29:28 +03:00
|
|
|
if (!isValidPathTxn(txn, storePath))
|
2005-01-25 23:28:25 +02:00
|
|
|
throw Error(format("path `%1%' is not valid") % storePath);
|
2005-12-13 23:04:48 +02:00
|
|
|
PathSet referrers2 = getReferrers(txn, storePath);
|
|
|
|
referrers.insert(referrers2.begin(), referrers2.end());
|
2005-01-19 18:59:56 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-11-30 19:43:04 +02:00
|
|
|
void LocalStore::queryReferrers(const Path & storePath,
|
|
|
|
PathSet & referrers)
|
|
|
|
{
|
|
|
|
nix::queryReferrers(noTxn, storePath, referrers);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-02-07 15:40:40 +02:00
|
|
|
void setDeriver(const Transaction & txn, const Path & storePath,
|
|
|
|
const Path & deriver)
|
|
|
|
{
|
|
|
|
assertStorePath(storePath);
|
|
|
|
if (deriver == "") return;
|
|
|
|
assertStorePath(deriver);
|
2007-08-12 03:29:28 +03:00
|
|
|
if (!isValidPathTxn(txn, storePath))
|
2005-02-07 15:40:40 +02:00
|
|
|
throw Error(format("path `%1%' is not valid") % storePath);
|
|
|
|
nixDB.setString(txn, dbDerivers, storePath, deriver);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-06-12 19:53:44 +03:00
|
|
|
static Path queryDeriver(const Transaction & txn, const Path & storePath)
|
2005-02-07 15:40:40 +02:00
|
|
|
{
|
2007-08-12 03:29:28 +03:00
|
|
|
if (!isValidPathTxn(txn, storePath))
|
2005-02-07 15:40:40 +02:00
|
|
|
throw Error(format("path `%1%' is not valid") % storePath);
|
|
|
|
Path deriver;
|
|
|
|
if (nixDB.queryString(txn, dbDerivers, storePath, deriver))
|
|
|
|
return deriver;
|
|
|
|
else
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-06-12 19:53:44 +03:00
|
|
|
Path LocalStore::queryDeriver(const Path & path)
|
|
|
|
{
|
|
|
|
return nix::queryDeriver(noTxn, path);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-08-12 03:29:28 +03:00
|
|
|
PathSet LocalStore::querySubstitutablePaths()
|
2003-07-10 18:11:48 +03:00
|
|
|
{
|
2007-08-12 03:29:28 +03:00
|
|
|
if (!substitutablePathsLoaded) {
|
|
|
|
for (Paths::iterator i = substituters.begin(); i != substituters.end(); ++i) {
|
|
|
|
debug(format("running `%1%' to find out substitutable paths") % *i);
|
|
|
|
Strings args;
|
|
|
|
args.push_back("--query-paths");
|
|
|
|
Strings ss = tokenizeString(runProgram(*i, false, args), "\n");
|
|
|
|
for (Strings::iterator j = ss.begin(); j != ss.end(); ++j) {
|
|
|
|
if (!isStorePath(*j))
|
|
|
|
throw Error(format("`%1%' returned a bad substitutable path `%2%'")
|
|
|
|
% *i % *j);
|
|
|
|
substitutablePaths.insert(*j);
|
|
|
|
}
|
2004-06-20 22:17:54 +03:00
|
|
|
}
|
2007-08-12 03:29:28 +03:00
|
|
|
substitutablePathsLoaded = true;
|
2004-06-20 22:17:54 +03:00
|
|
|
}
|
2003-12-05 13:05:19 +02:00
|
|
|
|
2007-08-12 03:29:28 +03:00
|
|
|
return substitutablePaths;
|
2004-06-20 22:17:54 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-08-12 03:29:28 +03:00
|
|
|
bool LocalStore::hasSubstitutes(const Path & path)
|
2004-06-20 22:17:54 +03:00
|
|
|
{
|
2007-08-12 03:29:28 +03:00
|
|
|
if (!substitutablePathsLoaded)
|
|
|
|
querySubstitutablePaths();
|
|
|
|
return substitutablePaths.find(path) != substitutablePaths.end();
|
2004-12-20 15:43:32 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-02-09 11:50:29 +02:00
|
|
|
static void setHash(const Transaction & txn, const Path & storePath,
|
|
|
|
const Hash & hash)
|
|
|
|
{
|
|
|
|
assert(hash.type == htSHA256);
|
|
|
|
nixDB.setString(txn, dbValidPaths, storePath, "sha256:" + printHash(hash));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static Hash queryHash(const Transaction & txn, const Path & storePath)
|
|
|
|
{
|
|
|
|
string s;
|
|
|
|
nixDB.queryString(txn, dbValidPaths, storePath, s);
|
2006-05-11 05:19:43 +03:00
|
|
|
string::size_type colon = s.find(':');
|
2005-02-09 11:50:29 +02:00
|
|
|
if (colon == string::npos)
|
|
|
|
throw Error(format("corrupt hash `%1%' in valid-path entry for `%2%'")
|
|
|
|
% s % storePath);
|
|
|
|
HashType ht = parseHashType(string(s, 0, colon));
|
|
|
|
if (ht == htUnknown)
|
|
|
|
throw Error(format("unknown hash type `%1%' in valid-path entry for `%2%'")
|
2005-02-14 11:53:11 +02:00
|
|
|
% string(s, 0, colon) % storePath);
|
2005-02-09 11:50:29 +02:00
|
|
|
return parseHash(ht, string(s, colon + 1));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-11-30 19:43:04 +02:00
|
|
|
Hash LocalStore::queryPathHash(const Path & path)
|
2005-03-02 17:57:06 +02:00
|
|
|
{
|
|
|
|
if (!isValidPath(path))
|
|
|
|
throw Error(format("path `%1%' is not valid") % path);
|
|
|
|
return queryHash(noTxn, path);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-01-19 18:39:47 +02:00
|
|
|
void registerValidPath(const Transaction & txn,
|
2005-03-23 15:07:28 +02:00
|
|
|
const Path & path, const Hash & hash, const PathSet & references,
|
2005-02-07 15:40:40 +02:00
|
|
|
const Path & deriver)
|
2003-07-07 12:25:26 +03:00
|
|
|
{
|
2005-03-23 15:07:28 +02:00
|
|
|
ValidPathInfo info;
|
|
|
|
info.path = path;
|
|
|
|
info.hash = hash;
|
|
|
|
info.references = references;
|
|
|
|
info.deriver = deriver;
|
|
|
|
ValidPathInfos infos;
|
|
|
|
infos.push_back(info);
|
|
|
|
registerValidPaths(txn, infos);
|
|
|
|
}
|
2005-01-19 18:39:47 +02:00
|
|
|
|
|
|
|
|
2005-03-23 15:07:28 +02:00
|
|
|
void registerValidPaths(const Transaction & txn,
|
|
|
|
const ValidPathInfos & infos)
|
|
|
|
{
|
|
|
|
PathSet newPaths;
|
|
|
|
for (ValidPathInfos::const_iterator i = infos.begin();
|
|
|
|
i != infos.end(); ++i)
|
|
|
|
newPaths.insert(i->path);
|
|
|
|
|
|
|
|
for (ValidPathInfos::const_iterator i = infos.begin();
|
|
|
|
i != infos.end(); ++i)
|
|
|
|
{
|
|
|
|
assertStorePath(i->path);
|
|
|
|
|
|
|
|
debug(format("registering path `%1%'") % i->path);
|
|
|
|
setHash(txn, i->path, i->hash);
|
2005-02-07 15:40:40 +02:00
|
|
|
|
2005-03-23 15:07:28 +02:00
|
|
|
setReferences(txn, i->path, i->references);
|
|
|
|
|
|
|
|
/* Check that all referenced paths are also valid (or about to
|
|
|
|
become valid). */
|
|
|
|
for (PathSet::iterator j = i->references.begin();
|
|
|
|
j != i->references.end(); ++j)
|
|
|
|
if (!isValidPathTxn(txn, *j) && newPaths.find(*j) == newPaths.end())
|
|
|
|
throw Error(format("cannot register path `%1%' as valid, since its reference `%2%' is invalid")
|
|
|
|
% i->path % *j);
|
|
|
|
|
|
|
|
setDeriver(txn, i->path, i->deriver);
|
|
|
|
}
|
2003-10-08 18:06:59 +03:00
|
|
|
}
|
2003-07-07 12:25:26 +03:00
|
|
|
|
2003-07-31 19:05:35 +03:00
|
|
|
|
2005-01-31 16:00:43 +02:00
|
|
|
/* Invalidate a path. The caller is responsible for checking that
|
2005-12-13 23:04:48 +02:00
|
|
|
there are no referrers. */
|
2005-03-03 15:10:44 +02:00
|
|
|
static void invalidatePath(Transaction & txn, const Path & path)
|
2003-07-08 12:54:47 +03:00
|
|
|
{
|
2007-08-12 03:29:28 +03:00
|
|
|
debug(format("invalidating path `%1%'") % path);
|
2003-07-08 12:54:47 +03:00
|
|
|
|
2005-01-27 17:21:29 +02:00
|
|
|
/* Clear the `references' entry for this path, as well as the
|
2007-08-12 03:29:28 +03:00
|
|
|
inverse `referrers' entries, and the `derivers' entry. */
|
|
|
|
setReferences(txn, path, PathSet());
|
|
|
|
nixDB.delPair(txn, dbDerivers, path);
|
2005-01-27 18:18:39 +02:00
|
|
|
nixDB.delPair(txn, dbValidPaths, path);
|
2003-07-08 12:54:47 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-12-01 22:51:18 +02:00
|
|
|
Path LocalStore::addToStore(const Path & _srcPath, bool fixed,
|
2006-12-13 01:05:01 +02:00
|
|
|
bool recursive, string hashAlgo, PathFilter & filter)
|
2003-07-07 12:25:26 +03:00
|
|
|
{
|
2003-10-08 18:06:59 +03:00
|
|
|
Path srcPath(absPath(_srcPath));
|
|
|
|
debug(format("adding `%1%' to the store") % srcPath);
|
2003-07-07 12:25:26 +03:00
|
|
|
|
2006-12-01 20:00:01 +02:00
|
|
|
std::pair<Path, Hash> pr =
|
2006-12-13 01:05:01 +02:00
|
|
|
computeStorePathForPath(srcPath, fixed, recursive, hashAlgo, filter);
|
2006-12-01 20:00:01 +02:00
|
|
|
Path & dstPath(pr.first);
|
|
|
|
Hash & h(pr.second);
|
2003-07-10 18:11:48 +03:00
|
|
|
|
2006-12-01 22:51:18 +02:00
|
|
|
addTempRoot(dstPath);
|
2005-01-31 12:27:25 +02:00
|
|
|
|
2006-12-01 22:51:18 +02:00
|
|
|
if (!isValidPath(dstPath)) {
|
2003-07-10 18:11:48 +03:00
|
|
|
|
2003-10-08 18:06:59 +03:00
|
|
|
/* The first check above is an optimisation to prevent
|
|
|
|
unnecessary lock acquisition. */
|
2003-07-22 18:15:15 +03:00
|
|
|
|
2006-06-01 21:13:33 +03:00
|
|
|
PathLocks outputLock(singleton<PathSet, Path>(dstPath));
|
2003-07-22 18:15:15 +03:00
|
|
|
|
2003-10-08 18:06:59 +03:00
|
|
|
if (!isValidPath(dstPath)) {
|
2004-06-21 10:46:02 +03:00
|
|
|
|
2006-12-09 02:26:24 +02:00
|
|
|
if (pathExists(dstPath)) deletePathWrapped(dstPath);
|
2004-10-25 17:38:23 +03:00
|
|
|
|
2006-12-13 01:05:01 +02:00
|
|
|
copyPath(srcPath, dstPath, filter);
|
2003-08-01 12:01:51 +03:00
|
|
|
|
2006-12-13 01:05:01 +02:00
|
|
|
Hash h2 = hashPath(htSHA256, dstPath, filter);
|
2005-01-14 15:51:38 +02:00
|
|
|
if (h != h2)
|
|
|
|
throw Error(format("contents of `%1%' changed while copying it to `%2%' (%3% -> %4%)")
|
2005-01-14 18:04:03 +02:00
|
|
|
% srcPath % dstPath % printHash(h) % printHash(h2));
|
2005-01-14 15:51:38 +02:00
|
|
|
|
2005-01-19 18:39:47 +02:00
|
|
|
canonicalisePathMetaData(dstPath);
|
2004-09-10 00:19:20 +03:00
|
|
|
|
2003-10-08 18:06:59 +03:00
|
|
|
Transaction txn(nixDB);
|
2005-02-07 15:40:40 +02:00
|
|
|
registerValidPath(txn, dstPath, h, PathSet(), "");
|
2003-10-08 18:06:59 +03:00
|
|
|
txn.commit();
|
2003-08-01 12:01:51 +03:00
|
|
|
}
|
2003-11-22 20:45:56 +02:00
|
|
|
|
|
|
|
outputLock.setDeletion(true);
|
2003-06-16 16:33:38 +03:00
|
|
|
}
|
2003-08-04 10:09:36 +03:00
|
|
|
|
2003-10-08 18:06:59 +03:00
|
|
|
return dstPath;
|
2003-06-16 16:33:38 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-11-30 19:43:04 +02:00
|
|
|
Path LocalStore::addTextToStore(const string & suffix, const string & s,
|
2005-01-25 23:28:25 +02:00
|
|
|
const PathSet & references)
|
2003-10-15 15:42:39 +03:00
|
|
|
{
|
2007-01-29 17:51:37 +02:00
|
|
|
Path dstPath = computeStorePathForText(suffix, s, references);
|
2004-02-14 23:44:18 +02:00
|
|
|
|
2006-12-01 22:51:18 +02:00
|
|
|
addTempRoot(dstPath);
|
2005-01-31 12:27:25 +02:00
|
|
|
|
2006-12-01 22:51:18 +02:00
|
|
|
if (!isValidPath(dstPath)) {
|
2003-10-15 15:42:39 +03:00
|
|
|
|
2006-06-01 21:13:33 +03:00
|
|
|
PathLocks outputLock(singleton<PathSet, Path>(dstPath));
|
2003-10-23 13:51:55 +03:00
|
|
|
|
|
|
|
if (!isValidPath(dstPath)) {
|
2004-06-21 10:46:02 +03:00
|
|
|
|
2006-12-09 02:26:24 +02:00
|
|
|
if (pathExists(dstPath)) deletePathWrapped(dstPath);
|
2004-06-21 10:46:02 +03:00
|
|
|
|
2003-11-22 17:58:34 +02:00
|
|
|
writeStringToFile(dstPath, s);
|
2003-10-15 15:42:39 +03:00
|
|
|
|
2005-01-19 18:39:47 +02:00
|
|
|
canonicalisePathMetaData(dstPath);
|
2004-09-10 00:19:20 +03:00
|
|
|
|
2003-10-23 13:51:55 +03:00
|
|
|
Transaction txn(nixDB);
|
2005-01-25 23:28:25 +02:00
|
|
|
registerValidPath(txn, dstPath,
|
2005-02-07 15:40:40 +02:00
|
|
|
hashPath(htSHA256, dstPath), references, "");
|
2003-10-23 13:51:55 +03:00
|
|
|
txn.commit();
|
|
|
|
}
|
2003-11-22 20:45:56 +02:00
|
|
|
|
|
|
|
outputLock.setDeletion(true);
|
2003-10-15 15:42:39 +03:00
|
|
|
}
|
2005-01-14 15:51:38 +02:00
|
|
|
|
|
|
|
return dstPath;
|
2003-10-15 15:42:39 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-02-21 16:31:42 +02:00
|
|
|
struct HashAndWriteSink : Sink
|
|
|
|
{
|
|
|
|
Sink & writeSink;
|
|
|
|
HashSink hashSink;
|
|
|
|
bool hashing;
|
|
|
|
HashAndWriteSink(Sink & writeSink) : writeSink(writeSink), hashSink(htSHA256)
|
|
|
|
{
|
|
|
|
hashing = true;
|
|
|
|
}
|
|
|
|
virtual void operator ()
|
|
|
|
(const unsigned char * data, unsigned int len)
|
|
|
|
{
|
|
|
|
writeSink(data, len);
|
|
|
|
if (hashing) hashSink(data, len);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
#define EXPORT_MAGIC 0x4558494e
|
|
|
|
|
|
|
|
|
2007-02-21 19:51:10 +02:00
|
|
|
static void checkSecrecy(const Path & path)
|
|
|
|
{
|
|
|
|
struct stat st;
|
|
|
|
if (stat(path.c_str(), &st))
|
|
|
|
throw SysError(format("getting status of `%1%'") % path);
|
|
|
|
if ((st.st_mode & (S_IRWXG | S_IRWXO)) != 0)
|
|
|
|
throw Error(format("file `%1%' should be secret (inaccessible to everybody else)!") % path);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-02-21 01:17:20 +02:00
|
|
|
void LocalStore::exportPath(const Path & path, bool sign,
|
|
|
|
Sink & sink)
|
|
|
|
{
|
|
|
|
assertStorePath(path);
|
2007-02-21 16:31:42 +02:00
|
|
|
|
2007-02-21 18:23:25 +02:00
|
|
|
/* Wrap all of this in a transaction to make sure that we export
|
|
|
|
consistent metadata. */
|
|
|
|
Transaction txn(nixDB);
|
|
|
|
addTempRoot(path);
|
|
|
|
if (!isValidPath(path))
|
|
|
|
throw Error(format("path `%1%' is not valid") % path);
|
|
|
|
|
2007-02-21 16:31:42 +02:00
|
|
|
HashAndWriteSink hashAndWriteSink(sink);
|
2007-02-21 01:17:20 +02:00
|
|
|
|
2007-02-21 16:31:42 +02:00
|
|
|
dumpPath(path, hashAndWriteSink);
|
2007-02-21 01:17:20 +02:00
|
|
|
|
2007-02-21 16:31:42 +02:00
|
|
|
writeInt(EXPORT_MAGIC, hashAndWriteSink);
|
|
|
|
|
|
|
|
writeString(path, hashAndWriteSink);
|
2007-02-21 01:17:20 +02:00
|
|
|
|
|
|
|
PathSet references;
|
2007-02-21 18:23:25 +02:00
|
|
|
nix::queryReferences(txn, path, references);
|
2007-02-21 16:31:42 +02:00
|
|
|
writeStringSet(references, hashAndWriteSink);
|
2007-02-21 01:17:20 +02:00
|
|
|
|
2007-06-12 19:53:44 +03:00
|
|
|
Path deriver = nix::queryDeriver(txn, path);
|
2007-02-21 16:31:42 +02:00
|
|
|
writeString(deriver, hashAndWriteSink);
|
|
|
|
|
|
|
|
if (sign) {
|
|
|
|
Hash hash = hashAndWriteSink.hashSink.finish();
|
|
|
|
hashAndWriteSink.hashing = false;
|
|
|
|
|
|
|
|
writeInt(1, hashAndWriteSink);
|
|
|
|
|
|
|
|
Path tmpDir = createTempDir();
|
|
|
|
AutoDelete delTmp(tmpDir);
|
|
|
|
Path hashFile = tmpDir + "/hash";
|
|
|
|
writeStringToFile(hashFile, printHash(hash));
|
|
|
|
|
2007-02-21 19:51:10 +02:00
|
|
|
Path secretKey = nixConfDir + "/signing-key.sec";
|
|
|
|
checkSecrecy(secretKey);
|
|
|
|
|
2007-02-21 16:31:42 +02:00
|
|
|
Strings args;
|
|
|
|
args.push_back("rsautl");
|
|
|
|
args.push_back("-sign");
|
|
|
|
args.push_back("-inkey");
|
2007-02-21 19:51:10 +02:00
|
|
|
args.push_back(secretKey);
|
2007-02-21 16:31:42 +02:00
|
|
|
args.push_back("-in");
|
|
|
|
args.push_back(hashFile);
|
2007-03-01 15:30:46 +02:00
|
|
|
string signature = runProgram(OPENSSL_PATH, true, args);
|
2007-02-21 16:31:42 +02:00
|
|
|
|
|
|
|
writeString(signature, hashAndWriteSink);
|
|
|
|
|
|
|
|
} else
|
|
|
|
writeInt(0, hashAndWriteSink);
|
2007-02-21 18:23:25 +02:00
|
|
|
|
|
|
|
txn.commit();
|
2007-02-21 01:17:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-02-21 17:45:32 +02:00
|
|
|
struct HashAndReadSource : Source
|
|
|
|
{
|
|
|
|
Source & readSource;
|
|
|
|
HashSink hashSink;
|
|
|
|
bool hashing;
|
|
|
|
HashAndReadSource(Source & readSource) : readSource(readSource), hashSink(htSHA256)
|
|
|
|
{
|
|
|
|
hashing = true;
|
|
|
|
}
|
|
|
|
virtual void operator ()
|
|
|
|
(unsigned char * data, unsigned int len)
|
|
|
|
{
|
|
|
|
readSource(data, len);
|
|
|
|
if (hashing) hashSink(data, len);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
Path LocalStore::importPath(bool requireSignature, Source & source)
|
|
|
|
{
|
|
|
|
HashAndReadSource hashAndReadSource(source);
|
|
|
|
|
|
|
|
/* We don't yet know what store path this archive contains (the
|
|
|
|
store path follows the archive data proper), and besides, we
|
|
|
|
don't know yet whether the signature is valid. */
|
|
|
|
Path tmpDir = createTempDir(nixStore);
|
|
|
|
AutoDelete delTmp(tmpDir);
|
|
|
|
Path unpacked = tmpDir + "/unpacked";
|
|
|
|
|
|
|
|
restorePath(unpacked, hashAndReadSource);
|
|
|
|
|
|
|
|
unsigned int magic = readInt(hashAndReadSource);
|
|
|
|
if (magic != EXPORT_MAGIC)
|
|
|
|
throw Error("Nix archive cannot be imported; wrong format");
|
|
|
|
|
|
|
|
Path dstPath = readStorePath(hashAndReadSource);
|
|
|
|
|
|
|
|
PathSet references = readStorePaths(hashAndReadSource);
|
|
|
|
|
2007-02-28 01:18:57 +02:00
|
|
|
Path deriver = readString(hashAndReadSource);
|
|
|
|
if (deriver != "") assertStorePath(deriver);
|
2007-02-21 17:45:32 +02:00
|
|
|
|
|
|
|
Hash hash = hashAndReadSource.hashSink.finish();
|
|
|
|
hashAndReadSource.hashing = false;
|
|
|
|
|
|
|
|
bool haveSignature = readInt(hashAndReadSource) == 1;
|
|
|
|
|
|
|
|
if (requireSignature && !haveSignature)
|
|
|
|
throw Error("imported archive lacks a signature");
|
|
|
|
|
|
|
|
if (haveSignature) {
|
|
|
|
string signature = readString(hashAndReadSource);
|
|
|
|
|
2007-03-01 14:30:24 +02:00
|
|
|
if (requireSignature) {
|
|
|
|
Path sigFile = tmpDir + "/sig";
|
|
|
|
writeStringToFile(sigFile, signature);
|
|
|
|
|
|
|
|
Strings args;
|
|
|
|
args.push_back("rsautl");
|
|
|
|
args.push_back("-verify");
|
|
|
|
args.push_back("-inkey");
|
|
|
|
args.push_back(nixConfDir + "/signing-key.pub");
|
|
|
|
args.push_back("-pubin");
|
|
|
|
args.push_back("-in");
|
|
|
|
args.push_back(sigFile);
|
2007-03-01 15:30:46 +02:00
|
|
|
string hash2 = runProgram(OPENSSL_PATH, true, args);
|
2007-03-01 14:30:24 +02:00
|
|
|
|
|
|
|
/* Note: runProgram() throws an exception if the signature
|
|
|
|
is invalid. */
|
|
|
|
|
|
|
|
if (printHash(hash) != hash2)
|
|
|
|
throw Error(
|
|
|
|
"signed hash doesn't match actual contents of imported "
|
|
|
|
"archive; archive could be corrupt, or someone is trying "
|
|
|
|
"to import a Trojan horse");
|
|
|
|
}
|
2007-02-21 17:45:32 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Do the actual import. */
|
|
|
|
|
|
|
|
/* !!! way too much code duplication with addTextToStore() etc. */
|
|
|
|
addTempRoot(dstPath);
|
|
|
|
|
|
|
|
if (!isValidPath(dstPath)) {
|
|
|
|
|
|
|
|
PathLocks outputLock(singleton<PathSet, Path>(dstPath));
|
|
|
|
|
|
|
|
if (!isValidPath(dstPath)) {
|
|
|
|
|
|
|
|
if (pathExists(dstPath)) deletePathWrapped(dstPath);
|
|
|
|
|
|
|
|
if (rename(unpacked.c_str(), dstPath.c_str()) == -1)
|
|
|
|
throw SysError(format("cannot move `%1%' to `%2%'")
|
|
|
|
% unpacked % dstPath);
|
|
|
|
|
|
|
|
canonicalisePathMetaData(dstPath);
|
|
|
|
|
|
|
|
Transaction txn(nixDB);
|
|
|
|
/* !!! if we were clever, we could prevent the hashPath()
|
|
|
|
here. */
|
2007-02-21 18:23:25 +02:00
|
|
|
if (!isValidPath(deriver)) deriver = "";
|
2007-02-21 17:45:32 +02:00
|
|
|
registerValidPath(txn, dstPath,
|
2007-02-21 18:23:25 +02:00
|
|
|
hashPath(htSHA256, dstPath), references, deriver);
|
2007-02-21 17:45:32 +02:00
|
|
|
txn.commit();
|
|
|
|
}
|
|
|
|
|
|
|
|
outputLock.setDeletion(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
return dstPath;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-12-15 23:11:39 +02:00
|
|
|
void deleteFromStore(const Path & _path, unsigned long long & bytesFreed)
|
2003-06-23 17:40:49 +03:00
|
|
|
{
|
2005-12-15 23:11:39 +02:00
|
|
|
bytesFreed = 0;
|
2003-10-08 18:06:59 +03:00
|
|
|
Path path(canonPath(_path));
|
|
|
|
|
2004-02-14 23:44:18 +02:00
|
|
|
assertStorePath(path);
|
2003-07-08 12:54:47 +03:00
|
|
|
|
2003-11-22 20:45:56 +02:00
|
|
|
Transaction txn(nixDB);
|
2005-01-31 16:00:43 +02:00
|
|
|
if (isValidPathTxn(txn, path)) {
|
2005-12-13 23:04:48 +02:00
|
|
|
PathSet referrers = getReferrers(txn, path);
|
|
|
|
for (PathSet::iterator i = referrers.begin();
|
|
|
|
i != referrers.end(); ++i)
|
2005-04-12 13:51:00 +03:00
|
|
|
if (*i != path && isValidPathTxn(txn, *i))
|
2007-01-14 18:24:49 +02:00
|
|
|
throw PathInUse(format("cannot delete path `%1%' because it is in use by path `%2%'") % path % *i);
|
2005-03-03 15:10:44 +02:00
|
|
|
invalidatePath(txn, path);
|
2005-01-31 16:00:43 +02:00
|
|
|
}
|
2003-11-22 20:45:56 +02:00
|
|
|
txn.commit();
|
2003-07-08 12:54:47 +03:00
|
|
|
|
2006-12-09 02:26:24 +02:00
|
|
|
deletePathWrapped(path, bytesFreed);
|
2003-06-23 17:40:49 +03:00
|
|
|
}
|
2003-07-17 15:27:55 +03:00
|
|
|
|
|
|
|
|
2005-02-08 15:48:53 +02:00
|
|
|
void verifyStore(bool checkContents)
|
2003-07-17 15:27:55 +03:00
|
|
|
{
|
2003-07-31 22:49:11 +03:00
|
|
|
Transaction txn(nixDB);
|
|
|
|
|
2007-01-14 19:28:30 +02:00
|
|
|
|
|
|
|
printMsg(lvlInfo, "checking path existence");
|
|
|
|
|
2003-10-10 18:14:29 +03:00
|
|
|
Paths paths;
|
2003-11-22 20:45:56 +02:00
|
|
|
PathSet validPaths;
|
2003-10-10 18:14:29 +03:00
|
|
|
nixDB.enumTable(txn, dbValidPaths, paths);
|
2003-07-17 15:27:55 +03:00
|
|
|
|
2003-12-05 13:05:19 +02:00
|
|
|
for (Paths::iterator i = paths.begin(); i != paths.end(); ++i) {
|
2007-05-01 16:21:05 +03:00
|
|
|
checkInterrupt();
|
2005-02-08 15:48:53 +02:00
|
|
|
if (!pathExists(*i)) {
|
|
|
|
printMsg(lvlError, format("path `%1%' disappeared") % *i);
|
2005-03-03 15:10:44 +02:00
|
|
|
invalidatePath(txn, *i);
|
2005-02-08 15:48:53 +02:00
|
|
|
} else if (!isStorePath(*i)) {
|
|
|
|
printMsg(lvlError, format("path `%1%' is not in the Nix store") % *i);
|
2005-03-03 15:10:44 +02:00
|
|
|
invalidatePath(txn, *i);
|
2005-02-08 15:48:53 +02:00
|
|
|
} else {
|
|
|
|
if (checkContents) {
|
2006-08-01 12:43:41 +03:00
|
|
|
debug(format("checking contents of `%1%'") % *i);
|
2005-02-08 15:48:53 +02:00
|
|
|
Hash expected = queryHash(txn, *i);
|
|
|
|
Hash current = hashPath(expected.type, *i);
|
|
|
|
if (current != expected) {
|
|
|
|
printMsg(lvlError, format("path `%1%' was modified! "
|
|
|
|
"expected hash `%2%', got `%3%'")
|
|
|
|
% *i % printHash(expected) % printHash(current));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
validPaths.insert(*i);
|
|
|
|
}
|
2003-07-17 15:27:55 +03:00
|
|
|
}
|
|
|
|
|
2007-01-14 19:28:30 +02:00
|
|
|
|
2007-08-12 03:29:28 +03:00
|
|
|
/* Check the cleanup invariant: only valid paths can have
|
2005-12-13 23:04:48 +02:00
|
|
|
`references', `referrers', or `derivers' entries. */
|
2005-02-08 15:23:55 +02:00
|
|
|
|
2007-01-14 19:28:30 +02:00
|
|
|
|
2005-02-08 15:23:55 +02:00
|
|
|
/* Check the `derivers' table. */
|
2007-01-14 19:28:30 +02:00
|
|
|
printMsg(lvlInfo, "checking the derivers table");
|
2005-02-08 15:23:55 +02:00
|
|
|
Paths deriversKeys;
|
|
|
|
nixDB.enumTable(txn, dbDerivers, deriversKeys);
|
|
|
|
for (Paths::iterator i = deriversKeys.begin();
|
|
|
|
i != deriversKeys.end(); ++i)
|
|
|
|
{
|
2007-08-12 03:29:28 +03:00
|
|
|
if (validPaths.find(*i) == validPaths.end()) {
|
|
|
|
printMsg(lvlError, format("removing deriver entry for invalid path `%1%'")
|
2005-02-08 15:23:55 +02:00
|
|
|
% *i);
|
|
|
|
nixDB.delPair(txn, dbDerivers, *i);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
Path deriver = queryDeriver(txn, *i);
|
|
|
|
if (!isStorePath(deriver)) {
|
2007-01-14 19:28:30 +02:00
|
|
|
printMsg(lvlError, format("removing corrupt deriver `%1%' for `%2%'")
|
2005-02-08 15:23:55 +02:00
|
|
|
% deriver % *i);
|
|
|
|
nixDB.delPair(txn, dbDerivers, *i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-01-14 19:28:30 +02:00
|
|
|
|
2005-02-08 15:23:55 +02:00
|
|
|
/* Check the `references' table. */
|
2007-01-14 19:28:30 +02:00
|
|
|
printMsg(lvlInfo, "checking the references table");
|
2005-02-08 15:23:55 +02:00
|
|
|
Paths referencesKeys;
|
|
|
|
nixDB.enumTable(txn, dbReferences, referencesKeys);
|
|
|
|
for (Paths::iterator i = referencesKeys.begin();
|
|
|
|
i != referencesKeys.end(); ++i)
|
|
|
|
{
|
2007-08-12 03:29:28 +03:00
|
|
|
if (validPaths.find(*i) == validPaths.end()) {
|
|
|
|
printMsg(lvlError, format("removing references entry for invalid path `%1%'")
|
2005-02-08 15:23:55 +02:00
|
|
|
% *i);
|
2005-03-25 16:21:49 +02:00
|
|
|
setReferences(txn, *i, PathSet());
|
2005-02-08 15:23:55 +02:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
PathSet references;
|
|
|
|
queryReferences(txn, *i, references);
|
|
|
|
for (PathSet::iterator j = references.begin();
|
|
|
|
j != references.end(); ++j)
|
|
|
|
{
|
2005-12-12 20:24:42 +02:00
|
|
|
string dummy;
|
|
|
|
if (!nixDB.queryString(txn, dbReferrers, addPrefix(*j, *i), dummy)) {
|
2007-01-14 19:28:30 +02:00
|
|
|
printMsg(lvlError, format("adding missing referrer mapping from `%1%' to `%2%'")
|
2005-02-08 15:23:55 +02:00
|
|
|
% *j % *i);
|
2005-12-12 20:24:42 +02:00
|
|
|
nixDB.setString(txn, dbReferrers, addPrefix(*j, *i), "");
|
2005-02-08 15:23:55 +02:00
|
|
|
}
|
2007-08-12 03:29:28 +03:00
|
|
|
if (validPaths.find(*j) == validPaths.end()) {
|
2005-02-09 11:50:29 +02:00
|
|
|
printMsg(lvlError, format("incomplete closure: `%1%' needs missing `%2%'")
|
|
|
|
% *i % *j);
|
|
|
|
}
|
2005-02-08 15:23:55 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-12-13 23:04:48 +02:00
|
|
|
/* Check the `referrers' table. */
|
2007-01-14 19:28:30 +02:00
|
|
|
printMsg(lvlInfo, "checking the referrers table");
|
|
|
|
Strings referrers;
|
|
|
|
nixDB.enumTable(txn, dbReferrers, referrers);
|
|
|
|
for (Strings::iterator i = referrers.begin(); i != referrers.end(); ++i) {
|
|
|
|
|
|
|
|
/* Decode the entry (it's a tuple of paths). */
|
|
|
|
string::size_type nul = i->find((char) 0);
|
|
|
|
if (nul == string::npos) {
|
|
|
|
printMsg(lvlError, format("removing bad referrer table entry `%1%'") % *i);
|
|
|
|
nixDB.delPair(txn, dbReferrers, *i);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
Path to(*i, 0, nul);
|
|
|
|
Path from(*i, nul + 1);
|
|
|
|
|
2007-08-12 03:29:28 +03:00
|
|
|
if (validPaths.find(to) == validPaths.end()) {
|
|
|
|
printMsg(lvlError, format("removing referrer entry from `%1%' to invalid `%2%'")
|
2007-01-14 19:28:30 +02:00
|
|
|
% from % to);
|
|
|
|
nixDB.delPair(txn, dbReferrers, *i);
|
|
|
|
}
|
|
|
|
|
2007-08-12 03:29:28 +03:00
|
|
|
else if (validPaths.find(from) == validPaths.end()) {
|
|
|
|
printMsg(lvlError, format("removing referrer entry from invalid `%1%' to `%2%'")
|
2007-01-14 19:28:30 +02:00
|
|
|
% from % to);
|
2005-12-13 23:04:48 +02:00
|
|
|
nixDB.delPair(txn, dbReferrers, *i);
|
2005-02-08 15:23:55 +02:00
|
|
|
}
|
2007-01-14 19:28:30 +02:00
|
|
|
|
2005-02-08 15:23:55 +02:00
|
|
|
else {
|
2007-01-14 19:28:30 +02:00
|
|
|
PathSet references;
|
|
|
|
queryReferences(txn, from, references);
|
|
|
|
if (find(references.begin(), references.end(), to) == references.end()) {
|
|
|
|
printMsg(lvlError, format("adding missing referrer mapping from `%1%' to `%2%'")
|
|
|
|
% from % to);
|
|
|
|
references.insert(to);
|
|
|
|
setReferences(txn, from, references);
|
2005-02-08 15:23:55 +02:00
|
|
|
}
|
|
|
|
}
|
2007-01-14 19:28:30 +02:00
|
|
|
|
2003-11-24 11:24:52 +02:00
|
|
|
}
|
2003-10-10 18:14:29 +03:00
|
|
|
|
2007-01-14 19:28:30 +02:00
|
|
|
|
2003-07-31 22:49:11 +03:00
|
|
|
txn.commit();
|
2003-07-17 15:27:55 +03:00
|
|
|
}
|
2005-02-09 11:50:29 +02:00
|
|
|
|
|
|
|
|
|
|
|
/* Upgrade from schema 1 (Nix <= 0.7) to schema 2 (Nix >= 0.8). */
|
2005-12-12 20:24:42 +02:00
|
|
|
static void upgradeStore07()
|
2005-02-09 11:50:29 +02:00
|
|
|
{
|
|
|
|
printMsg(lvlError, "upgrading Nix store to new schema (this may take a while)...");
|
|
|
|
|
|
|
|
Transaction txn(nixDB);
|
|
|
|
|
|
|
|
Paths validPaths2;
|
|
|
|
nixDB.enumTable(txn, dbValidPaths, validPaths2);
|
|
|
|
PathSet validPaths(validPaths2.begin(), validPaths2.end());
|
|
|
|
|
2006-09-05 00:06:23 +03:00
|
|
|
std::cerr << "hashing paths...";
|
2005-02-09 16:37:24 +02:00
|
|
|
int n = 0;
|
2005-02-09 11:50:29 +02:00
|
|
|
for (PathSet::iterator i = validPaths.begin(); i != validPaths.end(); ++i) {
|
|
|
|
checkInterrupt();
|
|
|
|
string s;
|
|
|
|
nixDB.queryString(txn, dbValidPaths, *i, s);
|
|
|
|
if (s == "") {
|
|
|
|
Hash hash = hashPath(htSHA256, *i);
|
|
|
|
setHash(txn, *i, hash);
|
2006-09-05 00:06:23 +03:00
|
|
|
std::cerr << ".";
|
2005-02-09 16:37:24 +02:00
|
|
|
if (++n % 1000 == 0) {
|
|
|
|
txn.commit();
|
|
|
|
txn.begin(nixDB);
|
|
|
|
}
|
2005-02-09 11:50:29 +02:00
|
|
|
}
|
|
|
|
}
|
2006-09-05 00:06:23 +03:00
|
|
|
std::cerr << std::endl;
|
2005-02-09 11:50:29 +02:00
|
|
|
|
2005-02-09 16:37:24 +02:00
|
|
|
txn.commit();
|
|
|
|
|
|
|
|
txn.begin(nixDB);
|
|
|
|
|
2006-09-05 00:06:23 +03:00
|
|
|
std::cerr << "processing closures...";
|
2005-02-09 11:50:29 +02:00
|
|
|
for (PathSet::iterator i = validPaths.begin(); i != validPaths.end(); ++i) {
|
|
|
|
checkInterrupt();
|
|
|
|
if (i->size() > 6 && string(*i, i->size() - 6) == ".store") {
|
|
|
|
ATerm t = ATreadFromNamedFile(i->c_str());
|
|
|
|
if (!t) throw Error(format("cannot read aterm from `%1%'") % *i);
|
|
|
|
|
|
|
|
ATermList roots, elems;
|
|
|
|
if (!matchOldClosure(t, roots, elems)) continue;
|
|
|
|
|
|
|
|
for (ATermIterator j(elems); j; ++j) {
|
|
|
|
|
|
|
|
ATerm path2;
|
|
|
|
ATermList references2;
|
|
|
|
if (!matchOldClosureElem(*j, path2, references2)) continue;
|
|
|
|
|
|
|
|
Path path = aterm2String(path2);
|
|
|
|
if (validPaths.find(path) == validPaths.end())
|
|
|
|
/* Skip this path; it's invalid. This is a normal
|
|
|
|
condition (Nix <= 0.7 did not enforce closure
|
|
|
|
on closure store expressions). */
|
|
|
|
continue;
|
|
|
|
|
|
|
|
PathSet references;
|
|
|
|
for (ATermIterator k(references2); k; ++k) {
|
|
|
|
Path reference = aterm2String(*k);
|
|
|
|
if (validPaths.find(reference) == validPaths.end())
|
|
|
|
/* Bad reference. Set it anyway and let the
|
|
|
|
user fix it. */
|
|
|
|
printMsg(lvlError, format("closure `%1%' contains reference from `%2%' "
|
|
|
|
"to invalid path `%3%' (run `nix-store --verify')")
|
|
|
|
% *i % path % reference);
|
|
|
|
references.insert(reference);
|
|
|
|
}
|
|
|
|
|
|
|
|
PathSet prevReferences;
|
|
|
|
queryReferences(txn, path, prevReferences);
|
|
|
|
if (prevReferences.size() > 0 && references != prevReferences)
|
|
|
|
printMsg(lvlError, format("warning: conflicting references for `%1%'") % path);
|
|
|
|
|
|
|
|
if (references != prevReferences)
|
|
|
|
setReferences(txn, path, references);
|
|
|
|
}
|
|
|
|
|
2006-09-05 00:06:23 +03:00
|
|
|
std::cerr << ".";
|
2005-02-09 11:50:29 +02:00
|
|
|
}
|
|
|
|
}
|
2006-09-05 00:06:23 +03:00
|
|
|
std::cerr << std::endl;
|
2005-02-09 11:50:29 +02:00
|
|
|
|
|
|
|
/* !!! maybe this transaction is way too big */
|
|
|
|
txn.commit();
|
|
|
|
}
|
2005-12-12 20:24:42 +02:00
|
|
|
|
|
|
|
|
|
|
|
/* Upgrade from schema 2 (0.8 <= Nix <= 0.9) to schema 3 (Nix >=
|
|
|
|
0.10). The only thing to do here is to upgrade the old `referer'
|
|
|
|
table (which causes quadratic complexity in some cases) to the new
|
|
|
|
(and properly spelled) `referrer' table. */
|
|
|
|
static void upgradeStore09()
|
|
|
|
{
|
2005-12-15 18:53:21 +02:00
|
|
|
/* !!! we should disallow concurrent upgrades */
|
|
|
|
|
2005-12-12 20:24:42 +02:00
|
|
|
if (!pathExists(nixDBPath + "/referers")) return;
|
|
|
|
|
2007-08-13 14:37:39 +03:00
|
|
|
printMsg(lvlError, "upgrading Nix store to new schema (this may take a while)...");
|
|
|
|
|
2005-12-12 20:24:42 +02:00
|
|
|
Transaction txn(nixDB);
|
|
|
|
|
2006-09-05 00:06:23 +03:00
|
|
|
std::cerr << "converting referers to referrers...";
|
2005-12-12 20:24:42 +02:00
|
|
|
|
|
|
|
TableId dbReferers = nixDB.openTable("referers"); /* sic! */
|
|
|
|
|
|
|
|
Paths referersKeys;
|
|
|
|
nixDB.enumTable(txn, dbReferers, referersKeys);
|
2005-12-15 18:53:21 +02:00
|
|
|
|
|
|
|
int n = 0;
|
2005-12-12 20:24:42 +02:00
|
|
|
for (Paths::iterator i = referersKeys.begin();
|
|
|
|
i != referersKeys.end(); ++i)
|
|
|
|
{
|
|
|
|
Paths referers;
|
|
|
|
nixDB.queryStrings(txn, dbReferers, *i, referers);
|
|
|
|
for (Paths::iterator j = referers.begin();
|
|
|
|
j != referers.end(); ++j)
|
|
|
|
nixDB.setString(txn, dbReferrers, addPrefix(*i, *j), "");
|
2005-12-15 18:53:21 +02:00
|
|
|
if (++n % 1000 == 0) {
|
|
|
|
txn.commit();
|
|
|
|
txn.begin(nixDB);
|
2006-09-05 00:06:23 +03:00
|
|
|
std::cerr << "|";
|
2005-12-15 18:53:21 +02:00
|
|
|
}
|
2006-09-05 00:06:23 +03:00
|
|
|
std::cerr << ".";
|
2005-12-12 20:24:42 +02:00
|
|
|
}
|
2005-12-12 21:14:38 +02:00
|
|
|
|
2005-12-12 20:24:42 +02:00
|
|
|
txn.commit();
|
2005-12-15 18:53:21 +02:00
|
|
|
|
2006-09-05 00:06:23 +03:00
|
|
|
std::cerr << std::endl;
|
2005-12-12 21:14:38 +02:00
|
|
|
|
|
|
|
nixDB.closeTable(dbReferers);
|
|
|
|
|
|
|
|
nixDB.deleteTable("referers");
|
2005-12-12 20:24:42 +02:00
|
|
|
}
|
2006-09-05 00:06:23 +03:00
|
|
|
|
|
|
|
|
2007-08-13 14:37:39 +03:00
|
|
|
/* Upgrade from schema 3 (Nix 0.10) to schema 4 (Nix >= 0.11). The
|
|
|
|
only thing to do here is to delete the substitutes table and get
|
|
|
|
rid of invalid but substitutable references/referrers. */
|
|
|
|
static void upgradeStore11()
|
|
|
|
{
|
|
|
|
if (!pathExists(nixDBPath + "/substitutes")) return;
|
|
|
|
|
|
|
|
printMsg(lvlError, "upgrading Nix store to new schema (this may take a while)...");
|
|
|
|
|
|
|
|
Transaction txn(nixDB);
|
|
|
|
TableId dbSubstitutes = nixDB.openTable("substitutes");
|
|
|
|
|
|
|
|
Paths subKeys;
|
|
|
|
nixDB.enumTable(txn, dbSubstitutes, subKeys);
|
|
|
|
for (Paths::iterator i = subKeys.begin(); i != subKeys.end(); ++i) {
|
|
|
|
if (!isValidPathTxn(txn, *i))
|
|
|
|
invalidatePath(txn, *i);
|
|
|
|
}
|
|
|
|
|
|
|
|
txn.commit();
|
|
|
|
nixDB.closeTable(dbSubstitutes);
|
|
|
|
nixDB.deleteTable("substitutes");
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-09-05 00:06:23 +03:00
|
|
|
}
|