2019-02-12 19:23:11 +02:00
|
|
|
#include "types.hh"
|
|
|
|
#include "flakeref.hh"
|
|
|
|
|
|
|
|
#include <variant>
|
2019-06-04 20:10:35 +03:00
|
|
|
#include <nlohmann/json.hpp>
|
2019-02-12 19:23:11 +02:00
|
|
|
|
|
|
|
namespace nix {
|
|
|
|
|
2019-05-29 16:31:07 +03:00
|
|
|
struct Value;
|
|
|
|
class EvalState;
|
|
|
|
|
|
|
|
namespace flake {
|
|
|
|
|
2019-04-30 12:03:31 +03:00
|
|
|
static const size_t FLAG_REGISTRY = 0;
|
|
|
|
static const size_t USER_REGISTRY = 1;
|
|
|
|
static const size_t GLOBAL_REGISTRY = 2;
|
|
|
|
|
2019-02-12 19:23:11 +02:00
|
|
|
struct FlakeRegistry
|
|
|
|
{
|
2019-04-08 20:03:00 +03:00
|
|
|
std::map<FlakeRef, FlakeRef> entries;
|
2019-02-12 19:23:11 +02:00
|
|
|
};
|
|
|
|
|
2019-06-04 20:10:35 +03:00
|
|
|
typedef std::vector<std::shared_ptr<FlakeRegistry>> Registries;
|
2019-05-21 15:55:43 +03:00
|
|
|
|
2019-06-04 20:10:35 +03:00
|
|
|
std::shared_ptr<FlakeRegistry> readRegistry(const Path &);
|
2019-03-29 17:18:25 +02:00
|
|
|
|
2019-06-04 20:10:35 +03:00
|
|
|
void writeRegistry(const FlakeRegistry &, const Path &);
|
2019-03-21 10:30:16 +02:00
|
|
|
|
2019-03-10 08:05:05 +02:00
|
|
|
Path getUserRegistryPath();
|
|
|
|
|
2019-05-22 14:46:07 +03:00
|
|
|
enum HandleLockFile : unsigned int
|
2019-05-14 12:34:45 +03:00
|
|
|
{ AllPure // Everything is handled 100% purely
|
|
|
|
, TopRefUsesRegistries // The top FlakeRef uses the registries, apart from that, everything happens 100% purely
|
|
|
|
, UpdateLockFile // Update the existing lockfile and write it to file
|
|
|
|
, UseUpdatedLockFile // `UpdateLockFile` without writing to file
|
|
|
|
, RecreateLockFile // Recreate the lockfile from scratch and write it to file
|
|
|
|
, UseNewLockFile // `RecreateLockFile` without writing to file
|
|
|
|
};
|
2019-04-16 16:02:02 +03:00
|
|
|
|
2019-06-04 20:10:35 +03:00
|
|
|
struct NonFlakeDep
|
|
|
|
{
|
|
|
|
FlakeRef ref;
|
|
|
|
Hash narHash;
|
2019-03-10 08:05:05 +02:00
|
|
|
|
2019-06-04 20:10:35 +03:00
|
|
|
NonFlakeDep(const FlakeRef & flakeRef, const Hash & narHash)
|
|
|
|
: ref(flakeRef), narHash(narHash) {};
|
|
|
|
|
|
|
|
NonFlakeDep(const nlohmann::json & json);
|
|
|
|
|
|
|
|
bool operator ==(const NonFlakeDep & other) const
|
|
|
|
{
|
|
|
|
return ref == other.ref && narHash == other.narHash;
|
|
|
|
}
|
|
|
|
|
|
|
|
nlohmann::json toJson() const;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct FlakeDep;
|
|
|
|
|
|
|
|
struct FlakeInputs
|
|
|
|
{
|
|
|
|
std::map<FlakeRef, FlakeDep> flakeDeps;
|
|
|
|
std::map<FlakeAlias, NonFlakeDep> nonFlakeDeps;
|
|
|
|
|
|
|
|
FlakeInputs() {};
|
|
|
|
FlakeInputs(const nlohmann::json & json);
|
|
|
|
|
|
|
|
nlohmann::json toJson() const;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct FlakeDep : FlakeInputs
|
|
|
|
{
|
|
|
|
FlakeId id;
|
|
|
|
FlakeRef ref;
|
|
|
|
Hash narHash;
|
|
|
|
|
|
|
|
FlakeDep(const FlakeId & id, const FlakeRef & flakeRef, const Hash & narHash)
|
|
|
|
: id(id), ref(flakeRef), narHash(narHash) {};
|
|
|
|
|
|
|
|
FlakeDep(const nlohmann::json & json);
|
|
|
|
|
|
|
|
bool operator ==(const FlakeDep & other) const
|
|
|
|
{
|
|
|
|
return
|
|
|
|
id == other.id
|
|
|
|
&& ref == other.ref
|
|
|
|
&& narHash == other.narHash
|
|
|
|
&& flakeDeps == other.flakeDeps
|
|
|
|
&& nonFlakeDeps == other.nonFlakeDeps;
|
|
|
|
}
|
|
|
|
|
|
|
|
nlohmann::json toJson() const;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct LockFile : FlakeInputs
|
|
|
|
{
|
|
|
|
bool operator ==(const LockFile & other) const
|
|
|
|
{
|
|
|
|
return
|
|
|
|
flakeDeps == other.flakeDeps
|
|
|
|
&& nonFlakeDeps == other.nonFlakeDeps;
|
|
|
|
}
|
|
|
|
|
|
|
|
LockFile() {}
|
|
|
|
LockFile(const nlohmann::json & json) : FlakeInputs(json) {}
|
|
|
|
LockFile(FlakeDep && dep)
|
|
|
|
{
|
|
|
|
flakeDeps = std::move(dep.flakeDeps);
|
|
|
|
nonFlakeDeps = std::move(dep.nonFlakeDeps);
|
|
|
|
}
|
|
|
|
|
|
|
|
nlohmann::json toJson() const;
|
|
|
|
};
|
2019-02-21 07:53:01 +02:00
|
|
|
|
2019-05-01 12:38:48 +03:00
|
|
|
struct SourceInfo
|
2019-04-16 16:40:58 +03:00
|
|
|
{
|
2019-05-28 21:34:02 +03:00
|
|
|
// Immutable flakeref that this source tree was obtained from.
|
2019-05-01 12:38:48 +03:00
|
|
|
FlakeRef resolvedRef;
|
2019-05-28 21:34:02 +03:00
|
|
|
|
2019-04-16 16:40:58 +03:00
|
|
|
Path storePath;
|
2019-05-28 21:34:02 +03:00
|
|
|
|
|
|
|
// Number of ancestors of the most recent commit.
|
2019-04-16 16:40:58 +03:00
|
|
|
std::optional<uint64_t> revCount;
|
2019-05-28 21:34:02 +03:00
|
|
|
|
|
|
|
// NAR hash of the store path.
|
|
|
|
Hash narHash;
|
|
|
|
|
|
|
|
// A stable timestamp of this source tree. For Git and GitHub
|
|
|
|
// flakes, the commit date (not author date!) of the most recent
|
|
|
|
// commit.
|
|
|
|
std::optional<time_t> lastModified;
|
|
|
|
|
2019-05-01 12:38:48 +03:00
|
|
|
SourceInfo(const FlakeRef & resolvRef) : resolvedRef(resolvRef) {};
|
2019-04-16 16:40:58 +03:00
|
|
|
};
|
|
|
|
|
2019-02-21 07:53:01 +02:00
|
|
|
struct Flake
|
|
|
|
{
|
|
|
|
FlakeId id;
|
2019-05-01 12:38:48 +03:00
|
|
|
FlakeRef originalRef;
|
2019-02-21 07:53:01 +02:00
|
|
|
std::string description;
|
2019-05-28 13:58:28 +03:00
|
|
|
SourceInfo sourceInfo;
|
2019-05-30 00:09:23 +03:00
|
|
|
std::vector<FlakeRef> inputs;
|
|
|
|
std::map<FlakeAlias, FlakeRef> nonFlakeInputs;
|
|
|
|
Value * vOutputs; // FIXME: gc
|
2019-05-22 15:31:40 +03:00
|
|
|
unsigned int epoch;
|
|
|
|
|
2019-05-28 14:12:43 +03:00
|
|
|
Flake(const FlakeRef & origRef, const SourceInfo & sourceInfo)
|
|
|
|
: originalRef(origRef), sourceInfo(sourceInfo) {};
|
2019-03-21 10:30:16 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
struct NonFlake
|
|
|
|
{
|
2019-03-21 10:30:16 +02:00
|
|
|
FlakeAlias alias;
|
2019-05-01 12:38:48 +03:00
|
|
|
FlakeRef originalRef;
|
2019-05-28 13:58:28 +03:00
|
|
|
SourceInfo sourceInfo;
|
2019-05-28 14:12:43 +03:00
|
|
|
NonFlake(const FlakeRef & origRef, const SourceInfo & sourceInfo)
|
|
|
|
: originalRef(origRef), sourceInfo(sourceInfo) {};
|
2019-02-21 07:53:01 +02:00
|
|
|
};
|
|
|
|
|
2019-03-29 17:18:25 +02:00
|
|
|
Flake getFlake(EvalState &, const FlakeRef &, bool impureIsAllowed);
|
2019-02-21 07:53:01 +02:00
|
|
|
|
2019-04-19 15:23:35 +03:00
|
|
|
struct ResolvedFlake
|
2019-03-21 10:30:16 +02:00
|
|
|
{
|
2019-03-29 17:18:25 +02:00
|
|
|
Flake flake;
|
2019-06-04 20:10:35 +03:00
|
|
|
LockFile lockFile;
|
|
|
|
ResolvedFlake(Flake && flake, LockFile && lockFile)
|
|
|
|
: flake(flake), lockFile(lockFile) {}
|
2019-03-21 10:30:16 +02:00
|
|
|
};
|
|
|
|
|
2019-05-14 12:34:45 +03:00
|
|
|
ResolvedFlake resolveFlake(EvalState &, const FlakeRef &, HandleLockFile);
|
2019-04-16 15:27:54 +03:00
|
|
|
|
2019-06-04 20:10:35 +03:00
|
|
|
void callFlake(EvalState & state,
|
|
|
|
const Flake & flake,
|
|
|
|
const FlakeInputs & inputs,
|
|
|
|
Value & v);
|
|
|
|
|
|
|
|
void callFlake(EvalState & state,
|
|
|
|
const ResolvedFlake & resFlake,
|
|
|
|
Value & v);
|
2019-05-24 00:42:13 +03:00
|
|
|
|
2019-05-16 23:48:16 +03:00
|
|
|
void updateLockFile(EvalState &, const FlakeRef & flakeRef, bool recreateLockFile);
|
|
|
|
|
|
|
|
void gitCloneFlake(FlakeRef flakeRef, EvalState &, Registries, const Path & destDir);
|
2019-03-21 10:30:16 +02:00
|
|
|
|
2019-02-12 19:23:11 +02:00
|
|
|
}
|
2019-05-29 16:31:07 +03:00
|
|
|
|
|
|
|
}
|