74c917454c
For every file "lib/debug/.build-id/<build-id>.debug" in every store path ending in -debug in a release, this command creates an S3 key "debuginfo/<build-id>" specifying the location of the NAR that contains the debug info file for that build ID. Thus, https://cache.nixos.org/debuginfo allows debug info files to be looked up by build ID. This allows debug info files to be fetched automatically by dwarffs [1]. One minor issue is that in theory, multiple store paths could provide debug info files for a particular build ID. So then when we garbage collect a store path from the binary cache, we migth invalidate a debug info link even though there are non-GC'ed store paths that provide the file. Ah well. I could have used HTTP redirects, but they're kind of a pain to configure with S3. [1] https://github.com/edolstra/dwarffs
98 lines
2.8 KiB
C++
98 lines
2.8 KiB
C++
#include <nix/config.h>
|
||
|
||
#include <regex>
|
||
|
||
#include "shared.hh"
|
||
#include "s3-binary-cache-store.hh"
|
||
#include "thread-pool.hh"
|
||
#include "nar-info.hh"
|
||
|
||
#include "file-cache.hh"
|
||
|
||
// cache.nixos.org/debuginfo/<build-id>
|
||
// => redirect to NAR
|
||
|
||
using namespace nix;
|
||
|
||
void mainWrapped(int argc, char * * argv)
|
||
{
|
||
initNix();
|
||
|
||
if (argc != 4) throw Error("usage: index-debuginfo CACHE-DB BINARY-CACHE-URI STORE-PATHS");
|
||
|
||
Path cacheDbPath = argv[1];
|
||
std::string binaryCacheUri = argv[2];
|
||
Path storePathsFile = argv[3];
|
||
|
||
FileCache fileCache(cacheDbPath);
|
||
|
||
if (hasSuffix(binaryCacheUri, "/")) binaryCacheUri.pop_back();
|
||
auto binaryCache = openStore(binaryCacheUri).cast<S3BinaryCacheStore>();
|
||
|
||
auto storePaths = tokenizeString<PathSet>(readFile(storePathsFile, true));
|
||
|
||
std::regex debugFileRegex("^lib/debug/\\.build-id/[0-9a-f]{2}/[0-9a-f]{38}\\.debug$");
|
||
|
||
ThreadPool threadPool(25);
|
||
|
||
auto doFile = [&](std::string member, std::string key, std::string target) {
|
||
checkInterrupt();
|
||
|
||
nlohmann::json json;
|
||
json["archive"] = target;
|
||
json["member"] = member;
|
||
|
||
// FIXME: or should we overwrite? The previous link may point
|
||
// to a GC'ed file, so overwriting might be useful...
|
||
if (binaryCache->fileExists(key)) return;
|
||
|
||
printError("redirecting ‘%s’ to ‘%s’", key, target);
|
||
|
||
binaryCache->upsertFile(key, json.dump(), "application/json");
|
||
};
|
||
|
||
auto doPath = [&](const Path & storePath) {
|
||
checkInterrupt();
|
||
|
||
try {
|
||
auto files = fileCache.getFiles(binaryCache, storePath);
|
||
|
||
std::string prefix = "lib/debug/.build-id/";
|
||
|
||
for (auto & file : files) {
|
||
if (file.second.type != FSAccessor::Type::tRegular
|
||
|| !std::regex_match(file.first, debugFileRegex))
|
||
continue;
|
||
|
||
std::string buildId =
|
||
std::string(file.first, prefix.size(), 2) +
|
||
std::string(file.first, prefix.size() + 3, 38);
|
||
|
||
auto info = binaryCache->queryPathInfo(storePath).cast<const NarInfo>();
|
||
|
||
assert(hasPrefix(info->url, "nar/"));
|
||
|
||
std::string key = "debuginfo/" + buildId;
|
||
std::string target = "../" + info->url;
|
||
|
||
threadPool.enqueue(std::bind(doFile, file.first, key, target));
|
||
}
|
||
|
||
} catch (BadJSON & e) {
|
||
printError("error: in %s: %s", storePath, e.what());
|
||
}
|
||
};
|
||
|
||
for (auto & storePath : storePaths)
|
||
if (hasSuffix(storePath, "-debug"))
|
||
threadPool.enqueue(std::bind(doPath, storePath));
|
||
|
||
threadPool.process();
|
||
}
|
||
|
||
int main(int argc, char * * argv)
|
||
{
|
||
return handleExceptions(argv[0], [&]() {
|
||
mainWrapped(argc, argv);
|
||
});
|
||
}
|