Skip to content

Commit 0a513df

Browse files
committed
feat: start using lazyDerivation for faster TUI response times
1 parent c4d2c85 commit 0a513df

10 files changed

+242
-133
lines changed

cells/lib/ops.nix

+1
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,7 @@ in {
1414
mkSetup = import ./ops/mkSetup.nix {inherit inputs cell;};
1515
mkUser = import ./ops/mkUser.nix {inherit inputs cell;};
1616
writeScript = import ./ops/writeScript.nix {inherit inputs cell;};
17+
lazyDerivation = import ./ops/lazyDerivation.nix {inherit inputs cell;};
1718

1819
mkOCI = import ./ops/mkOCI.nix {inherit inputs cell;};
1920
mkDevOCI = import ./ops/mkDevOCI.nix {inherit inputs cell;};

cells/lib/ops/lazyDerivation.nix

+108
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,108 @@
1+
{
2+
inputs,
3+
cell,
4+
}: let
5+
inherit (inputs) nixpkgs std;
6+
l = nixpkgs.lib // builtins;
7+
inherit (l) throwIfNot;
8+
in (
9+
if l ? lazyDerivation
10+
then l.lazyDerivation
11+
else
12+
{
13+
/*
14+
Restrict a derivation to a predictable set of attribute names, so
15+
that the returned attrset is not strict in the actual derivation,
16+
saving a lot of computation when the derivation is non-trivial.
17+
18+
This is useful in situations where a derivation might only be used for its
19+
passthru attributes, improving evaluation performance.
20+
21+
The returned attribute set is lazy in `derivation`. Specifically, this
22+
means that the derivation will not be evaluated in at least the
23+
situations below.
24+
25+
For illustration and/or testing, we define derivation such that its
26+
evaluation is very noticeable.
27+
28+
let derivation = throw "This won't be evaluated.";
29+
30+
In the following expressions, `derivation` will _not_ be evaluated:
31+
32+
(lazyDerivation { inherit derivation; }).type
33+
34+
attrNames (lazyDerivation { inherit derivation; })
35+
36+
(lazyDerivation { inherit derivation; } // { foo = true; }).foo
37+
38+
(lazyDerivation { inherit derivation; meta.foo = true; }).meta
39+
40+
In these expressions, it `derivation` _will_ be evaluated:
41+
42+
"${lazyDerivation { inherit derivation }}"
43+
44+
(lazyDerivation { inherit derivation }).outPath
45+
46+
(lazyDerivation { inherit derivation }).meta
47+
48+
And the following expressions are not valid, because the refer to
49+
implementation details and/or attributes that may not be present on
50+
some derivations:
51+
52+
(lazyDerivation { inherit derivation }).buildInputs
53+
54+
(lazyDerivation { inherit derivation }).passthru
55+
56+
(lazyDerivation { inherit derivation }).pythonPath
57+
58+
*/
59+
lazyDerivation = args @ {
60+
# The derivation to be wrapped.
61+
derivation,
62+
# Optional meta attribute.
63+
#
64+
# While this function is primarily about derivations, it can improve
65+
# the `meta` package attribute, which is usually specified through
66+
# `mkDerivation`.
67+
meta ? null,
68+
# Optional extra values to add to the returned attrset.
69+
#
70+
# This can be used for adding package attributes, such as `tests`.
71+
passthru ? {},
72+
}: let
73+
# These checks are strict in `drv` and some `drv` attributes, but the
74+
# attrset spine returned by lazyDerivation does not depend on it.
75+
# Instead, the individual derivation attributes do depend on it.
76+
checked =
77+
throwIfNot (derivation.type or null == "derivation")
78+
"lazySimpleDerivation: input must be a derivation."
79+
throwIfNot
80+
(derivation.outputs == ["out"])
81+
# Supporting multiple outputs should be a matter of inheriting more attrs.
82+
"The derivation ${derivation.name or "<unknown>"} has multiple outputs. This is not supported by lazySimpleDerivation yet. Support could be added, and be useful as long as the set of outputs is known in advance, without evaluating the actual derivation."
83+
derivation;
84+
in
85+
{
86+
# Hardcoded `type`
87+
#
88+
# `lazyDerivation` requires its `derivation` argument to be a derivation,
89+
# so if it is not, that is a programming error by the caller and not
90+
# something that `lazyDerivation` consumers should be able to correct
91+
# for after the fact.
92+
# So, to improve laziness, we assume correctness here and check it only
93+
# when actual derivation values are accessed later.
94+
type = "derivation";
95+
96+
# A fixed set of derivation values, so that `lazyDerivation` can return
97+
# its attrset before evaluating `derivation`.
98+
# This must only list attributes that are available on _all_ derivations.
99+
inherit (checked) outputs out outPath outputName drvPath name system;
100+
101+
# The meta attribute can either be taken from the derivation, or if the
102+
# `lazyDerivation` caller knew a shortcut, be taken from there.
103+
meta = args.meta or checked.meta;
104+
}
105+
// passthru;
106+
}
107+
.lazyDerivation
108+
)

cells/lib/ops/mkDevOCI.nix

+1-1
Original file line numberDiff line numberDiff line change
@@ -42,7 +42,7 @@ in
4242
vscode ? false,
4343
slim ? false,
4444
user ? "user",
45-
tag ? "",
45+
tag ? null,
4646
pkgs ? [],
4747
setup ? [],
4848
perms ? [],

cells/lib/ops/mkOCI.nix

+61-49
Original file line numberDiff line numberDiff line change
@@ -31,7 +31,7 @@ in
3131
tag ?
3232
if meta ? tags && (l.length meta.tags) > 0
3333
then l.head meta.tags
34-
else "",
34+
else null,
3535
setup ? [],
3636
layers ? [],
3737
runtimeInputs ? [],
@@ -47,57 +47,69 @@ in
4747
mkdir -p $out/bin
4848
ln -s ${l.getExe entrypoint} $out/bin/entrypoint
4949
'';
50-
options' =
51-
{
52-
inherit name meta;
5350

54-
# Layers are nested to reduce duplicate paths in the image
55-
layers =
56-
[
57-
# Primary layer is the entrypoint layer
58-
(n2c.buildLayer {
59-
deps = [entrypoint];
60-
maxLayers = 50;
61-
layers = [
62-
# Runtime inputs layer
63-
(n2c.buildLayer {
64-
deps = runtimeInputs;
65-
maxLayers = 10;
66-
})
67-
];
68-
})
69-
]
70-
++ layers;
51+
image =
52+
l.throwIf (args ? tag && meta ? tags)
53+
"mkOCI/mkStandardOCI/mkDevOCI: use of `tag` and `meta.tags` arguments are not supported together. Remove the former."
54+
n2c.buildImage (
55+
l.recursiveUpdate options {
56+
inherit name tag;
7157

72-
maxLayers = 25;
73-
copyToRoot =
74-
[
75-
(nixpkgs.buildEnv {
76-
name = "root";
77-
paths = [setupLinks] ++ setup;
78-
})
79-
]
80-
++ options.copyToRoot or [];
58+
# Layers are nested to reduce duplicate paths in the image
59+
layers =
60+
[
61+
# Primary layer is the entrypoint layer
62+
(n2c.buildLayer {
63+
deps = [entrypoint];
64+
maxLayers = 50;
65+
layers = [
66+
# Runtime inputs layer
67+
(n2c.buildLayer {
68+
deps = runtimeInputs;
69+
maxLayers = 10;
70+
})
71+
];
72+
})
73+
]
74+
++ layers;
8175

82-
config = l.recursiveUpdate config {
83-
User = uid;
84-
Group = gid;
85-
Entrypoint = ["/bin/entrypoint"];
86-
Labels = l.mapAttrs' (n: v: l.nameValuePair "org.opencontainers.image.${n}" v) labels;
87-
};
76+
maxLayers = 25;
77+
copyToRoot =
78+
[
79+
(nixpkgs.buildEnv {
80+
name = "root";
81+
paths = [setupLinks] ++ setup;
82+
})
83+
]
84+
++ options.copyToRoot or [];
8885

89-
# Setup tasks can include permissions via the passthru.perms attribute
90-
perms = l.flatten ((l.map (s: l.optionalAttrs (s ? passthru && s.passthru ? perms) s.passthru.perms)) setup) ++ perms;
91-
}
92-
// l.throwIf (args ? tag && meta ? tags)
93-
"mkOCI: use of `tag` and `meta.tags` arguments are not supported together. Remove the former."
94-
(l.optionalAttrs (tag != "") {inherit tag;});
86+
config = l.recursiveUpdate config {
87+
User = uid;
88+
Group = gid;
89+
Entrypoint = ["/bin/entrypoint"];
90+
Labels = l.mapAttrs' (n: v: l.nameValuePair "org.opencontainers.image.${n}" v) labels;
91+
};
9592

96-
image = n2c.buildImage (l.recursiveUpdate options options');
93+
# Setup tasks can include permissions via the passthru.perms attribute
94+
perms = l.flatten ((l.map (s: l.optionalAttrs (s ? passthru && s.passthru ? perms) s.passthru.perms)) setup) ++ perms;
95+
}
96+
);
97+
in let
98+
mainTag =
99+
if tag != null && tag != ""
100+
then tag
101+
else
102+
builtins.unsafeDiscardStringContext
103+
(l.head (l.strings.splitString "-" (baseNameOf image.outPath)));
104+
tags = l.unique ([mainTag] ++ meta.tags or []);
97105
in
98-
l.removeAttrs image [
99-
"copyTo"
100-
"copyToDockerDaemon"
101-
"copyToPodman"
102-
"copyToRegistry"
103-
]
106+
cell.ops.lazyDerivation {
107+
inherit meta;
108+
derivation = image;
109+
passthru = {
110+
image.name = "${name}:${mainTag}";
111+
image.repo = name;
112+
image.tag = mainTag;
113+
image.tags = l.unique ([mainTag] ++ meta.tags or []);
114+
};
115+
}

cells/lib/ops/mkOperable.nix

+15-12
Original file line numberDiff line numberDiff line change
@@ -30,6 +30,7 @@ in
3030
debugInputs ? [],
3131
livenessProbe ? null,
3232
readinessProbe ? null,
33+
meta ? {},
3334
}: let
3435
# nixpkgs.runtimeShell is a path to the shell, not a derivation
3536
runtimeShellBin =
@@ -69,18 +70,20 @@ in
6970
'';
7071
};
7172
in
72-
(cell.ops.writeScript
73-
({
74-
inherit runtimeInputs runtimeEnv;
75-
name = "operable-${package.name}";
76-
text = ''
77-
${runtimeScript}
78-
'';
79-
}
80-
// l.optionalAttrs (runtimeShell != null) {
81-
inherit runtimeShell;
82-
}))
83-
// {
73+
cell.ops.lazyDerivation {
74+
inherit meta;
75+
derivation =
76+
cell.ops.writeScript
77+
({
78+
inherit runtimeInputs runtimeEnv;
79+
name = "operable-${package.name}";
80+
text = ''
81+
${runtimeScript}
82+
'';
83+
}
84+
// l.optionalAttrs (runtimeShell != null) {
85+
inherit runtimeShell;
86+
});
8487
passthru =
8588
# These attributes are useful for informing later stages
8689
{

cells/lib/ops/mkStandardOCI.nix

+7-12
Original file line numberDiff line numberDiff line change
@@ -29,7 +29,7 @@ in
2929
args @ {
3030
name,
3131
operable,
32-
tag ? "",
32+
tag ? null,
3333
setup ? [],
3434
uid ? "65534",
3535
gid ? "65534",
@@ -40,11 +40,10 @@ in
4040
options ? {},
4141
meta ? {},
4242
}: let
43-
inherit (operable) passthru;
44-
inherit (operable.passthru) livenessProbe readinessProbe runtimeInputs runtime debug;
43+
inherit (operable) livenessProbe readinessProbe runtimeInputs runtime debug;
4544

46-
hasLivenessProbe = passthru ? livenessProbe;
47-
hasReadinessProbe = passthru ? readinessProbe;
45+
hasLivenessProbe = operable ? livenessProbe;
46+
hasReadinessProbe = operable ? readinessProbe;
4847
hasDebug = args.debug or false;
4948

5049
# Link useful paths into the container.
@@ -105,19 +104,15 @@ in
105104
nss = nixpkgs.writeTextDir "etc/nsswitch.conf" ''
106105
hosts: files dns
107106
'';
108-
109-
tag' =
110-
l.throwIf (args ? tag && meta ? tags)
111-
"mkStandardOCI: use of `tag` and `meta.tags` arguments are not supported together. Remove the former."
112-
l.optionalString (tag != "") ((import "${inputs.self}/deprecation.nix" inputs).warnLegacyTag tag);
113107
in
114108
with dmerge;
109+
l.throwIf (args ? tag && meta ? tags)
110+
"mkStandardOCI: use of `tag` and `meta.tags` arguments are not supported together. Remove the former."
115111
cell.ops.mkOCI (
116112
merge
117113
{
118-
inherit name uid gid labels options perms config meta setup runtimeInputs;
114+
inherit name tag uid gid labels options perms config meta setup runtimeInputs;
119115
entrypoint = operable';
120-
tag = tag'; # ideally ''
121116
}
122117
{
123118
# mkStandardOCI differentiators over mkOCI

0 commit comments

Comments
 (0)