2010-05-31 18:36:24 +02:00
|
|
|
#include "xmlgraph.hh"
|
|
|
|
#include "util.hh"
|
|
|
|
#include "store-api.hh"
|
|
|
|
|
|
|
|
#include <iostream>
|
|
|
|
|
|
|
|
|
|
|
|
using std::cout;
|
|
|
|
|
|
|
|
namespace nix {
|
|
|
|
|
|
|
|
|
|
|
|
static inline const string & xmlQuote(const string & s)
|
|
|
|
{
|
|
|
|
// Luckily, store paths shouldn't contain any character that needs to be
|
|
|
|
// quoted.
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static string makeEdge(const string & src, const string & dst)
|
|
|
|
{
|
|
|
|
format f = format(" <edge src=\"%1%\" dst=\"%2%\"/>\n")
|
|
|
|
% xmlQuote(src) % xmlQuote(dst);
|
|
|
|
return f.str();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static string makeNode(const string & id)
|
|
|
|
{
|
|
|
|
format f = format(" <node name=\"%1%\"/>\n") % xmlQuote(id);
|
|
|
|
return f.str();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-02-04 14:48:42 +01:00
|
|
|
void printXmlGraph(Store & store, const PathSet & roots)
|
2010-05-31 18:36:24 +02:00
|
|
|
{
|
|
|
|
PathSet workList(roots);
|
|
|
|
PathSet doneSet;
|
|
|
|
|
|
|
|
cout << "<?xml version='1.0' encoding='utf-8'?>\n"
|
2016-01-28 16:03:32 +01:00
|
|
|
<< "<nix>\n";
|
2010-05-31 18:36:24 +02:00
|
|
|
|
|
|
|
while (!workList.empty()) {
|
2016-01-28 16:03:32 +01:00
|
|
|
Path path = *(workList.begin());
|
|
|
|
workList.erase(path);
|
2010-05-31 18:36:24 +02:00
|
|
|
|
2016-01-28 16:03:32 +01:00
|
|
|
if (doneSet.find(path) != doneSet.end()) continue;
|
|
|
|
doneSet.insert(path);
|
2010-05-31 18:36:24 +02:00
|
|
|
|
2016-01-28 16:03:32 +01:00
|
|
|
cout << makeNode(path);
|
2010-05-31 18:36:24 +02:00
|
|
|
|
2016-01-28 16:03:32 +01:00
|
|
|
PathSet references;
|
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 14:28:26 +01:00
|
|
|
store.queryReferences(path, references);
|
2010-05-31 18:36:24 +02:00
|
|
|
|
2016-01-28 16:03:32 +01:00
|
|
|
for (PathSet::iterator i = references.begin();
|
|
|
|
i != references.end(); ++i)
|
|
|
|
{
|
|
|
|
if (*i != path) {
|
|
|
|
workList.insert(*i);
|
|
|
|
cout << makeEdge(*i, path);
|
|
|
|
}
|
|
|
|
}
|
2010-05-31 18:36:24 +02:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
cout << "</nix>\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
}
|