forked from lix-project/lix
75989bdca7
The fact that queryPathInfo() is synchronous meant that we needed a thread for every concurrent binary cache lookup, even though they end up being handled by the same download thread. Requiring hundreds of threads is not a good idea. So now there is an asynchronous version of queryPathInfo() that takes a callback function to process the result. Similarly, enqueueDownload() now takes a callback rather than returning a future. Thus, a command like nix path-info --store https://cache.nixos.org/ -r /nix/store/slljrzwmpygy1daay14kjszsr9xix063-nixos-16.09beta231.dccf8c5 that returns 4941 paths now takes 1.87s using only 2 threads (the main thread and the downloader thread). (This is with a prewarmed CloudFront.)
103 lines
2.6 KiB
C++
103 lines
2.6 KiB
C++
#include "binary-cache-store.hh"
|
||
#include "globals.hh"
|
||
#include "nar-info-disk-cache.hh"
|
||
|
||
namespace nix {
|
||
|
||
class LocalBinaryCacheStore : public BinaryCacheStore
|
||
{
|
||
private:
|
||
|
||
Path binaryCacheDir;
|
||
|
||
public:
|
||
|
||
LocalBinaryCacheStore(
|
||
const Params & params, const Path & binaryCacheDir)
|
||
: BinaryCacheStore(params)
|
||
, binaryCacheDir(binaryCacheDir)
|
||
{
|
||
}
|
||
|
||
void init() override;
|
||
|
||
std::string getUri() override
|
||
{
|
||
return "file://" + binaryCacheDir;
|
||
}
|
||
|
||
protected:
|
||
|
||
bool fileExists(const std::string & path) override;
|
||
|
||
void upsertFile(const std::string & path, const std::string & data) override;
|
||
|
||
void getFile(const std::string & path,
|
||
std::function<void(std::shared_ptr<std::string>)> success,
|
||
std::function<void(std::exception_ptr exc)> failure) override
|
||
{
|
||
sync2async<std::shared_ptr<std::string>>(success, failure, [&]() {
|
||
try {
|
||
return std::make_shared<std::string>(readFile(binaryCacheDir + "/" + path));
|
||
} catch (SysError & e) {
|
||
if (e.errNo == ENOENT) return std::shared_ptr<std::string>();
|
||
throw;
|
||
}
|
||
});
|
||
}
|
||
|
||
PathSet queryAllValidPaths() override
|
||
{
|
||
PathSet paths;
|
||
|
||
for (auto & entry : readDirectory(binaryCacheDir)) {
|
||
if (entry.name.size() != 40 ||
|
||
!hasSuffix(entry.name, ".narinfo"))
|
||
continue;
|
||
paths.insert(storeDir + "/" + entry.name.substr(0, entry.name.size() - 8));
|
||
}
|
||
|
||
return paths;
|
||
}
|
||
|
||
};
|
||
|
||
void LocalBinaryCacheStore::init()
|
||
{
|
||
createDirs(binaryCacheDir + "/nar");
|
||
BinaryCacheStore::init();
|
||
}
|
||
|
||
static void atomicWrite(const Path & path, const std::string & s)
|
||
{
|
||
Path tmp = path + ".tmp." + std::to_string(getpid());
|
||
AutoDelete del(tmp, false);
|
||
writeFile(tmp, s);
|
||
if (rename(tmp.c_str(), path.c_str()))
|
||
throw SysError(format("renaming ‘%1%’ to ‘%2%’") % tmp % path);
|
||
del.cancel();
|
||
}
|
||
|
||
bool LocalBinaryCacheStore::fileExists(const std::string & path)
|
||
{
|
||
return pathExists(binaryCacheDir + "/" + path);
|
||
}
|
||
|
||
void LocalBinaryCacheStore::upsertFile(const std::string & path, const std::string & data)
|
||
{
|
||
atomicWrite(binaryCacheDir + "/" + path, data);
|
||
}
|
||
|
||
static RegisterStoreImplementation regStore([](
|
||
const std::string & uri, const Store::Params & params)
|
||
-> std::shared_ptr<Store>
|
||
{
|
||
if (getEnv("_NIX_FORCE_HTTP_BINARY_CACHE_STORE") == "1" ||
|
||
std::string(uri, 0, 7) != "file://")
|
||
return 0;
|
||
auto store = std::make_shared<LocalBinaryCacheStore>(params, std::string(uri, 7));
|
||
store->init();
|
||
return store;
|
||
});
|
||
|
||
}
|