2003-06-23 13:27:59 +00:00
|
|
|
#include <iostream>
|
|
|
|
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/wait.h>
|
|
|
|
|
2003-07-07 07:43:58 +00:00
|
|
|
#include "store.hh"
|
2003-06-16 13:33:38 +00:00
|
|
|
#include "globals.hh"
|
|
|
|
#include "db.hh"
|
2003-06-23 13:27:59 +00:00
|
|
|
#include "archive.hh"
|
2003-08-04 07:09:36 +00:00
|
|
|
#include "pathlocks.hh"
|
2003-07-20 19:29:38 +00:00
|
|
|
#include "normalise.hh"
|
2003-06-23 13:27:59 +00:00
|
|
|
|
|
|
|
|
|
|
|
struct CopySink : DumpSink
|
|
|
|
{
|
|
|
|
int fd;
|
|
|
|
virtual void operator () (const unsigned char * data, unsigned int len)
|
|
|
|
{
|
2003-07-20 21:11:43 +00:00
|
|
|
writeFull(fd, data, len);
|
2003-06-23 13:27:59 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
struct CopySource : RestoreSource
|
|
|
|
{
|
|
|
|
int fd;
|
2003-07-20 21:11:43 +00:00
|
|
|
virtual void operator () (unsigned char * data, unsigned int len)
|
2003-06-23 13:27:59 +00:00
|
|
|
{
|
2003-07-20 21:11:43 +00:00
|
|
|
readFull(fd, data, len);
|
2003-06-23 13:27:59 +00:00
|
|
|
}
|
|
|
|
};
|
2003-06-16 13:33:38 +00:00
|
|
|
|
|
|
|
|
2003-07-06 14:20:47 +00:00
|
|
|
void copyPath(string src, string dst)
|
2003-06-16 13:33:38 +00:00
|
|
|
{
|
2003-07-31 16:05:35 +00:00
|
|
|
debug(format("copying `%1%' to `%2%'") % src % dst);
|
|
|
|
|
2003-06-23 13:27:59 +00:00
|
|
|
/* Unfortunately C++ doesn't support coprocedures, so we have no
|
|
|
|
nice way to chain CopySink and CopySource together. Instead we
|
|
|
|
fork off a child to run the sink. (Fork-less platforms should
|
|
|
|
use a thread). */
|
|
|
|
|
|
|
|
/* Create a pipe. */
|
|
|
|
int fds[2];
|
|
|
|
if (pipe(fds) == -1) throw SysError("creating pipe");
|
|
|
|
|
|
|
|
/* Fork. */
|
|
|
|
pid_t pid;
|
|
|
|
switch (pid = fork()) {
|
|
|
|
|
|
|
|
case -1:
|
|
|
|
throw SysError("unable to fork");
|
|
|
|
|
|
|
|
case 0: /* child */
|
|
|
|
try {
|
|
|
|
close(fds[1]);
|
|
|
|
CopySource source;
|
|
|
|
source.fd = fds[0];
|
|
|
|
restorePath(dst, source);
|
|
|
|
_exit(0);
|
|
|
|
} catch (exception & e) {
|
|
|
|
cerr << "error: " << e.what() << endl;
|
|
|
|
}
|
|
|
|
_exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
close(fds[0]);
|
|
|
|
|
|
|
|
/* Parent. */
|
|
|
|
|
|
|
|
CopySink sink;
|
|
|
|
sink.fd = fds[1];
|
|
|
|
dumpPath(src, sink);
|
|
|
|
|
|
|
|
/* Wait for the child to finish. */
|
|
|
|
int status;
|
|
|
|
if (waitpid(pid, &status, 0) != pid)
|
|
|
|
throw SysError("waiting for child");
|
|
|
|
|
|
|
|
if (!WIFEXITED(status) || WEXITSTATUS(status) != 0)
|
|
|
|
throw Error("cannot copy file: child died");
|
2003-06-16 13:33:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-07-15 16:28:54 +00:00
|
|
|
void registerSubstitute(const FSId & srcId, const FSId & subId)
|
2003-07-10 15:11:48 +00:00
|
|
|
{
|
2003-07-16 20:00:51 +00:00
|
|
|
#if 0
|
2003-07-10 15:11:48 +00:00
|
|
|
Strings subs;
|
2003-07-15 16:28:54 +00:00
|
|
|
queryListDB(nixDB, dbSubstitutes, srcId, subs); /* non-existence = ok */
|
2003-07-10 15:11:48 +00:00
|
|
|
|
|
|
|
for (Strings::iterator it = subs.begin(); it != subs.end(); it++)
|
2003-07-15 16:28:54 +00:00
|
|
|
if (parseHash(*it) == subId) return;
|
2003-07-10 15:11:48 +00:00
|
|
|
|
2003-07-15 16:28:54 +00:00
|
|
|
subs.push_back(subId);
|
2003-07-10 15:11:48 +00:00
|
|
|
|
2003-07-16 20:00:51 +00:00
|
|
|
setListDB(nixDB, dbSubstitutes, srcId, subs);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* For now, accept only one substitute per id. */
|
|
|
|
Strings subs;
|
|
|
|
subs.push_back(subId);
|
2003-07-31 16:05:35 +00:00
|
|
|
|
|
|
|
Transaction txn(nixDB);
|
|
|
|
nixDB.setStrings(txn, dbSubstitutes, srcId, subs);
|
|
|
|
txn.commit();
|
2003-07-10 15:11:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-08-01 15:41:47 +00:00
|
|
|
void registerPath(const Transaction & txn,
|
|
|
|
const string & _path, const FSId & id)
|
2003-07-07 09:25:26 +00:00
|
|
|
{
|
|
|
|
string path(canonPath(_path));
|
|
|
|
|
2003-07-31 16:05:35 +00:00
|
|
|
debug(format("registering path `%1%' with id %2%")
|
|
|
|
% path % (string) id);
|
2003-07-07 09:25:26 +00:00
|
|
|
|
2003-07-31 16:05:35 +00:00
|
|
|
string oldId;
|
|
|
|
if (nixDB.queryString(txn, dbPath2Id, path, oldId)) {
|
|
|
|
if (id != parseHash(oldId))
|
|
|
|
throw Error(format("path `%1%' already contains id %2%")
|
|
|
|
% path % oldId);
|
|
|
|
return;
|
|
|
|
}
|
2003-07-07 09:25:26 +00:00
|
|
|
|
2003-07-31 16:05:35 +00:00
|
|
|
nixDB.setString(txn, dbPath2Id, path, id);
|
|
|
|
|
|
|
|
Strings paths;
|
|
|
|
nixDB.queryStrings(txn, dbId2Paths, id, paths); /* non-existence = ok */
|
2003-07-07 09:25:26 +00:00
|
|
|
|
|
|
|
paths.push_back(path);
|
|
|
|
|
2003-07-31 16:05:35 +00:00
|
|
|
nixDB.setStrings(txn, dbId2Paths, id, paths);
|
2003-07-07 09:25:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-07-08 09:54:47 +00:00
|
|
|
void unregisterPath(const string & _path)
|
|
|
|
{
|
|
|
|
string path(canonPath(_path));
|
2003-07-31 16:05:35 +00:00
|
|
|
Transaction txn(nixDB);
|
|
|
|
|
|
|
|
debug(format("unregistering path `%1%'") % path);
|
2003-07-08 09:54:47 +00:00
|
|
|
|
2003-07-15 16:28:54 +00:00
|
|
|
string _id;
|
2003-07-31 16:05:35 +00:00
|
|
|
if (!nixDB.queryString(txn, dbPath2Id, path, _id)) {
|
|
|
|
txn.abort();
|
|
|
|
return;
|
|
|
|
}
|
2003-07-15 16:28:54 +00:00
|
|
|
FSId id(parseHash(_id));
|
|
|
|
|
2003-07-31 16:05:35 +00:00
|
|
|
nixDB.delPair(txn, dbPath2Id, path);
|
2003-07-16 20:00:51 +00:00
|
|
|
|
2003-07-08 09:54:47 +00:00
|
|
|
Strings paths, paths2;
|
2003-07-31 16:05:35 +00:00
|
|
|
nixDB.queryStrings(txn, dbId2Paths, id, paths); /* non-existence = ok */
|
2003-07-08 09:54:47 +00:00
|
|
|
|
|
|
|
for (Strings::iterator it = paths.begin();
|
|
|
|
it != paths.end(); it++)
|
2003-07-31 16:05:35 +00:00
|
|
|
if (*it != path) paths2.push_back(*it);
|
2003-07-08 09:54:47 +00:00
|
|
|
|
2003-07-31 16:05:35 +00:00
|
|
|
nixDB.setStrings(txn, dbId2Paths, id, paths2);
|
2003-07-16 20:00:51 +00:00
|
|
|
|
2003-07-31 16:05:35 +00:00
|
|
|
txn.commit();
|
2003-07-08 09:54:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-07-21 14:46:01 +00:00
|
|
|
bool queryPathId(const string & path, FSId & id)
|
|
|
|
{
|
|
|
|
string s;
|
2003-07-31 13:47:13 +00:00
|
|
|
if (!nixDB.queryString(noTxn, dbPath2Id, absPath(path), s)) return false;
|
2003-07-21 14:46:01 +00:00
|
|
|
id = parseHash(s);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-07-07 09:25:26 +00:00
|
|
|
bool isInPrefix(const string & path, const string & _prefix)
|
|
|
|
{
|
|
|
|
string prefix = canonPath(_prefix + "/");
|
|
|
|
return string(path, 0, prefix.size()) == prefix;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-07-15 16:28:54 +00:00
|
|
|
string expandId(const FSId & id, const string & target,
|
2003-08-01 09:01:51 +00:00
|
|
|
const string & prefix, FSIdSet pending, bool ignoreSubstitutes)
|
2003-07-07 09:25:26 +00:00
|
|
|
{
|
2003-07-24 08:53:43 +00:00
|
|
|
Nest nest(lvlDebug, format("expanding %1%") % (string) id);
|
2003-07-22 15:15:15 +00:00
|
|
|
|
2003-07-07 09:25:26 +00:00
|
|
|
Strings paths;
|
|
|
|
|
2003-07-10 13:41:28 +00:00
|
|
|
if (!target.empty() && !isInPrefix(target, prefix))
|
|
|
|
abort();
|
|
|
|
|
2003-07-31 13:47:13 +00:00
|
|
|
nixDB.queryStrings(noTxn, dbId2Paths, id, paths);
|
2003-07-10 13:41:28 +00:00
|
|
|
|
|
|
|
/* Pick one equal to `target'. */
|
|
|
|
if (!target.empty()) {
|
|
|
|
|
|
|
|
for (Strings::iterator i = paths.begin();
|
|
|
|
i != paths.end(); i++)
|
|
|
|
{
|
|
|
|
string path = *i;
|
2003-07-15 16:28:54 +00:00
|
|
|
if (path == target && pathExists(path))
|
|
|
|
return path;
|
2003-07-10 13:41:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2003-07-07 09:25:26 +00:00
|
|
|
|
2003-07-10 13:41:28 +00:00
|
|
|
/* Arbitrarily pick the first one that exists and isn't stale. */
|
2003-07-07 09:25:26 +00:00
|
|
|
for (Strings::iterator it = paths.begin();
|
|
|
|
it != paths.end(); it++)
|
|
|
|
{
|
|
|
|
string path = *it;
|
2003-07-15 16:28:54 +00:00
|
|
|
if (isInPrefix(path, prefix) && pathExists(path)) {
|
|
|
|
if (target.empty())
|
|
|
|
return path;
|
|
|
|
else {
|
2003-08-04 07:09:36 +00:00
|
|
|
/* Acquire a lock on the target path. */
|
|
|
|
Strings lockPaths;
|
|
|
|
lockPaths.push_back(target);
|
|
|
|
PathLocks outputLock(lockPaths);
|
|
|
|
|
|
|
|
/* Copy. */
|
2003-07-15 16:28:54 +00:00
|
|
|
copyPath(path, target);
|
2003-08-04 07:09:36 +00:00
|
|
|
|
|
|
|
/* Register the target path. */
|
2003-08-01 15:41:47 +00:00
|
|
|
Transaction txn(nixDB);
|
|
|
|
registerPath(txn, target, id);
|
|
|
|
txn.commit();
|
2003-08-04 07:09:36 +00:00
|
|
|
|
2003-07-15 16:28:54 +00:00
|
|
|
return target;
|
2003-07-10 13:41:28 +00:00
|
|
|
}
|
2003-07-07 09:25:26 +00:00
|
|
|
}
|
|
|
|
}
|
2003-07-10 15:11:48 +00:00
|
|
|
|
2003-08-01 09:01:51 +00:00
|
|
|
if (!ignoreSubstitutes) {
|
|
|
|
|
|
|
|
if (pending.find(id) != pending.end())
|
|
|
|
throw Error(format("id %1% already being expanded") % (string) id);
|
|
|
|
pending.insert(id);
|
2003-07-10 15:11:48 +00:00
|
|
|
|
2003-08-01 09:01:51 +00:00
|
|
|
/* Try to realise the substitutes, but only if this id is not
|
|
|
|
already being realised by a substitute. */
|
|
|
|
Strings subs;
|
|
|
|
nixDB.queryStrings(noTxn, dbSubstitutes, id, subs); /* non-existence = ok */
|
2003-07-10 15:11:48 +00:00
|
|
|
|
2003-08-01 09:01:51 +00:00
|
|
|
for (Strings::iterator it = subs.begin(); it != subs.end(); it++) {
|
|
|
|
FSId subId = parseHash(*it);
|
2003-07-22 15:15:15 +00:00
|
|
|
|
2003-08-01 09:01:51 +00:00
|
|
|
debug(format("trying substitute %1%") % (string) subId);
|
2003-07-22 15:15:15 +00:00
|
|
|
|
2003-08-01 09:01:51 +00:00
|
|
|
realiseSlice(normaliseFState(subId, pending), pending);
|
|
|
|
|
|
|
|
return expandId(id, target, prefix, pending);
|
|
|
|
}
|
2003-07-22 15:15:15 +00:00
|
|
|
|
2003-07-10 15:11:48 +00:00
|
|
|
}
|
2003-07-07 09:25:26 +00:00
|
|
|
|
2003-07-15 16:28:54 +00:00
|
|
|
throw Error(format("cannot expand id `%1%'") % (string) id);
|
2003-07-07 09:25:26 +00:00
|
|
|
}
|
|
|
|
|
2003-07-09 16:12:40 +00:00
|
|
|
|
2003-07-15 16:28:54 +00:00
|
|
|
void addToStore(string srcPath, string & dstPath, FSId & id,
|
2003-07-11 08:41:03 +00:00
|
|
|
bool deterministicName)
|
2003-06-16 13:33:38 +00:00
|
|
|
{
|
2003-07-31 16:05:35 +00:00
|
|
|
debug(format("adding `%1%' to the store") % srcPath);
|
|
|
|
|
2003-06-27 13:55:12 +00:00
|
|
|
srcPath = absPath(srcPath);
|
2003-07-15 16:28:54 +00:00
|
|
|
id = hashPath(srcPath);
|
2003-06-16 13:33:38 +00:00
|
|
|
|
2003-07-11 08:41:03 +00:00
|
|
|
string baseName = baseNameOf(srcPath);
|
2003-07-15 16:28:54 +00:00
|
|
|
dstPath = canonPath(nixStore + "/" + (string) id + "-" + baseName);
|
2003-07-11 08:41:03 +00:00
|
|
|
|
2003-07-07 09:25:26 +00:00
|
|
|
try {
|
2003-08-01 09:01:51 +00:00
|
|
|
dstPath = expandId(id, deterministicName ? dstPath : "",
|
|
|
|
nixStore, FSIdSet(), true);
|
2003-06-27 13:55:12 +00:00
|
|
|
return;
|
2003-07-07 09:25:26 +00:00
|
|
|
} catch (...) {
|
2003-06-16 13:33:38 +00:00
|
|
|
}
|
2003-07-07 09:25:26 +00:00
|
|
|
|
2003-08-04 07:09:36 +00:00
|
|
|
Strings lockPaths;
|
|
|
|
lockPaths.push_back(dstPath);
|
|
|
|
PathLocks outputLock(lockPaths);
|
|
|
|
|
2003-07-06 14:20:47 +00:00
|
|
|
copyPath(srcPath, dstPath);
|
2003-08-04 07:09:36 +00:00
|
|
|
|
2003-08-01 15:41:47 +00:00
|
|
|
Transaction txn(nixDB);
|
|
|
|
registerPath(txn, dstPath, id);
|
|
|
|
txn.commit();
|
2003-06-16 13:33:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-06-27 14:56:12 +00:00
|
|
|
void deleteFromStore(const string & path)
|
2003-06-23 14:40:49 +00:00
|
|
|
{
|
2003-07-09 16:12:40 +00:00
|
|
|
string prefix = + "/";
|
|
|
|
if (!isInPrefix(path, nixStore))
|
2003-06-27 14:56:12 +00:00
|
|
|
throw Error(format("path %1% is not in the store") % path);
|
2003-07-08 09:54:47 +00:00
|
|
|
|
|
|
|
unregisterPath(path);
|
|
|
|
|
2003-06-27 14:56:12 +00:00
|
|
|
deletePath(path);
|
2003-06-23 14:40:49 +00:00
|
|
|
}
|
2003-07-17 12:27:55 +00:00
|
|
|
|
|
|
|
|
|
|
|
void verifyStore()
|
|
|
|
{
|
2003-07-31 19:49:11 +00:00
|
|
|
Transaction txn(nixDB);
|
|
|
|
|
|
|
|
/* !!! verify that the result is consistent */
|
|
|
|
|
2003-07-17 12:27:55 +00:00
|
|
|
Strings paths;
|
2003-07-31 19:49:11 +00:00
|
|
|
nixDB.enumTable(txn, dbPath2Id, paths);
|
2003-07-17 12:27:55 +00:00
|
|
|
|
|
|
|
for (Strings::iterator i = paths.begin();
|
|
|
|
i != paths.end(); i++)
|
|
|
|
{
|
|
|
|
bool erase = true;
|
|
|
|
string path = *i;
|
|
|
|
|
|
|
|
if (!pathExists(path)) {
|
|
|
|
debug(format("path `%1%' disappeared") % path);
|
|
|
|
}
|
|
|
|
|
|
|
|
else {
|
|
|
|
string id;
|
2003-07-31 19:49:11 +00:00
|
|
|
if (!nixDB.queryString(txn, dbPath2Id, path, id)) abort();
|
2003-07-17 12:27:55 +00:00
|
|
|
|
|
|
|
Strings idPaths;
|
2003-07-31 19:49:11 +00:00
|
|
|
nixDB.queryStrings(txn, dbId2Paths, id, idPaths);
|
2003-07-17 12:27:55 +00:00
|
|
|
|
|
|
|
bool found = false;
|
|
|
|
for (Strings::iterator j = idPaths.begin();
|
|
|
|
j != idPaths.end(); j++)
|
|
|
|
if (path == *j) {
|
|
|
|
found = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (found)
|
|
|
|
erase = false;
|
|
|
|
else
|
|
|
|
/* !!! perhaps we should add path to idPaths? */
|
|
|
|
debug(format("reverse mapping for path `%1%' missing") % path);
|
|
|
|
}
|
|
|
|
|
2003-07-31 19:49:11 +00:00
|
|
|
if (erase) nixDB.delPair(txn, dbPath2Id, path);
|
2003-07-17 12:27:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Strings ids;
|
2003-07-31 19:49:11 +00:00
|
|
|
nixDB.enumTable(txn, dbId2Paths, ids);
|
2003-07-17 12:27:55 +00:00
|
|
|
|
|
|
|
for (Strings::iterator i = ids.begin();
|
|
|
|
i != ids.end(); i++)
|
|
|
|
{
|
|
|
|
FSId id = parseHash(*i);
|
|
|
|
|
|
|
|
Strings idPaths;
|
2003-07-31 19:49:11 +00:00
|
|
|
nixDB.queryStrings(txn, dbId2Paths, id, idPaths);
|
2003-07-17 12:27:55 +00:00
|
|
|
|
|
|
|
for (Strings::iterator j = idPaths.begin();
|
|
|
|
j != idPaths.end(); )
|
|
|
|
{
|
|
|
|
string id2;
|
2003-07-31 19:49:11 +00:00
|
|
|
if (!nixDB.queryString(txn, dbPath2Id, *j, id2) ||
|
2003-07-17 12:27:55 +00:00
|
|
|
id != parseHash(id2)) {
|
|
|
|
debug(format("erasing path `%1%' from mapping for id %2%")
|
|
|
|
% *j % (string) id);
|
|
|
|
j = idPaths.erase(j);
|
|
|
|
} else j++;
|
|
|
|
}
|
|
|
|
|
2003-07-31 19:49:11 +00:00
|
|
|
nixDB.setStrings(txn, dbId2Paths, id, idPaths);
|
2003-07-17 12:27:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Strings subs;
|
2003-07-31 19:49:11 +00:00
|
|
|
nixDB.enumTable(txn, dbSubstitutes, subs);
|
2003-07-17 12:27:55 +00:00
|
|
|
|
|
|
|
for (Strings::iterator i = subs.begin();
|
|
|
|
i != subs.end(); i++)
|
|
|
|
{
|
|
|
|
FSId srcId = parseHash(*i);
|
|
|
|
|
|
|
|
Strings subIds;
|
2003-07-31 19:49:11 +00:00
|
|
|
nixDB.queryStrings(txn, dbSubstitutes, srcId, subIds);
|
2003-07-17 12:27:55 +00:00
|
|
|
|
|
|
|
for (Strings::iterator j = subIds.begin();
|
|
|
|
j != subIds.end(); )
|
|
|
|
{
|
|
|
|
FSId subId = parseHash(*j);
|
|
|
|
|
|
|
|
Strings subPaths;
|
2003-07-31 19:49:11 +00:00
|
|
|
nixDB.queryStrings(txn, dbId2Paths, subId, subPaths);
|
2003-07-17 12:27:55 +00:00
|
|
|
if (subPaths.size() == 0) {
|
|
|
|
debug(format("erasing substitute %1% for %2%")
|
|
|
|
% (string) subId % (string) srcId);
|
|
|
|
j = subIds.erase(j);
|
|
|
|
} else j++;
|
|
|
|
}
|
|
|
|
|
2003-07-31 19:49:11 +00:00
|
|
|
nixDB.setStrings(txn, dbSubstitutes, srcId, subIds);
|
2003-07-17 12:27:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Strings sucs;
|
2003-07-31 19:49:11 +00:00
|
|
|
nixDB.enumTable(txn, dbSuccessors, sucs);
|
2003-07-17 12:27:55 +00:00
|
|
|
|
|
|
|
for (Strings::iterator i = sucs.begin();
|
|
|
|
i != sucs.end(); i++)
|
|
|
|
{
|
|
|
|
FSId id1 = parseHash(*i);
|
|
|
|
|
|
|
|
string id2;
|
2003-07-31 19:49:11 +00:00
|
|
|
if (!nixDB.queryString(txn, dbSuccessors, id1, id2)) abort();
|
2003-07-17 12:27:55 +00:00
|
|
|
|
|
|
|
Strings id2Paths;
|
2003-07-31 19:49:11 +00:00
|
|
|
nixDB.queryStrings(txn, dbId2Paths, id2, id2Paths);
|
2003-07-17 12:27:55 +00:00
|
|
|
if (id2Paths.size() == 0) {
|
|
|
|
Strings id2Subs;
|
2003-07-31 19:49:11 +00:00
|
|
|
nixDB.queryStrings(txn, dbSubstitutes, id2, id2Subs);
|
2003-07-17 12:27:55 +00:00
|
|
|
if (id2Subs.size() == 0) {
|
|
|
|
debug(format("successor %1% for %2% missing")
|
|
|
|
% id2 % (string) id1);
|
2003-07-31 19:49:11 +00:00
|
|
|
nixDB.delPair(txn, dbSuccessors, (string) id1);
|
2003-07-17 12:27:55 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2003-07-31 19:49:11 +00:00
|
|
|
|
|
|
|
txn.commit();
|
2003-07-17 12:27:55 +00:00
|
|
|
}
|