forked from lix-project/lix
Merge pull request #9656 from edolstra/nix-profile-stable-names
Make profile element names stable
This commit is contained in:
commit
52f949bbf5
|
@ -3,4 +3,6 @@ synopsis: "`nix profile` now allows referring to elements by human-readable name
|
||||||
prs: 8678
|
prs: 8678
|
||||||
---
|
---
|
||||||
|
|
||||||
[`nix profile`](@docroot@/command-ref/new-cli/nix3-profile.md) now uses names to refer to installed packages when running [`list`](@docroot@/command-ref/new-cli/nix3-profile-list.md), [`remove`](@docroot@/command-ref/new-cli/nix3-profile-remove.md) or [`upgrade`](@docroot@/command-ref/new-cli/nix3-profile-upgrade.md) as opposed to indices. Indices are deprecated and will be removed in a future version.
|
[`nix profile`](@docroot@/command-ref/new-cli/nix3-profile.md) now uses names to refer to installed packages when running [`list`](@docroot@/command-ref/new-cli/nix3-profile-list.md), [`remove`](@docroot@/command-ref/new-cli/nix3-profile-remove.md) or [`upgrade`](@docroot@/command-ref/new-cli/nix3-profile-upgrade.md) as opposed to indices. Profile element names are generated when a package is installed and remain the same until the package is removed.
|
||||||
|
|
||||||
|
**Warning**: The `manifest.nix` file used to record the contents of profiles has changed. Nix will automatically upgrade profiles to the new version when you modify the profile. After that, the profile can no longer be used by older versions of Nix.
|
||||||
|
|
|
@ -7,14 +7,12 @@ R""(
|
||||||
```console
|
```console
|
||||||
# nix profile list
|
# nix profile list
|
||||||
Name: gdb
|
Name: gdb
|
||||||
Index: 0
|
|
||||||
Flake attribute: legacyPackages.x86_64-linux.gdb
|
Flake attribute: legacyPackages.x86_64-linux.gdb
|
||||||
Original flake URL: flake:nixpkgs
|
Original flake URL: flake:nixpkgs
|
||||||
Locked flake URL: github:NixOS/nixpkgs/7b38b03d76ab71bdc8dc325e3f6338d984cc35ca
|
Locked flake URL: github:NixOS/nixpkgs/7b38b03d76ab71bdc8dc325e3f6338d984cc35ca
|
||||||
Store paths: /nix/store/indzcw5wvlhx6vwk7k4iq29q15chvr3d-gdb-11.1
|
Store paths: /nix/store/indzcw5wvlhx6vwk7k4iq29q15chvr3d-gdb-11.1
|
||||||
|
|
||||||
Name: blender-bin
|
Name: blender-bin
|
||||||
Index: 1
|
|
||||||
Flake attribute: packages.x86_64-linux.default
|
Flake attribute: packages.x86_64-linux.default
|
||||||
Original flake URL: flake:blender-bin
|
Original flake URL: flake:blender-bin
|
||||||
Locked flake URL: github:edolstra/nix-warez/91f2ffee657bf834e4475865ae336e2379282d34?dir=blender
|
Locked flake URL: github:edolstra/nix-warez/91f2ffee657bf834e4475865ae336e2379282d34?dir=blender
|
||||||
|
|
|
@ -8,13 +8,6 @@ R""(
|
||||||
# nix profile remove hello
|
# nix profile remove hello
|
||||||
```
|
```
|
||||||
|
|
||||||
* Remove a package by index
|
|
||||||
*(deprecated, will be removed in a future version)*:
|
|
||||||
|
|
||||||
```console
|
|
||||||
# nix profile remove 3
|
|
||||||
```
|
|
||||||
|
|
||||||
* Remove all packages:
|
* Remove all packages:
|
||||||
|
|
||||||
```console
|
```console
|
||||||
|
|
|
@ -15,13 +15,6 @@ R""(
|
||||||
# nix profile upgrade hello
|
# nix profile upgrade hello
|
||||||
```
|
```
|
||||||
|
|
||||||
* Upgrade a specific package by index
|
|
||||||
*(deprecated, will be removed in a future version)*:
|
|
||||||
|
|
||||||
```console
|
|
||||||
# nix profile upgrade 0
|
|
||||||
```
|
|
||||||
|
|
||||||
# Description
|
# Description
|
||||||
|
|
||||||
This command upgrades a previously installed package in a Nix profile,
|
This command upgrades a previously installed package in a Nix profile,
|
||||||
|
|
|
@ -45,7 +45,6 @@ const int defaultPriority = 5;
|
||||||
struct ProfileElement
|
struct ProfileElement
|
||||||
{
|
{
|
||||||
StorePathSet storePaths;
|
StorePathSet storePaths;
|
||||||
std::string name;
|
|
||||||
std::optional<ProfileElementSource> source;
|
std::optional<ProfileElementSource> source;
|
||||||
bool active = true;
|
bool active = true;
|
||||||
int priority = defaultPriority;
|
int priority = defaultPriority;
|
||||||
|
@ -82,11 +81,6 @@ struct ProfileElement
|
||||||
return showVersions(versions);
|
return showVersions(versions);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool operator < (const ProfileElement & other) const
|
|
||||||
{
|
|
||||||
return std::tuple(identifier(), storePaths) < std::tuple(other.identifier(), other.storePaths);
|
|
||||||
}
|
|
||||||
|
|
||||||
void updateStorePaths(
|
void updateStorePaths(
|
||||||
ref<Store> evalStore,
|
ref<Store> evalStore,
|
||||||
ref<Store> store,
|
ref<Store> store,
|
||||||
|
@ -109,7 +103,9 @@ struct ProfileElement
|
||||||
|
|
||||||
struct ProfileManifest
|
struct ProfileManifest
|
||||||
{
|
{
|
||||||
std::vector<ProfileElement> elements;
|
using ProfileElementName = std::string;
|
||||||
|
|
||||||
|
std::map<ProfileElementName, ProfileElement> elements;
|
||||||
|
|
||||||
ProfileManifest() { }
|
ProfileManifest() { }
|
||||||
|
|
||||||
|
@ -119,8 +115,6 @@ struct ProfileManifest
|
||||||
|
|
||||||
if (pathExists(manifestPath)) {
|
if (pathExists(manifestPath)) {
|
||||||
auto json = nlohmann::json::parse(readFile(manifestPath));
|
auto json = nlohmann::json::parse(readFile(manifestPath));
|
||||||
/* Keep track of already found names to allow preventing duplicates. */
|
|
||||||
std::set<std::string> foundNames;
|
|
||||||
|
|
||||||
auto version = json.value("version", 0);
|
auto version = json.value("version", 0);
|
||||||
std::string sUrl;
|
std::string sUrl;
|
||||||
|
@ -131,6 +125,7 @@ struct ProfileManifest
|
||||||
sOriginalUrl = "originalUri";
|
sOriginalUrl = "originalUri";
|
||||||
break;
|
break;
|
||||||
case 2:
|
case 2:
|
||||||
|
case 3:
|
||||||
sUrl = "url";
|
sUrl = "url";
|
||||||
sOriginalUrl = "originalUrl";
|
sOriginalUrl = "originalUrl";
|
||||||
break;
|
break;
|
||||||
|
@ -138,7 +133,9 @@ struct ProfileManifest
|
||||||
throw Error("profile manifest '%s' has unsupported version %d", manifestPath, version);
|
throw Error("profile manifest '%s' has unsupported version %d", manifestPath, version);
|
||||||
}
|
}
|
||||||
|
|
||||||
for (auto & e : json["elements"]) {
|
auto elems = json["elements"];
|
||||||
|
for (auto & elem : elems.items()) {
|
||||||
|
auto & e = elem.value();
|
||||||
ProfileElement element;
|
ProfileElement element;
|
||||||
for (auto & p : e["storePaths"])
|
for (auto & p : e["storePaths"])
|
||||||
element.storePaths.insert(state.store->parseStorePath((std::string) p));
|
element.storePaths.insert(state.store->parseStorePath((std::string) p));
|
||||||
|
@ -155,25 +152,14 @@ struct ProfileManifest
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string nameCandidate = element.identifier();
|
std::string name =
|
||||||
if (e.contains("name")) {
|
elems.is_object()
|
||||||
nameCandidate = e["name"];
|
? elem.key()
|
||||||
}
|
: element.source
|
||||||
else if (element.source) {
|
? getNameFromURL(parseURL(element.source->to_string())).value_or(element.identifier())
|
||||||
auto url = parseURL(element.source->to_string());
|
: element.identifier();
|
||||||
auto name = getNameFromURL(url);
|
|
||||||
if (name)
|
|
||||||
nameCandidate = *name;
|
|
||||||
}
|
|
||||||
|
|
||||||
auto finalName = nameCandidate;
|
addElement(name, std::move(element));
|
||||||
for (int i = 1; foundNames.contains(finalName); ++i) {
|
|
||||||
finalName = nameCandidate + std::to_string(i);
|
|
||||||
}
|
|
||||||
element.name = finalName;
|
|
||||||
foundNames.insert(element.name);
|
|
||||||
|
|
||||||
elements.emplace_back(std::move(element));
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -187,16 +173,34 @@ struct ProfileManifest
|
||||||
for (auto & drvInfo : drvInfos) {
|
for (auto & drvInfo : drvInfos) {
|
||||||
ProfileElement element;
|
ProfileElement element;
|
||||||
element.storePaths = {drvInfo.queryOutPath()};
|
element.storePaths = {drvInfo.queryOutPath()};
|
||||||
element.name = element.identifier();
|
addElement(std::move(element));
|
||||||
elements.emplace_back(std::move(element));
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void addElement(std::string_view nameCandidate, ProfileElement element)
|
||||||
|
{
|
||||||
|
std::string finalName(nameCandidate);
|
||||||
|
for (int i = 1; elements.contains(finalName); ++i)
|
||||||
|
finalName = nameCandidate + "-" + std::to_string(i);
|
||||||
|
|
||||||
|
elements.insert_or_assign(finalName, std::move(element));
|
||||||
|
}
|
||||||
|
|
||||||
|
void addElement(ProfileElement element)
|
||||||
|
{
|
||||||
|
auto name =
|
||||||
|
element.source
|
||||||
|
? getNameFromURL(parseURL(element.source->to_string()))
|
||||||
|
: std::nullopt;
|
||||||
|
auto name2 = name ? *name : element.identifier();
|
||||||
|
addElement(name2, std::move(element));
|
||||||
|
}
|
||||||
|
|
||||||
nlohmann::json toJSON(Store & store) const
|
nlohmann::json toJSON(Store & store) const
|
||||||
{
|
{
|
||||||
auto array = nlohmann::json::array();
|
auto es = nlohmann::json::object();
|
||||||
for (auto & element : elements) {
|
for (auto & [name, element] : elements) {
|
||||||
auto paths = nlohmann::json::array();
|
auto paths = nlohmann::json::array();
|
||||||
for (auto & path : element.storePaths)
|
for (auto & path : element.storePaths)
|
||||||
paths.push_back(store.printStorePath(path));
|
paths.push_back(store.printStorePath(path));
|
||||||
|
@ -210,11 +214,11 @@ struct ProfileManifest
|
||||||
obj["attrPath"] = element.source->attrPath;
|
obj["attrPath"] = element.source->attrPath;
|
||||||
obj["outputs"] = element.source->outputs;
|
obj["outputs"] = element.source->outputs;
|
||||||
}
|
}
|
||||||
array.push_back(obj);
|
es[name] = obj;
|
||||||
}
|
}
|
||||||
nlohmann::json json;
|
nlohmann::json json;
|
||||||
json["version"] = 2;
|
json["version"] = 3;
|
||||||
json["elements"] = array;
|
json["elements"] = es;
|
||||||
return json;
|
return json;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -225,7 +229,7 @@ struct ProfileManifest
|
||||||
StorePathSet references;
|
StorePathSet references;
|
||||||
|
|
||||||
Packages pkgs;
|
Packages pkgs;
|
||||||
for (auto & element : elements) {
|
for (auto & [name, element] : elements) {
|
||||||
for (auto & path : element.storePaths) {
|
for (auto & path : element.storePaths) {
|
||||||
if (element.active)
|
if (element.active)
|
||||||
pkgs.emplace_back(store->printStorePath(path), true, element.priority);
|
pkgs.emplace_back(store->printStorePath(path), true, element.priority);
|
||||||
|
@ -267,33 +271,27 @@ struct ProfileManifest
|
||||||
|
|
||||||
static void printDiff(const ProfileManifest & prev, const ProfileManifest & cur, std::string_view indent)
|
static void printDiff(const ProfileManifest & prev, const ProfileManifest & cur, std::string_view indent)
|
||||||
{
|
{
|
||||||
auto prevElems = prev.elements;
|
auto i = prev.elements.begin();
|
||||||
std::sort(prevElems.begin(), prevElems.end());
|
auto j = cur.elements.begin();
|
||||||
|
|
||||||
auto curElems = cur.elements;
|
|
||||||
std::sort(curElems.begin(), curElems.end());
|
|
||||||
|
|
||||||
auto i = prevElems.begin();
|
|
||||||
auto j = curElems.begin();
|
|
||||||
|
|
||||||
bool changes = false;
|
bool changes = false;
|
||||||
|
|
||||||
while (i != prevElems.end() || j != curElems.end()) {
|
while (i != prev.elements.end() || j != cur.elements.end()) {
|
||||||
if (j != curElems.end() && (i == prevElems.end() || i->identifier() > j->identifier())) {
|
if (j != cur.elements.end() && (i == prev.elements.end() || i->first > j->first)) {
|
||||||
logger->cout("%s%s: ∅ -> %s", indent, j->identifier(), j->versions());
|
logger->cout("%s%s: ∅ -> %s", indent, j->second.identifier(), j->second.versions());
|
||||||
changes = true;
|
changes = true;
|
||||||
++j;
|
++j;
|
||||||
}
|
}
|
||||||
else if (i != prevElems.end() && (j == curElems.end() || i->identifier() < j->identifier())) {
|
else if (i != prev.elements.end() && (j == cur.elements.end() || i->first < j->first)) {
|
||||||
logger->cout("%s%s: %s -> ∅", indent, i->identifier(), i->versions());
|
logger->cout("%s%s: %s -> ∅", indent, i->second.identifier(), i->second.versions());
|
||||||
changes = true;
|
changes = true;
|
||||||
++i;
|
++i;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
auto v1 = i->versions();
|
auto v1 = i->second.versions();
|
||||||
auto v2 = j->versions();
|
auto v2 = j->second.versions();
|
||||||
if (v1 != v2) {
|
if (v1 != v2) {
|
||||||
logger->cout("%s%s: %s -> %s", indent, i->identifier(), v1, v2);
|
logger->cout("%s%s: %s -> %s", indent, i->second.identifier(), v1, v2);
|
||||||
changes = true;
|
changes = true;
|
||||||
}
|
}
|
||||||
++i;
|
++i;
|
||||||
|
@ -392,7 +390,7 @@ struct CmdProfileInstall : InstallablesCommand, MixDefaultProfile
|
||||||
|
|
||||||
element.updateStorePaths(getEvalStore(), store, res);
|
element.updateStorePaths(getEvalStore(), store, res);
|
||||||
|
|
||||||
manifest.elements.push_back(std::move(element));
|
manifest.addElement(std::move(element));
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
|
@ -402,7 +400,7 @@ struct CmdProfileInstall : InstallablesCommand, MixDefaultProfile
|
||||||
// See https://github.com/NixOS/nix/compare/3efa476c5439f8f6c1968a6ba20a31d1239c2f04..1fe5d172ece51a619e879c4b86f603d9495cc102
|
// See https://github.com/NixOS/nix/compare/3efa476c5439f8f6c1968a6ba20a31d1239c2f04..1fe5d172ece51a619e879c4b86f603d9495cc102
|
||||||
auto findRefByFilePath = [&]<typename Iterator>(Iterator begin, Iterator end) {
|
auto findRefByFilePath = [&]<typename Iterator>(Iterator begin, Iterator end) {
|
||||||
for (auto it = begin; it != end; it++) {
|
for (auto it = begin; it != end; it++) {
|
||||||
auto profileElement = *it;
|
auto & profileElement = it->second;
|
||||||
for (auto & storePath : profileElement.storePaths) {
|
for (auto & storePath : profileElement.storePaths) {
|
||||||
if (conflictError.fileA.starts_with(store->printStorePath(storePath))) {
|
if (conflictError.fileA.starts_with(store->printStorePath(storePath))) {
|
||||||
return std::pair(conflictError.fileA, profileElement.toInstallables(*store));
|
return std::pair(conflictError.fileA, profileElement.toInstallables(*store));
|
||||||
|
@ -470,43 +468,35 @@ public:
|
||||||
std::string pattern;
|
std::string pattern;
|
||||||
std::regex reg;
|
std::regex reg;
|
||||||
};
|
};
|
||||||
typedef std::variant<size_t, Path, RegexPattern> Matcher;
|
typedef std::variant<Path, RegexPattern> Matcher;
|
||||||
|
|
||||||
std::vector<Matcher> getMatchers(ref<Store> store)
|
std::vector<Matcher> getMatchers(ref<Store> store)
|
||||||
{
|
{
|
||||||
std::vector<Matcher> res;
|
std::vector<Matcher> res;
|
||||||
|
|
||||||
auto anyIndexMatchers = false;
|
|
||||||
|
|
||||||
for (auto & s : _matchers) {
|
for (auto & s : _matchers) {
|
||||||
if (auto n = string2Int<size_t>(s)) {
|
if (auto n = string2Int<size_t>(s))
|
||||||
res.push_back(*n);
|
throw Error("'nix profile' no longer supports indices ('%d')", *n);
|
||||||
anyIndexMatchers = true;
|
|
||||||
}
|
|
||||||
else if (store->isStorePath(s))
|
else if (store->isStorePath(s))
|
||||||
res.push_back(s);
|
res.push_back(s);
|
||||||
else
|
else
|
||||||
res.push_back(RegexPattern{s,std::regex(s, std::regex::extended | std::regex::icase)});
|
res.push_back(RegexPattern{s,std::regex(s, std::regex::extended | std::regex::icase)});
|
||||||
}
|
}
|
||||||
|
|
||||||
if (anyIndexMatchers) {
|
|
||||||
warn("Indices are deprecated and will be removed in a future version!\n"
|
|
||||||
" Refer to packages by their `Name` as printed by `nix profile list`.\n"
|
|
||||||
" See https://github.com/NixOS/nix/issues/9171 for more information.");
|
|
||||||
}
|
|
||||||
|
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool matches(const Store & store, const ProfileElement & element, size_t pos, const std::vector<Matcher> & matchers)
|
bool matches(
|
||||||
|
const Store & store,
|
||||||
|
const std::string & name,
|
||||||
|
const ProfileElement & element,
|
||||||
|
const std::vector<Matcher> & matchers)
|
||||||
{
|
{
|
||||||
for (auto & matcher : matchers) {
|
for (auto & matcher : matchers) {
|
||||||
if (auto n = std::get_if<size_t>(&matcher)) {
|
if (auto path = std::get_if<Path>(&matcher)) {
|
||||||
if (*n == pos) return true;
|
|
||||||
} else if (auto path = std::get_if<Path>(&matcher)) {
|
|
||||||
if (element.storePaths.count(store.parseStorePath(*path))) return true;
|
if (element.storePaths.count(store.parseStorePath(*path))) return true;
|
||||||
} else if (auto regex = std::get_if<RegexPattern>(&matcher)) {
|
} else if (auto regex = std::get_if<RegexPattern>(&matcher)) {
|
||||||
if (std::regex_match(element.name, regex->reg))
|
if (std::regex_match(name, regex->reg))
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -537,10 +527,9 @@ struct CmdProfileRemove : virtual EvalCommand, MixDefaultProfile, MixProfileElem
|
||||||
|
|
||||||
ProfileManifest newManifest;
|
ProfileManifest newManifest;
|
||||||
|
|
||||||
for (size_t i = 0; i < oldManifest.elements.size(); ++i) {
|
for (auto & [name, element] : oldManifest.elements) {
|
||||||
auto & element(oldManifest.elements[i]);
|
if (!matches(*store, name, element, matchers)) {
|
||||||
if (!matches(*store, element, i, matchers)) {
|
newManifest.elements.insert_or_assign(name, std::move(element));
|
||||||
newManifest.elements.push_back(std::move(element));
|
|
||||||
} else {
|
} else {
|
||||||
notice("removing '%s'", element.identifier());
|
notice("removing '%s'", element.identifier());
|
||||||
}
|
}
|
||||||
|
@ -553,9 +542,7 @@ struct CmdProfileRemove : virtual EvalCommand, MixDefaultProfile, MixProfileElem
|
||||||
|
|
||||||
if (removedCount == 0) {
|
if (removedCount == 0) {
|
||||||
for (auto matcher: matchers) {
|
for (auto matcher: matchers) {
|
||||||
if (const size_t * index = std::get_if<size_t>(&matcher)){
|
if (const Path * path = std::get_if<Path>(&matcher)) {
|
||||||
warn("'%d' is not a valid index", *index);
|
|
||||||
} else if (const Path * path = std::get_if<Path>(&matcher)){
|
|
||||||
warn("'%s' does not match any paths", *path);
|
warn("'%s' does not match any paths", *path);
|
||||||
} else if (const RegexPattern * regex = std::get_if<RegexPattern>(&matcher)) {
|
} else if (const RegexPattern * regex = std::get_if<RegexPattern>(&matcher)) {
|
||||||
warn("'%s' does not match any packages", regex->pattern);
|
warn("'%s' does not match any packages", regex->pattern);
|
||||||
|
@ -588,14 +575,13 @@ struct CmdProfileUpgrade : virtual SourceExprCommand, MixDefaultProfile, MixProf
|
||||||
auto matchers = getMatchers(store);
|
auto matchers = getMatchers(store);
|
||||||
|
|
||||||
Installables installables;
|
Installables installables;
|
||||||
std::vector<size_t> indices;
|
std::vector<ProfileElement *> elems;
|
||||||
|
|
||||||
auto matchedCount = 0;
|
auto matchedCount = 0;
|
||||||
auto upgradedCount = 0;
|
auto upgradedCount = 0;
|
||||||
|
|
||||||
for (size_t i = 0; i < manifest.elements.size(); ++i) {
|
for (auto & [name, element] : manifest.elements) {
|
||||||
auto & element(manifest.elements[i]);
|
if (!matches(*store, name, element, matchers)) {
|
||||||
if (!matches(*store, element, i, matchers)) {
|
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -651,15 +637,13 @@ struct CmdProfileUpgrade : virtual SourceExprCommand, MixDefaultProfile, MixProf
|
||||||
};
|
};
|
||||||
|
|
||||||
installables.push_back(installable);
|
installables.push_back(installable);
|
||||||
indices.push_back(i);
|
elems.push_back(&element);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (upgradedCount == 0) {
|
if (upgradedCount == 0) {
|
||||||
if (matchedCount == 0) {
|
if (matchedCount == 0) {
|
||||||
for (auto & matcher : matchers) {
|
for (auto & matcher : matchers) {
|
||||||
if (const size_t * index = std::get_if<size_t>(&matcher)){
|
if (const Path * path = std::get_if<Path>(&matcher)) {
|
||||||
warn("'%d' is not a valid index", *index);
|
|
||||||
} else if (const Path * path = std::get_if<Path>(&matcher)){
|
|
||||||
warn("'%s' does not match any paths", *path);
|
warn("'%s' does not match any paths", *path);
|
||||||
} else if (const RegexPattern * regex = std::get_if<RegexPattern>(&matcher)) {
|
} else if (const RegexPattern * regex = std::get_if<RegexPattern>(&matcher)) {
|
||||||
warn("'%s' does not match any packages", regex->pattern);
|
warn("'%s' does not match any packages", regex->pattern);
|
||||||
|
@ -677,7 +661,7 @@ struct CmdProfileUpgrade : virtual SourceExprCommand, MixDefaultProfile, MixProf
|
||||||
|
|
||||||
for (size_t i = 0; i < installables.size(); ++i) {
|
for (size_t i = 0; i < installables.size(); ++i) {
|
||||||
auto & installable = installables.at(i);
|
auto & installable = installables.at(i);
|
||||||
auto & element = manifest.elements[indices.at(i)];
|
auto & element = *elems.at(i);
|
||||||
element.updateStorePaths(
|
element.updateStorePaths(
|
||||||
getEvalStore(),
|
getEvalStore(),
|
||||||
store,
|
store,
|
||||||
|
@ -709,13 +693,12 @@ struct CmdProfileList : virtual EvalCommand, virtual StoreCommand, MixDefaultPro
|
||||||
if (json) {
|
if (json) {
|
||||||
std::cout << manifest.toJSON(*store).dump() << "\n";
|
std::cout << manifest.toJSON(*store).dump() << "\n";
|
||||||
} else {
|
} else {
|
||||||
for (size_t i = 0; i < manifest.elements.size(); ++i) {
|
for (const auto & [i, e] : enumerate(manifest.elements)) {
|
||||||
auto & element(manifest.elements[i]);
|
auto & [name, element] = e;
|
||||||
if (i) logger->cout("");
|
if (i) logger->cout("");
|
||||||
logger->cout("Name: " ANSI_BOLD "%s" ANSI_NORMAL "%s",
|
logger->cout("Name: " ANSI_BOLD "%s" ANSI_NORMAL "%s",
|
||||||
element.name,
|
name,
|
||||||
element.active ? "" : " " ANSI_RED "(inactive)" ANSI_NORMAL);
|
element.active ? "" : " " ANSI_RED "(inactive)" ANSI_NORMAL);
|
||||||
logger->cout("Index: %s", i);
|
|
||||||
if (element.source) {
|
if (element.source) {
|
||||||
logger->cout("Flake attribute: %s%s", element.source->attrPath, element.source->outputs.to_string());
|
logger->cout("Flake attribute: %s%s", element.source->attrPath, element.source->outputs.to_string());
|
||||||
logger->cout("Original flake URL: %s", element.source->originalRef.to_string());
|
logger->cout("Original flake URL: %s", element.source->originalRef.to_string());
|
||||||
|
|
|
@ -49,7 +49,7 @@ cp ./config.nix $flake1Dir/
|
||||||
nix-env -f ./user-envs.nix -i foo-1.0
|
nix-env -f ./user-envs.nix -i foo-1.0
|
||||||
nix profile list | grep -A2 'Name:.*foo' | grep 'Store paths:.*foo-1.0'
|
nix profile list | grep -A2 'Name:.*foo' | grep 'Store paths:.*foo-1.0'
|
||||||
nix profile install $flake1Dir -L
|
nix profile install $flake1Dir -L
|
||||||
nix profile list | grep -A4 'Index:.*1' | grep 'Locked flake URL:.*narHash'
|
nix profile list | grep -A4 'Name:.*flake1' | grep 'Locked flake URL:.*narHash'
|
||||||
[[ $($TEST_HOME/.nix-profile/bin/hello) = "Hello World" ]]
|
[[ $($TEST_HOME/.nix-profile/bin/hello) = "Hello World" ]]
|
||||||
[ -e $TEST_HOME/.nix-profile/share/man ]
|
[ -e $TEST_HOME/.nix-profile/share/man ]
|
||||||
(! [ -e $TEST_HOME/.nix-profile/include ])
|
(! [ -e $TEST_HOME/.nix-profile/include ])
|
||||||
|
@ -58,9 +58,8 @@ nix profile history | grep "packages.$system.default: ∅ -> 1.0"
|
||||||
nix profile diff-closures | grep 'env-manifest.nix: ε → ∅'
|
nix profile diff-closures | grep 'env-manifest.nix: ε → ∅'
|
||||||
|
|
||||||
# Test XDG Base Directories support
|
# Test XDG Base Directories support
|
||||||
|
|
||||||
export NIX_CONFIG="use-xdg-base-directories = true"
|
export NIX_CONFIG="use-xdg-base-directories = true"
|
||||||
nix profile remove 1
|
nix profile remove flake1 2>&1 | grep 'removed 1 packages'
|
||||||
nix profile install $flake1Dir
|
nix profile install $flake1Dir
|
||||||
[[ $($TEST_HOME/.local/state/nix/profile/bin/hello) = "Hello World" ]]
|
[[ $($TEST_HOME/.local/state/nix/profile/bin/hello) = "Hello World" ]]
|
||||||
unset NIX_CONFIG
|
unset NIX_CONFIG
|
||||||
|
@ -68,7 +67,7 @@ unset NIX_CONFIG
|
||||||
# Test upgrading a package.
|
# Test upgrading a package.
|
||||||
printf NixOS > $flake1Dir/who
|
printf NixOS > $flake1Dir/who
|
||||||
printf 2.0 > $flake1Dir/version
|
printf 2.0 > $flake1Dir/version
|
||||||
nix profile upgrade 1
|
nix profile upgrade flake1
|
||||||
[[ $($TEST_HOME/.nix-profile/bin/hello) = "Hello NixOS" ]]
|
[[ $($TEST_HOME/.nix-profile/bin/hello) = "Hello NixOS" ]]
|
||||||
nix profile history | grep "packages.$system.default: 1.0, 1.0-man -> 2.0, 2.0-man"
|
nix profile history | grep "packages.$system.default: 1.0, 1.0-man -> 2.0, 2.0-man"
|
||||||
|
|
||||||
|
@ -81,7 +80,7 @@ nix profile rollback
|
||||||
|
|
||||||
# Test uninstall.
|
# Test uninstall.
|
||||||
[ -e $TEST_HOME/.nix-profile/bin/foo ]
|
[ -e $TEST_HOME/.nix-profile/bin/foo ]
|
||||||
nix profile remove foo
|
nix profile remove foo 2>&1 | grep 'removed 1 packages'
|
||||||
(! [ -e $TEST_HOME/.nix-profile/bin/foo ])
|
(! [ -e $TEST_HOME/.nix-profile/bin/foo ])
|
||||||
nix profile history | grep 'foo: 1.0 -> ∅'
|
nix profile history | grep 'foo: 1.0 -> ∅'
|
||||||
nix profile diff-closures | grep 'Version 3 -> 4'
|
nix profile diff-closures | grep 'Version 3 -> 4'
|
||||||
|
@ -89,7 +88,7 @@ nix profile diff-closures | grep 'Version 3 -> 4'
|
||||||
# Test installing a non-flake package.
|
# Test installing a non-flake package.
|
||||||
nix profile install --file ./simple.nix ''
|
nix profile install --file ./simple.nix ''
|
||||||
[[ $(cat $TEST_HOME/.nix-profile/hello) = "Hello World!" ]]
|
[[ $(cat $TEST_HOME/.nix-profile/hello) = "Hello World!" ]]
|
||||||
nix profile remove 1
|
nix profile remove simple 2>&1 | grep 'removed 1 packages'
|
||||||
nix profile install $(nix-build --no-out-link ./simple.nix)
|
nix profile install $(nix-build --no-out-link ./simple.nix)
|
||||||
[[ $(cat $TEST_HOME/.nix-profile/hello) = "Hello World!" ]]
|
[[ $(cat $TEST_HOME/.nix-profile/hello) = "Hello World!" ]]
|
||||||
|
|
||||||
|
@ -97,8 +96,9 @@ nix profile install $(nix-build --no-out-link ./simple.nix)
|
||||||
mkdir $TEST_ROOT/simple-too
|
mkdir $TEST_ROOT/simple-too
|
||||||
cp ./simple.nix ./config.nix simple.builder.sh $TEST_ROOT/simple-too
|
cp ./simple.nix ./config.nix simple.builder.sh $TEST_ROOT/simple-too
|
||||||
nix profile install --file $TEST_ROOT/simple-too/simple.nix ''
|
nix profile install --file $TEST_ROOT/simple-too/simple.nix ''
|
||||||
nix profile list | grep -A4 'Name:.*simple' | grep 'Name:.*simple1'
|
nix profile list | grep -A4 'Name:.*simple' | grep 'Name:.*simple-1'
|
||||||
nix profile remove simple1
|
nix profile remove simple 2>&1 | grep 'removed 1 packages'
|
||||||
|
nix profile remove simple-1 2>&1 | grep 'removed 1 packages'
|
||||||
|
|
||||||
# Test wipe-history.
|
# Test wipe-history.
|
||||||
nix profile wipe-history
|
nix profile wipe-history
|
||||||
|
@ -107,11 +107,11 @@ nix profile wipe-history
|
||||||
# Test upgrade to CA package.
|
# Test upgrade to CA package.
|
||||||
printf true > $flake1Dir/ca.nix
|
printf true > $flake1Dir/ca.nix
|
||||||
printf 3.0 > $flake1Dir/version
|
printf 3.0 > $flake1Dir/version
|
||||||
nix profile upgrade 0
|
nix profile upgrade flake1
|
||||||
nix profile history | grep "packages.$system.default: 1.0, 1.0-man -> 3.0, 3.0-man"
|
nix profile history | grep "packages.$system.default: 1.0, 1.0-man -> 3.0, 3.0-man"
|
||||||
|
|
||||||
# Test new install of CA package.
|
# Test new install of CA package.
|
||||||
nix profile remove flake1
|
nix profile remove flake1 2>&1 | grep 'removed 1 packages'
|
||||||
printf 4.0 > $flake1Dir/version
|
printf 4.0 > $flake1Dir/version
|
||||||
printf Utrecht > $flake1Dir/who
|
printf Utrecht > $flake1Dir/who
|
||||||
nix profile install $flake1Dir
|
nix profile install $flake1Dir
|
||||||
|
@ -132,14 +132,14 @@ nix profile upgrade flake1
|
||||||
[ -e $TEST_HOME/.nix-profile/share/man ]
|
[ -e $TEST_HOME/.nix-profile/share/man ]
|
||||||
[ -e $TEST_HOME/.nix-profile/include ]
|
[ -e $TEST_HOME/.nix-profile/include ]
|
||||||
|
|
||||||
nix profile remove flake1
|
nix profile remove flake1 2>&1 | grep 'removed 1 packages'
|
||||||
nix profile install "$flake1Dir^man"
|
nix profile install "$flake1Dir^man"
|
||||||
(! [ -e $TEST_HOME/.nix-profile/bin/hello ])
|
(! [ -e $TEST_HOME/.nix-profile/bin/hello ])
|
||||||
[ -e $TEST_HOME/.nix-profile/share/man ]
|
[ -e $TEST_HOME/.nix-profile/share/man ]
|
||||||
(! [ -e $TEST_HOME/.nix-profile/include ])
|
(! [ -e $TEST_HOME/.nix-profile/include ])
|
||||||
|
|
||||||
# test priority
|
# test priority
|
||||||
nix profile remove flake1
|
nix profile remove flake1 2>&1 | grep 'removed 1 packages'
|
||||||
|
|
||||||
# Make another flake.
|
# Make another flake.
|
||||||
flake2Dir=$TEST_ROOT/flake2
|
flake2Dir=$TEST_ROOT/flake2
|
||||||
|
@ -193,3 +193,12 @@ nix profile install $flake2Dir --priority 0
|
||||||
clearProfiles
|
clearProfiles
|
||||||
nix profile install $(nix build $flake1Dir --no-link --print-out-paths)
|
nix profile install $(nix build $flake1Dir --no-link --print-out-paths)
|
||||||
expect 1 nix profile install --impure --expr "(builtins.getFlake ''$flake2Dir'').packages.$system.default"
|
expect 1 nix profile install --impure --expr "(builtins.getFlake ''$flake2Dir'').packages.$system.default"
|
||||||
|
|
||||||
|
# Test upgrading from profile version 2.
|
||||||
|
clearProfiles
|
||||||
|
mkdir -p $TEST_ROOT/import-profile
|
||||||
|
outPath=$(nix build --no-link --print-out-paths $flake1Dir/flake.nix^out)
|
||||||
|
printf '{ "version": 2, "elements": [ { "active": true, "attrPath": "legacyPackages.x86_64-linux.hello", "originalUrl": "flake:nixpkgs", "outputs": null, "priority": 5, "storePaths": [ "%s" ], "url": "github:NixOS/nixpkgs/aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa" } ] }' "$outPath" > $TEST_ROOT/import-profile/manifest.json
|
||||||
|
nix build --profile $TEST_HOME/.nix-profile $(nix store add-path $TEST_ROOT/import-profile)
|
||||||
|
nix profile list | grep -A4 'Name:.*hello' | grep "Store paths:.*$outPath"
|
||||||
|
nix profile remove hello 2>&1 | grep 'removed 1 packages, kept 0 packages'
|
||||||
|
|
Loading…
Reference in a new issue