-
-
Notifications
You must be signed in to change notification settings - Fork 14.7k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
lib.extendMkDerivation: init #234651
base: master
Are you sure you want to change the base?
lib.extendMkDerivation: init #234651
Conversation
a2bd999
to
301bd5b
Compare
I should have checked more carefully. There's a function called Updat: fixed, see below. |
301bd5b
to
4f7af7a
Compare
Fix |
This pull request has been mentioned on NixOS Discourse. There might be relevant details there: |
This pull request has been mentioned on NixOS Discourse. There might be relevant details there: https://discourse.nixos.org/t/avoid-rec-expresions-in-nixpkgs/8293/18 |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
A mix of incremental and fundamental suggestions, because I can't decide for the nixpkgs-python maintainers.
I'm concerned about the increasing complexity, while the goal can be achieved through simplification instead: removing the python-specific level of overriding and turning it into an "overlay" on the mkDerivation
arguments instead.
The alternative strategy is to
- Provide this python layer, which contains the relevant
mkPyton*
logic in a way that works with overlay-style overriding. This can be done by reading the existing code, attribute for attribute, and adding the logic to the python layer. - Change the python packages to use that layer in combination with
mkDerivation
, instead of the currentmkPython*
functions. - Perhaps make the
mkPython
functions reuse the overlay so that they don't literally reimplement the same logic. I don't know if this would be worthwhile. - Eventually deprecate the
mkPython*
functions.
I have played around with step 1 of the alternative strategy. It is feasible, but it requires a bit of migration for each package. I'm not a nixpkgs-python maintainer, so I don't feel like I should be the one to make the decision whether to accept the complexity of this PR, or refactor to actually simplify the python logic by fitting it into a mkDerivation
layer.
doc/builders/builders.chapter.md
Outdated
}: | ||
|
||
lib.extendRecursiveBuilder stdenv.mkDerivation [ ] (finalAttrs: | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It's worth noting that finalAttrs
contains the final arguments to mkDerivation
attrs, and not the final arguments to the function you're constructing.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Now the arguments after applying the modifier
function is also considered.
IMO, the finalAttrs
is meant to be the final state of the attributes passing to the base builder (mkDerivation
). That's how user could access finalAttrs.finalPackage
and other goodies. If we just want to get the input argument set manually, the user could just lib.fix
the function themselves.
Input arguments that doesn't mean to be passed to the base builder are subject to special care, and should never enter the recursion. Those not-to-pass arguments are also the reason why builder overlays are not drop-in replacement of current builders.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I tried to add those special arguments to finalAttrs
, but that greatly increase the complexity of the code and makes behavior surprising (the specified result will be different from what is got through finalAttrs
by design).
A better way would be to encourage passing all the arguments. When all the arguments are passed, they will all be available inside finalAttrs
, and we could then switch to the overlay-based workflow specification (from the current, build-helper-based one).
in | ||
if builtins.isAttrs result then result | ||
else if builtins.isFunction result then { | ||
overridePythonAttrs = newArgs: makeOverridablePythonPackage f (overrideWith newArgs); | ||
__functor = self: result; | ||
} | ||
else result; | ||
else result); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This feels too custom or repeated (not sure yet which). Does this add mkDerivation
-like fixpoint logic at the python attrs level?
I believe we should merge the python attrs level into the mkDerivation attrs, so that the interface and implementation become simpler. Having multiple levels of overriding has a huge complexity cost, so getting rid of an unnecessary level would be a huge win. We'd get rid of overridePythonAttrs
and all the user facing complexity, implementation complexity and bugs that come with it.
The python-specific attrs can almost be implemented as an "overlay" on the mkDerivation
arguments. When I tried this, I think only like 3 attributes had the same name but a slightly different meaning. That made it a breaking change, but migrating those attributes is feasible and would vastly simplify the python/mkDerivation wiring.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
With same names that is generally speaking because the Python builder will extend the lists with some "defaults". That's really the only value of the custom builder over just plain hooks.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This feels too custom or repeated (not sure yet which). Does this add
mkDerivation
-like fixpoint logic at the python attrs level?
I personally prefer passing most of the attributes into mkDerivation
, encourage the use of overrideAttrs
and gradually deprecates overridePythonAttrs
as well as other builder-specific override methods.
That will be a mass rebuild, so I just work around the makeOverridablePythonPackage
obstacle in order to demonstrate the possibility to add the recursive attributes support without rebuild.
Recently the Packages Modules Working Group started investigating whether something like the module system could be used to evaluate packages. We're tracking all work in this repository, meeting weekly, but working mostly asynchronously. It would be great if you could join the Matrix room of the WG and chat with us, or even join the team yourself to work on such issues! |
Thank you for taking time reviewing this!
The idea to shift workflow-specific overlays sounds neat, and that could also be friendlier when packaging multi-language projects. Nevertheless, there are some issue on the way to the switch:
Overall, the goal of the proposed function is to add |
doc/builders/builders.chapter.md
Outdated
} | ||
``` | ||
|
||
A list of functions to modify the resulting derivation can be specified after the base builder. Modifications such as overriding and `extendDerivation` can be applied there. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Where do we find information about extendDerivation
?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
lib.customisation
is currently not presented in the Nixpkgs manual. We'll need to add the documentation for those functions before referring to them.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Besides, the Nixpkgs Library Functions documentation automatically generated from the comment don't have anchors. So there's no way to link against them so far.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
My fault. The anchor in the form function-library-<long attribute path>
is also automatically generated. E. g. function-library-lib.customisation.extendMkDerivation
.
This usage is very hard to discover, as the documentation is missing, and the manual provides no hyperlink in the title of each function document.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I addressed this review in the changes to add a new section about using lib.extendMkDerivatien
to define build helpers. Please take a look.
Just reviewing your change to input-remapper, I like how much this cleaned up its definition. :) |
954f48e
to
2ee39a7
Compare
f3c34fb
to
d868837
Compare
f5b76fb
to
b7aedc8
Compare
I rebased and built the manual successfully on my machine. with nix-build --no-out-link doc/ Why is the Nixpkgs Manual still failing to build on CI? |
b7aedc8
to
39e75d1
Compare
39e75d1
to
09d0d08
Compare
It gets built now. |
@philiptaron, how do you think about leaving only If people prefer the current implementation, I could still revert. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Couple curious questions.
lib/customisation.nix
Outdated
`attrsOverlay` | ||
: An overlay of attribute set, like the one taken by [overrideAttrs](#sec-pkg-overrideAttrs). | ||
: It is the implementation detail of the result build helper. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Since the function returned by calling this argument is added to the functionArgs, is this "implementation detail" note true?
There are a couple important properties of the the function returned by calling this argument:
- It must be a function of one argument, which is an attrset.
- If it is declared with the attribute set argument syntax (
{ ... }
) it must include a...
to accept arbitrary names in the provided attrset, and it must provide a default for all its other arguments.
I'd like to have a nicer name for this "inner function", but my attempts to name it are pretty unsuccessful. Maybe "inner overlay extension function"?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Since the function returned by calling this argument is added to the functionArgs, is this "implementation detail" not true?
Maybe it should be the "implementation" instead of the "implementation detail."
If it is declared with the attribute set argument syntax (
{ ... }
) it must include a...
to accept arbitrary names in the provided attrset, and it must provide a default for all its other arguments.
I think this requirement would hold only when inheritFunctionArgs = true
. Functions with attribute-set-like set patterns without ellipsis still show their arguments via lib.functionArgs
.
I'd like to have a nicer name for this "inner function", but my attempts to name it are pretty unsuccessful. Maybe "inner overlay extension function"?
That is one of the two hard things in computer science 😆 The idea of calling it an "attribute overlay" comes from overrideAttrs
, which now accepts an extension (overlay) that would be applied to the attribute set passing to mkDerivation
.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
That is one of the two hard things in computer science 😆 The idea of calling it an "attribute overlay" comes from
overrideAttrs
, which now accepts an extension (overlay) that would be applied to the attribute set passing tomkDerivation
.
Yes absolutely. 🙂 The term "attribute overlay" was (I thought) about the two-arity function of the form final: inner: { ... result ... }
. I'm searching for a term for the inner function, inner: result
or { ... }: result
, that distinguishes it from the encompassing one.
I think this requirement would hold only when inheritFunctionArgs = true. Functions with attribute-set-like set patterns without ellipsis still show their arguments via lib.functionArgs.
I tried experimenting with this in practice, and I think that's not the case. The ellipsis is required not because of the function args but because the function called with derivation construction args.
Here's my test case:
--- a/pkgs/test/overriding.nix
+++ b/pkgs/test/overriding.nix
@@ -66,13 +66,12 @@ let
test-extendMkDerivation =
let
- mkLocalDerivation = lib.extendMkDerivation { } pkgs.stdenv.mkDerivation (
+ mkLocalDerivation = lib.extendMkDerivation { inheritFunctionArgs = false; } pkgs.stdenv.mkDerivation (
finalAttrs:
{
preferLocalBuild ? true,
allowSubstitute ? false,
- ...
- }@args:
+ }:
{
inherit preferLocalBuild allowSubstitute;
}
The test nix-build -A tests.overriding
now fails to evaluate with this:
error: function 'anonymous lambda' called with unexpected argument 'version'
at /home/philip/Code/github.com/philiptaron/nixpkgs/master/pkgs/test/overriding.nix:71:9:
70| finalAttrs:
71| {
| ^
72| preferLocalBuild ? true,
So you can see that it must accept all the attributes that are passed to the derivation initially, such as version
, pname
, src
, etc.
The function arg stuff is absolutely fine with no ellipsis; but due to the usecase, the actual callers are not.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
So you can see that it must accept all the attributes that are passed to the derivation initially, such as
version
,pname
,src
, etc.The function arg stuff is absolutely fine with no ellipsis; but due to the usecase, the actual callers are not.
Most build helpers, especially those language- and framework-specific ones, do require ellipsis, as most of the attributes are expected to pass all the way down stdenv.mkDeriavtion
. However, some other build helpers, such as trivial build helpers and fetchers, tend to limit the argument they accept to make typos discoverable. Some provide an derivationArgs
attribute to accept extra attributes to pass down.
Let's say we would like to implement something like fetchFromGitHub
with lib.extendMkDerivation
. As we now probably don't need to support something other than hash
, we don't need the ellipses. Assuming fetchGit
would take fixed-point arguments, the code would be as follows:
{
lib,
fetchgit,
};
lib.extendMkDerivation {
constructDrv = fetchgit;
extendDerivation =
finalAttrs:
{
owner,
repo,
rev ? null,
tag ? null,
name ? "source",
githubBase ? "github.com"
passthru ? { },
meta ? { },
}:
{
inherit
name
;
url = "https://${githubBase}/${owner}/${repo}.git";
passthru = {
inherit
owner
repo
;
} // passthru;
};
removedArgNames = [
"githubBase"
"owner"
"repo"
];
}
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Here's a working implementation of a writeTextFile
-like build helper:
{
lib ? import <nixpkgs/lib>,
stdenvNoCC ? (import <nixpkgs> { }).stdenvNoCC,
}:
lib.extendMkDerivation {
constructDrv = stdenvNoCC.mkDerivation;
extendArgs =
finalAttrs:
{
name,
text,
destination ? "",
executable ? false,
derivationArgs ? { },
}:
let
baseName = baseNameOf finalAttrs.destination;
in
{
inherit
executable
destination
;
target = (placeholder "out") + finalAttrs.destination;
dontUnpack = true;
dontPatch = true;
dontConfigure = true;
dontBuild = true;
installPhase = ''
runHook preInstall
mkdir -p "$(dirname "$target")"
if [[ -e "$textPath" ]]; then
mv "$textPath" "$target"
else
echo -n "$text" > "$target"
fi
if [[ -n "''${executable-}" ]]; then
chmod +x "$target"
fi
runHook postInstall
'';
dontFixup = true;
doInstallCheck = true;
installCheckPhase = ''
runHook preInstallCheck
runHook postInstallCheck
'';
dontDist = true;
}
// derivationArgs
// {
passAsFile = [ "text" ] ++ derivationArgs.passAsFile or [ ];
meta =
lib.optionalAttrs (finalAttrs.executable && finalAttrs.destination == "/bin/${baseName}") {
mainProgram = baseName;
}
// derivationArgs.meta or { };
};
inheritFunctionArgs = false;
removedArgNames = [
"derivationArgs"
];
}
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@philiptaron I name the overlay extendArgs
, to emphasize that it's an "extension" (that's how the lib.fixedPoints
functions called an overlay) to apply to the fixed-point arguments (finalAttrs: { ... }
), and to distinguish from (the notorious) lib.extendDerivation
.
By the way, should we rename the removedArgNames
attribute? I'm out of naming ideas for this one.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm not in love with extendArgs
, since it's not clear to me whose args they are. I like extendDrvArgs
better, but I think that extendDrv
is just as good.
The naming scheme for all the other arguments is "imperative verb phrase". So "construct", "transform", and "extend" are all words that, if you write them out with the name of the function, make sense like this:
"[extendMkDerivation], construct [the] derivation with this!"
"[extendMkDerivation], transform [the] derivation with this!"
"[extendMkDerivation], extend [the] derivation with this!"
"[extendMkDerivation], inherit [the] function args [of the derivation constructor]!"
What's in square brackets is the implied subject and object.
So that should bias us towards towards "exclude" or "remove" rather than "excluded" or "removed", I think. What do you think about "excludeDrvArgNames"? It's a little wordy, but I think it both fits the pattern and is clear about the sort of thing it expects.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Regarding the attribute naming, I hope they can reflect their variable types.
The data diagram below shows that extend
and exclude
handle arguments, mkDrv
takes the arguments and returns a derivation, and transformDrv
handles the derivation.
How about the following names?
- extendDrvArgs
- excludeDrvArgNames
- constructDrv / makeDrv
- transformDrv
BTW, should we say doInheritFunctionArgs
instead of inheritFunctionArgs
, as it's a boolean switch?
+----------+------------finalAttrs-------------------+
| | |
v | ^
input --+----(--> exclude -+ |
| | +--> |
| | // --> mkDrv ==> transformDrv ==> output
| v +-->
+-> extend --------+
Legend:
---> arguments
===> derivation
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I don't think do
is needed as a prefix. Inherit is a fine verb. I like all the names that you have proposed, and prefer construct
to make
due to constructor
being a well-known concept in programming vs. maker
.
I like adding the |
09d0d08
to
f26ee78
Compare
f26ee78
to
83c0e00
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think this is now as simple as it needs to be. I'm approving, trusting that you'll finish dotting the i's and crossing the t's before merge.
I found your worked example of a |
requested change ("make as simple as possible, one function") completed
83c0e00
to
3d2084a
Compare
Add functions extendMkDerivation to lib.customisation. Co-authored-by: Robert Hensing <[email protected]> Co-authored-by: Valentin Gagarin <[email protected]> Co-authored-by: Lin Jian <[email protected]> Co-authored-by: Philip Taron <[email protected]>
0bd67a4
to
1e06a3f
Compare
I addressed @philiptaron's naming suggestions and simplified the documentation. As the transition process to reduce excluded/removed arguments and limit custom overriders like |
Add "Fixed-point arguments of build helpers" chapter in "Builde helpers" part. Co-authored-by: nicoo <[email protected]> Co-authored-by: Silvan Mosberger <[email protected]> Co-authored-by: Valentin Gagarin <[email protected]> Co-authored-by: Lin Jian <[email protected]> Co-authored-by: Philip Taron <[email protected]>
1e06a3f
to
bbdf860
Compare
@fricklerhandwerk, IIRC, the changes you requested concern mainly the documentation. If you are available, could you kindly review the (now greatly simplified) documentation? |
Description of changes
This PR introduces a unified approach to implementing build helpers that support fixed-point arguments and bring such support to existing build helpers.
The fixed-point arguments support in
stdenv.mkDerivation
is introduced in #119942, and there are ongoing attempts to make other build helpers support such syntax (buildRustPackage
refactor #194475,buildGoModule
refactor #225051). The overlay styleoverrideAttrs
brought by thestdenv.mkDerivation
change can be used to implement the functionality, which is adopted by thebuildRustPackage
refactor and the previous version of thebuildGoModule
refactor. The challenge of such an approach is that the whole set pattern matching the input arguments are degenerated into a single identifier, making it hard to see from the source code which attributes to the build helper accepts.The new Nixpkgs Library function,
lib.extendMkDerivation
accepts a base build helper and an attribute overlay (an overlay in the formfinalAttrs: args: <attrsToUpdate>
), and returns a new build helper by extending the base build helper with the attribute overlay via its<pkg>.overrideAttrs
.The following is the definition of an example build helper,
mkLocalDerivation
:For existing build helpers with arguments that cannot be passed to the base build helper,
lib.extendMkDerivation
provides an attributeremovedAttrNames
to specify arguments not to pass down to the base build helper.Aside from
removedAttrNames
lib.extendMkDerivation
, other optional arguments are used to manipulate their behaviors. For example,lib.adaptMkDerivation { transformDrv = toPythonModule; }
applies the functiontoPythonModule
to the derivation produced by the derived build helper.Cc:
Python maintainers: @FRidh @mweinelt @jonringer
Author of the
buildRustPackage
refactor PR @amesgenReviewer of the
buildGoModule
refactor PR @zowoqAuthor of the merged recursive
stdenv.mkDerivation
PR @roberthPeople who mention 119942 in Python application definition: @LunNova
Things done
sandbox = true
set innix.conf
? (See Nix manual)nix-shell -p nixpkgs-review --run "nixpkgs-review rev HEAD"
. Note: all changes have to be committed, also see nixpkgs-review usage./result/bin/
)