Merge pull request #7788 from bobvanderlinden/pr-improve-nix-profile-install-error
Improve error on conflict for nix profile install
This commit is contained in:
commit
306e5c5ce5
4 changed files with 118 additions and 8 deletions
|
@ -92,13 +92,11 @@ static void createLinks(State & state, const Path & srcDir, const Path & dstDir,
|
||||||
if (S_ISLNK(dstSt.st_mode)) {
|
if (S_ISLNK(dstSt.st_mode)) {
|
||||||
auto prevPriority = state.priorities[dstFile];
|
auto prevPriority = state.priorities[dstFile];
|
||||||
if (prevPriority == priority)
|
if (prevPriority == priority)
|
||||||
throw Error(
|
throw BuildEnvFileConflictError(
|
||||||
"files '%1%' and '%2%' have the same priority %3%; "
|
readLink(dstFile),
|
||||||
"use 'nix-env --set-flag priority NUMBER INSTALLED_PKGNAME' "
|
srcFile,
|
||||||
"or type 'nix profile install --help' if using 'nix profile' to find out how "
|
priority
|
||||||
"to change the priority of one of the conflicting packages"
|
);
|
||||||
" (0 being the highest priority)",
|
|
||||||
srcFile, readLink(dstFile), priority);
|
|
||||||
if (prevPriority < priority)
|
if (prevPriority < priority)
|
||||||
continue;
|
continue;
|
||||||
if (unlink(dstFile.c_str()) == -1)
|
if (unlink(dstFile.c_str()) == -1)
|
||||||
|
|
|
@ -12,6 +12,32 @@ struct Package {
|
||||||
Package(const Path & path, bool active, int priority) : path{path}, active{active}, priority{priority} {}
|
Package(const Path & path, bool active, int priority) : path{path}, active{active}, priority{priority} {}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class BuildEnvFileConflictError : public Error
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
const Path fileA;
|
||||||
|
const Path fileB;
|
||||||
|
int priority;
|
||||||
|
|
||||||
|
BuildEnvFileConflictError(
|
||||||
|
const Path fileA,
|
||||||
|
const Path fileB,
|
||||||
|
int priority
|
||||||
|
)
|
||||||
|
: Error(
|
||||||
|
"Unable to build profile. There is a conflict for the following files:\n"
|
||||||
|
"\n"
|
||||||
|
" %1%\n"
|
||||||
|
" %2%",
|
||||||
|
fileA,
|
||||||
|
fileB
|
||||||
|
)
|
||||||
|
, fileA(fileA)
|
||||||
|
, fileB(fileB)
|
||||||
|
, priority(priority)
|
||||||
|
{}
|
||||||
|
};
|
||||||
|
|
||||||
typedef std::vector<Package> Packages;
|
typedef std::vector<Package> Packages;
|
||||||
|
|
||||||
void buildProfile(const Path & out, Packages && pkgs);
|
void buildProfile(const Path & out, Packages && pkgs);
|
||||||
|
|
|
@ -330,7 +330,63 @@ struct CmdProfileInstall : InstallablesCommand, MixDefaultProfile
|
||||||
manifest.elements.push_back(std::move(element));
|
manifest.elements.push_back(std::move(element));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
updateProfile(manifest.build(store));
|
updateProfile(manifest.build(store));
|
||||||
|
} catch (BuildEnvFileConflictError & conflictError) {
|
||||||
|
// FIXME use C++20 std::ranges once macOS has it
|
||||||
|
// See https://github.com/NixOS/nix/compare/3efa476c5439f8f6c1968a6ba20a31d1239c2f04..1fe5d172ece51a619e879c4b86f603d9495cc102
|
||||||
|
auto findRefByFilePath = [&]<typename Iterator>(Iterator begin, Iterator end) {
|
||||||
|
for (auto it = begin; it != end; it++) {
|
||||||
|
auto profileElement = *it;
|
||||||
|
for (auto & storePath : profileElement.storePaths) {
|
||||||
|
if (conflictError.fileA.starts_with(store->printStorePath(storePath))) {
|
||||||
|
return std::pair(conflictError.fileA, profileElement.source->originalRef);
|
||||||
|
}
|
||||||
|
if (conflictError.fileB.starts_with(store->printStorePath(storePath))) {
|
||||||
|
return std::pair(conflictError.fileB, profileElement.source->originalRef);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
throw conflictError;
|
||||||
|
};
|
||||||
|
// There are 2 conflicting files. We need to find out which one is from the already installed package and
|
||||||
|
// which one is the package that is the new package that is being installed.
|
||||||
|
// The first matching package is the one that was already installed (original).
|
||||||
|
auto [originalConflictingFilePath, originalConflictingRef] = findRefByFilePath(manifest.elements.begin(), manifest.elements.end());
|
||||||
|
// The last matching package is the one that was going to be installed (new).
|
||||||
|
auto [newConflictingFilePath, newConflictingRef] = findRefByFilePath(manifest.elements.rbegin(), manifest.elements.rend());
|
||||||
|
|
||||||
|
throw Error(
|
||||||
|
"An existing package already provides the following file:\n"
|
||||||
|
"\n"
|
||||||
|
" %1%\n"
|
||||||
|
"\n"
|
||||||
|
"This is the conflicting file from the new package:\n"
|
||||||
|
"\n"
|
||||||
|
" %2%\n"
|
||||||
|
"\n"
|
||||||
|
"To remove the existing package:\n"
|
||||||
|
"\n"
|
||||||
|
" nix profile remove %3%\n"
|
||||||
|
"\n"
|
||||||
|
"The new package can also be installed next to the existing one by assigning a different priority.\n"
|
||||||
|
"The conflicting packages have a priority of %5%.\n"
|
||||||
|
"To prioritise the new package:\n"
|
||||||
|
"\n"
|
||||||
|
" nix profile install %4% --priority %6%\n"
|
||||||
|
"\n"
|
||||||
|
"To prioritise the existing package:\n"
|
||||||
|
"\n"
|
||||||
|
" nix profile install %4% --priority %7%\n",
|
||||||
|
originalConflictingFilePath,
|
||||||
|
newConflictingFilePath,
|
||||||
|
originalConflictingRef.to_string(),
|
||||||
|
newConflictingRef.to_string(),
|
||||||
|
conflictError.priority,
|
||||||
|
conflictError.priority - 1,
|
||||||
|
conflictError.priority + 1
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -140,6 +140,36 @@ printf World2 > $flake2Dir/who
|
||||||
|
|
||||||
nix profile install $flake1Dir
|
nix profile install $flake1Dir
|
||||||
[[ $($TEST_HOME/.nix-profile/bin/hello) = "Hello World" ]]
|
[[ $($TEST_HOME/.nix-profile/bin/hello) = "Hello World" ]]
|
||||||
|
expect 1 nix profile install $flake2Dir
|
||||||
|
diff -u <(
|
||||||
|
nix --offline profile install $flake2Dir 2>&1 1> /dev/null \
|
||||||
|
| grep -vE "^warning: " \
|
||||||
|
|| true
|
||||||
|
) <(cat << EOF
|
||||||
|
error: An existing package already provides the following file:
|
||||||
|
|
||||||
|
$(nix build --no-link --print-out-paths ${flake1Dir}"#default.out")/bin/hello
|
||||||
|
|
||||||
|
This is the conflicting file from the new package:
|
||||||
|
|
||||||
|
$(nix build --no-link --print-out-paths ${flake2Dir}"#default.out")/bin/hello
|
||||||
|
|
||||||
|
To remove the existing package:
|
||||||
|
|
||||||
|
nix profile remove path:${flake1Dir}
|
||||||
|
|
||||||
|
The new package can also be installed next to the existing one by assigning a different priority.
|
||||||
|
The conflicting packages have a priority of 5.
|
||||||
|
To prioritise the new package:
|
||||||
|
|
||||||
|
nix profile install path:${flake2Dir} --priority 4
|
||||||
|
|
||||||
|
To prioritise the existing package:
|
||||||
|
|
||||||
|
nix profile install path:${flake2Dir} --priority 6
|
||||||
|
EOF
|
||||||
|
)
|
||||||
|
[[ $($TEST_HOME/.nix-profile/bin/hello) = "Hello World" ]]
|
||||||
nix profile install $flake2Dir --priority 100
|
nix profile install $flake2Dir --priority 100
|
||||||
[[ $($TEST_HOME/.nix-profile/bin/hello) = "Hello World" ]]
|
[[ $($TEST_HOME/.nix-profile/bin/hello) = "Hello World" ]]
|
||||||
nix profile install $flake2Dir --priority 0
|
nix profile install $flake2Dir --priority 0
|
||||||
|
|
Loading…
Reference in a new issue