Minimize Download of Dependency Flakes (on demand)

Hi everyone,

long time reader here, never posted something. Until today, I’m stuck.

I have created NixOS-Flake to configure several hosts. Since every Host has a different role, the inputs (Input Flakes) start piling up. Every Host has to download all the inputs, even if they only use a subset.

With a very high autoUpdate interval (hourly), my hosts make so many requests, that I get rate limited from github. (I’m aware that with an github account, the amount of allowed Github-API calls will be more than enough, but thats beside the point)

Since I modularized my repo by roles, I was asking myself, how i have to organize my code, such that when Import a module (role) then and only then will the input be downloaded.

I don’t care for reproducibility that much (I dont write lock files at all) in case that would change your answer.

Feel free to make suggestions, and ask questions if I wasn’t clear. I’m not a native speaker :slight_smile:

Thanks :slight_smile:

1 Like

What kind of flake inputs exactly do you have there? Does the automatic update actually updates the inputs? Or does it only check whether there are new commits on the branches you specify in the main flake.nix? I know that these Nix settings are probably relevant:

https://nix.dev/manual/nix/2.24/command-ref/conf-file#conf-tarball-ttl

https://nix.dev/manual/nix/2.24/command-ref/conf-file#conf-narinfo-cache-negative-ttl

https://nix.dev/manual/nix/2.24/command-ref/conf-file#conf-narinfo-cache-positive-ttl

1 Like

Flake nix:

...
inputs = {
    nixpkgs.url = "github:NixOS/nixpkgs/nixos-unstable-small";
  
    home-manager.url = "github:nix-community/home-manager/release-24.11";
    home-manager.inputs.nixpkgs.follows = "nixpkgs";

    sops-nix.url = "github:Mic92/sops-nix";
    sops-nix.inputs.nixpkgs.follows = "nixpkgs";

    conduwuit = {
      url = "github:girlbossceo/conduwuit";
      inputs.nixpkgs.follows = "nixpkgs";
    };

    nixos-hardware = {
      url = "github:NixOS/nixos-hardware/master";
      inputs.nixpkgs.follows = "nixpkgs";
    };
  };
...

Updates:

system.autoUpgrade = {
      enable = true;
      flake = "github:j340m3/nixos";
      #flake = inputs.self.outPath;
      flags = [ 
      # "--update-input" "nixpkgs"
        "--no-write-lock-file"
      ];
      allowReboot = config.allowReboot;
      dates = lib.mkDefault "hourly";
      rebootWindow = {
        lower = "22:00";
        upper = "08:00";
      };
    };

When I manually update the system, I usually run:

nixos-rebuild switch --upgrade --flake github:j340m3/nixos --option tarball-ttl 0 --no-write-lock-file 

And I basically want to move the conduwuit input download form flake.nix to /subfolder/role1.nix and only when role1.nix is imported, then nix should download the dependency (if there is a newer version)

Remove --upgrade, it does nothing with flakes.

I’d also recommend you not use this, this will essentially disable your user’s flake cache, and will cause you to hit the rate-limit sooner.

Anyway, what you want is not possible with flake inputs currently.

There’s some speculative PRs to improve the situation. Though even then, all inputs must be located in the flake.nix, you can’t split them up into multiple files.

1 Like

OK nothing very special I see there - the inputs are not updated frequently enough to justify such an intensive hourly automatic update. Currently Hydra’s web interface gives me 504 timeout, but in principal here you should be able see the frequency in which the channel you are using is updated. Other channels’ update frequency can be inferred from here. IIRC that for nixos-unstable-small it is less then twice a day.

I believe that Nixpkgs is the most frequently updated flake input you have there, and hence running your update 4 times a day should be enough to guarantee your are up to date all the time. See Nyquist–Shannon sampling theorem - Wikipedia .

I think you might be able to hack something with builtins.getFlake, but the problem is you won’t get the benefits of committing flake.lock and tracking the commit hashes you are actually using in each system evaluation.

Long story short: No need to update flake inputs hourly :).

Also, why are you using flakes at all if you don’t care about lockfiles? The entire point of flakes is to provide pure eval which implies a lockfile spec.

Have a nerd cookie for citing Nyquist-Shannon :+1: (But wouln’t that just ensure that I would get
every state change? I would still have to wait an average of 3 hours until I would get my tasty updates :wink: )

What is the time interval string for every 6 hours? I once tried a few strings but then got frustrated and went with hourly.

Is there a way to set this option granulary for each input? When I push a change to my github repo, I would have to wait for a long while until the nixos-rebuild switch-command would get the newer commit.

You’re absolutly right, it’s a bit nonsensical. Here some explanations:

  • I’m still learning nix, I didn’t know better. Maybe technical debt?
  • Lots of code snippets and tutorials use flakes and it was harder for me to stick to a non-flake version.
  • I (mis-)use NixOS as kind of a reverse Ansible. I liked the idea of reproducibility, but at one point, logging into every host to update the lock files felt tedious. So I changed parts of my setup so that I only need to touch it, when the syntax of packages changes (e.g hardware.alsamixerservices.alsamixer…).
2 Likes

And nixos did complain, that it couldn’t write a lock file for the flake…

As the docs of this dates option states, the syntax is explained (pretty good IMO) in the manpage systemd.time(7). I think that every 6 hours can be set via simply dates = "6h";.

Have an obsession cookie :cookie: for asking that :laughing: . And seriously: If you are really that impatient - setup a GitHub API key.

1 Like

I tried it with “6h”, and “6hours”, which would also be in spec. Nixos begs to differ:

restarting the following units: nixos-upgrade.timer
Failed to restart nixos-upgrade.timer: Unit nixos-upgrade.timer has a bad unit file setting.
warning: error(s) occurred while switching to the new configuration

And about the updates, I was just nitpicking :wink:

Every 6 h = *-*-* 0/6:00:00 for anyone that finds this :slight_smile: