lib.lists.foldl': Make strict in the initial accumulator

To maintain backwards compatibility, this can't be changed in the Nix language.
We can however ensure that the version Nixpkgs has the more intuitive behavior.
This commit is contained in:
Silvan Mosberger 2023-09-22 12:22:04 +02:00
parent 857a844ea8
commit 3b6169f87b
4 changed files with 16 additions and 6 deletions

View file

@ -392,7 +392,7 @@ rec {
foldlAttrs :: ( a -> String -> b -> a ) -> a -> { ... :: b } -> a
*/
foldlAttrs = f: init: set:
foldl'
builtins.foldl'
(acc: name: f acc name set.${name})
init
(attrNames set);

View file

@ -90,9 +90,12 @@ rec {
Reduce a list by applying a binary operator from left to right,
starting with an initial accumulator.
After each application of the operator, the resulting value is evaluated.
Before each application of the operator, the accumulator value is evaluated.
This behavior makes this function stricter than [`foldl`](#function-library-lib.lists.foldl).
Unlike [`builtins.foldl'`](https://nixos.org/manual/nix/unstable/language/builtins.html#builtins-foldl'),
the initial accumulator argument is evaluated before the first iteration.
A call like
```nix
@ -104,7 +107,7 @@ rec {
```nix
let
acc = op acc x ;
acc = builtins.seq acc (op acc x );
acc = builtins.seq acc (op acc x );
acc = builtins.seq acc (op acc x );
...
@ -135,7 +138,11 @@ rec {
# The list to fold
list:
builtins.foldl' op acc list;
# The builtin `foldl'` is a bit lazier than one might expect.
# See https://github.com/NixOS/nix/pull/7158.
# In particular, the initial accumulator value is not forced before the first iteration starts.
builtins.seq acc
(builtins.foldl' op acc list);
/* Map with index starting from 0

View file

@ -524,10 +524,10 @@ runTests {
expected = [ 3 2 1 ];
};
# The same as builtins.foldl', lib.foldl' doesn't evaluate the first accumulator strictly
# Compared to builtins.foldl', lib.foldl' evaluates the first accumulator strictly too
testFoldl'StrictInitial = {
expr = (builtins.tryEval (foldl' (acc: el: el) (throw "hello") [])).success;
expected = true;
expected = false;
};
# Make sure we don't get a stack overflow for large lists

View file

@ -226,6 +226,9 @@
- `networking.networkmanager.firewallBackend` was removed as NixOS is now using iptables-nftables-compat even when using iptables, therefore Networkmanager now uses the nftables backend unconditionally.
- [`lib.lists.foldl'`](https://nixos.org/manual/nixpkgs/stable#function-library-lib.lists.foldl-prime) now always evaluates the initial accumulator argument first.
If you depend on the lazier behavior, consider using [`lib.lists.foldl`](https://nixos.org/manual/nixpkgs/stable#function-library-lib.lists.foldl) or [`builtins.foldl'`](https://nixos.org/manual/nix/stable/language/builtins.html#builtins-foldl') instead.
- `rome` was removed because it is no longer maintained and is succeeded by `biome`.
- The `services.mtr-exporter.target` has been removed in favor of `services.mtr-exporter.jobs` which allows specifying multiple targets.