2016-02-29 15:14:39 +00:00
|
|
|
#include "binary-cache-store.hh"
|
2016-03-04 16:23:42 +00:00
|
|
|
#include "globals.hh"
|
2016-05-30 12:53:57 +00:00
|
|
|
#include "nar-info-disk-cache.hh"
|
2016-02-24 13:48:16 +00:00
|
|
|
|
|
|
|
namespace nix {
|
|
|
|
|
2016-02-29 15:14:39 +00:00
|
|
|
class LocalBinaryCacheStore : public BinaryCacheStore
|
|
|
|
{
|
|
|
|
private:
|
|
|
|
|
|
|
|
Path binaryCacheDir;
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
2016-05-04 18:15:41 +00:00
|
|
|
LocalBinaryCacheStore(
|
2016-06-01 12:49:12 +00:00
|
|
|
const Params & params, const Path & binaryCacheDir)
|
2016-05-04 18:15:41 +00:00
|
|
|
: BinaryCacheStore(params)
|
2016-04-29 14:47:20 +00:00
|
|
|
, binaryCacheDir(binaryCacheDir)
|
|
|
|
{
|
|
|
|
}
|
2016-02-29 15:14:39 +00:00
|
|
|
|
|
|
|
void init() override;
|
|
|
|
|
2016-04-29 14:26:16 +00:00
|
|
|
std::string getUri() override
|
|
|
|
{
|
|
|
|
return "file://" + binaryCacheDir;
|
|
|
|
}
|
|
|
|
|
2016-02-29 15:14:39 +00:00
|
|
|
protected:
|
|
|
|
|
|
|
|
bool fileExists(const std::string & path) override;
|
|
|
|
|
2017-03-14 14:26:01 +00:00
|
|
|
void upsertFile(const std::string & path,
|
|
|
|
const std::string & data,
|
|
|
|
const std::string & mimeType) override;
|
2016-02-29 15:14:39 +00:00
|
|
|
|
2016-09-16 16:54:14 +00:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2016-02-29 15:14:39 +00:00
|
|
|
|
2016-04-29 15:34:31 +00:00
|
|
|
PathSet queryAllValidPaths() override
|
|
|
|
{
|
|
|
|
PathSet paths;
|
|
|
|
|
|
|
|
for (auto & entry : readDirectory(binaryCacheDir)) {
|
|
|
|
if (entry.name.size() != 40 ||
|
|
|
|
!hasSuffix(entry.name, ".narinfo"))
|
|
|
|
continue;
|
2016-06-01 12:49:12 +00:00
|
|
|
paths.insert(storeDir + "/" + entry.name.substr(0, entry.name.size() - 8));
|
2016-04-29 15:34:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return paths;
|
|
|
|
}
|
|
|
|
|
2016-02-29 15:14:39 +00:00
|
|
|
};
|
|
|
|
|
2016-02-24 13:48:16 +00:00
|
|
|
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()))
|
2017-07-30 11:27:57 +00:00
|
|
|
throw SysError(format("renaming '%1%' to '%2%'") % tmp % path);
|
2016-02-24 13:48:16 +00:00
|
|
|
del.cancel();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool LocalBinaryCacheStore::fileExists(const std::string & path)
|
|
|
|
{
|
|
|
|
return pathExists(binaryCacheDir + "/" + path);
|
|
|
|
}
|
|
|
|
|
2017-03-14 14:26:01 +00:00
|
|
|
void LocalBinaryCacheStore::upsertFile(const std::string & path,
|
|
|
|
const std::string & data,
|
|
|
|
const std::string & mimeType)
|
2016-02-24 13:48:16 +00:00
|
|
|
{
|
|
|
|
atomicWrite(binaryCacheDir + "/" + path, data);
|
|
|
|
}
|
|
|
|
|
2016-04-29 14:26:16 +00:00
|
|
|
static RegisterStoreImplementation regStore([](
|
2016-06-01 12:49:12 +00:00
|
|
|
const std::string & uri, const Store::Params & params)
|
2016-04-29 14:26:16 +00:00
|
|
|
-> std::shared_ptr<Store>
|
|
|
|
{
|
2016-06-01 13:15:21 +00:00
|
|
|
if (getEnv("_NIX_FORCE_HTTP_BINARY_CACHE_STORE") == "1" ||
|
|
|
|
std::string(uri, 0, 7) != "file://")
|
|
|
|
return 0;
|
2016-05-04 18:15:41 +00:00
|
|
|
auto store = std::make_shared<LocalBinaryCacheStore>(params, std::string(uri, 7));
|
2016-04-29 14:47:20 +00:00
|
|
|
store->init();
|
|
|
|
return store;
|
2016-02-29 15:11:11 +00:00
|
|
|
});
|
|
|
|
|
2016-02-24 13:48:16 +00:00
|
|
|
}
|