aboutsummaryrefslogtreecommitdiff
path: root/src/libcmd/installable-derived-path.cc
blob: a9921b90125bee4dee5356f6d0214746f594c1a5 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
#include "installable-derived-path.hh"
#include "derivations.hh"

namespace nix {

std::string InstallableDerivedPath::what() const
{
    return derivedPath.to_string(*store);
}

DerivedPathsWithInfo InstallableDerivedPath::toDerivedPaths()
{
    return {{.path = derivedPath, .info = {} }};
}

std::optional<StorePath> InstallableDerivedPath::getStorePath()
{
    return std::visit(overloaded {
        [&](const DerivedPath::Built & bfd) {
            return bfd.drvPath;
        },
        [&](const DerivedPath::Opaque & bo) {
            return bo.path;
        },
    }, derivedPath.raw());
}

InstallableDerivedPath InstallableDerivedPath::parse(
    ref<Store> store,
    std::string_view prefix,
    ExtendedOutputsSpec extendedOutputsSpec)
{
    auto derivedPath = std::visit(overloaded {
        // If the user did not use ^, we treat the output more liberally.
        [&](const ExtendedOutputsSpec::Default &) -> DerivedPath {
            // First, we accept a symlink chain or an actual store path.
            auto storePath = store->followLinksToStorePath(prefix);
            // Second, we see if the store path ends in `.drv` to decide what sort
            // of derived path they want.
            //
            // This handling predates the `^` syntax. The `^*` in
            // `/nix/store/hash-foo.drv^*` unambiguously means "do the
            // `DerivedPath::Built` case", so plain `/nix/store/hash-foo.drv` could
            // also unambiguously mean "do the DerivedPath::Opaque` case".
            //
            // Issue #7261 tracks reconsidering this `.drv` dispatching.
            return storePath.isDerivation()
                ? (DerivedPath) DerivedPath::Built {
                    .drvPath = std::move(storePath),
                    .outputs = OutputsSpec::All {},
                }
                : (DerivedPath) DerivedPath::Opaque {
                    .path = std::move(storePath),
                };
        },
        // If the user did use ^, we just do exactly what is written.
        [&](const ExtendedOutputsSpec::Explicit & outputSpec) -> DerivedPath {
            return DerivedPath::Built {
                .drvPath = store->parseStorePath(prefix),
                .outputs = outputSpec,
            };
        },
    }, extendedOutputsSpec.raw());
    return InstallableDerivedPath {
        store,
        std::move(derivedPath),
    };
}

}