1# Contributing to Nixpkgs 2 3This document is for people wanting to contribute to Nixpkgs. 4This involves changes that are proposed using [GitHub](https://github.com) [pull requests](https://docs.github.com/pull-requests) to the [Nixpkgs repository](https://github.com/nixos/nixpkgs). 5 6A GitHub account is recommended, which you can sign up for [here](https://github.com/signup). 7See [here](https://discourse.nixos.org/t/about-the-patches-category/477) for how to contribute without a GitHub account. 8 9This document assumes that you already know how to use GitHub and Git. 10If that's not the case, we recommend learning about it [here](https://docs.github.com/en/get-started/quickstart/hello-world). 11 12## Overview 13[overview]: #overview 14 15This file contains general contributing information. 16More specific information about individual parts of Nixpkgs can be found here: 17- [`doc`](./doc/README.md): Sources and infrastructure for the [Nixpkgs manual](https://nixos.org/manual/nixpkgs/stable/) 18- [`lib`](./lib/README.md): Sources and documentation of the [library functions](https://nixos.org/manual/nixpkgs/stable/#chap-functions) 19- [`maintainers`](./maintainers/README.md): Nixpkgs maintainer and team listings, maintainer scripts 20- [`nixos`](./nixos/README.md): Implementation of [NixOS](https://nixos.org/manual/nixos/stable/) 21- [`pkgs`](./pkgs/README.md): Package and [builder](https://nixos.org/manual/nixpkgs/stable/#part-builders) definitions 22 23# How to's 24 25## How to create pull requests 26[pr-create]: #how-to-create-pull-requests 27 28This section describes how changes can be proposed with a pull request (PR). 29 30> [!Note] 31> Be aware that contributing implies licensing those contributions under the terms of [COPYING](./COPYING), an MIT-like license. 32 330. Set up a local version of Nixpkgs to work with: 34 1. [Fork](https://docs.github.com/en/get-started/quickstart/fork-a-repo#forking-a-repository) the [Nixpkgs repository](https://github.com/nixos/nixpkgs). 35 1. [Clone the forked repository](https://docs.github.com/en/get-started/quickstart/fork-a-repo#cloning-your-forked-repository) into a local `nixpkgs` directory. 36 1. [Configure the upstream Nixpkgs repository](https://docs.github.com/en/get-started/quickstart/fork-a-repo#configuring-git-to-sync-your-fork-with-the-upstream-repository). 37 381. Select the appropriate [base branch](https://docs.github.com/en/pull-requests/collaborating-with-pull-requests/proposing-changes-to-your-work-with-pull-requests/about-branches#working-with-branches) for the change, as [described here][branch]. 39 If in doubt, use `master`. 40 This can be changed later by [rebasing][rebase]. 41 422. Create a new Git branch, ideally such that: 43 - The name of the branch hints at your change, e.g. `update-hello`. 44 - The branch contains the most recent base branch. 45 46 We'll assume the base branch `master` here. 47 48 ```bash 49 # Make sure you have the latest changes from upstream Nixpkgs 50 git fetch upstream 51 52 # Create and switch to a new branch, based on the base branch in Nixpkgs 53 git switch --create update-hello upstream/master 54 ``` 55 56 To avoid potentially having to download and build many derivations, you can base on a specific [Git commit](https://www.git-scm.com/docs/gitglossary#def_commit) instead: 57 - The commit of the latest `nixpkgs-unstable` channel, available [here](https://channels.nixos.org/nixpkgs-unstable/git-revision). 58 - The commit of a local Nixpkgs downloaded using [nix-channel](https://nixos.org/manual/nix/stable/command-ref/nix-channel), available using `nix-instantiate --eval --expr '(import <nixpkgs/lib>).trivial.revisionWithDefault null'` 59 - If you're using NixOS, the commit of your NixOS installation, available with `nixos-version --revision`. 60 61 You can use this commit instead of `upstream/master` in the above command: 62 ```bash 63 # Here, b9c03fbb is an example commit from nixpkgs-unstable 64 git switch --create update-hello b9c03fbb 65 ``` 66 673. Make your changes in the local Nixpkgs repository and: 68 - Adhere to both the [general code conventions][code-conventions], and the relevant [specific code conventions][overview]. 69 - Test the changes. 70 - If necessary, document the changes. 71 72 See the [overview section][overview] for more specific information. 73 744. Commit your changes using `git commit`. 75 Make sure to adhere to the [commit conventions](#commit-conventions). 76 77 Repeat the steps 3-4 as many times as necessary. 78 Advance to the next step once all the commits make sense together. 79 You can view your commits with `git log`. 80 815. Push your commits to your fork of Nixpkgs: 82 ``` 83 git push --set-upstream origin HEAD 84 ``` 85 86 The above command will output a link to directly do the next step: 87 ``` 88 remote: Create a pull request for 'update-hello' on GitHub by visiting: 89 remote: https://github.com/myUser/nixpkgs/pull/new/update-hello 90 ``` 91 926. [Create a pull request](https://docs.github.com/en/pull-requests/collaborating-with-pull-requests/proposing-changes-to-your-work-with-pull-requests/creating-a-pull-request#creating-the-pull-request) from the new branch in your Nixpkgs fork to the upstream Nixpkgs repository. 93 Use the branch from step 1 as the PR's base branch. 94 Go through the [pull request template][pr-template]. 95 967. Respond to review comments and potentially to CI failures and merge conflicts by updating the PR. 97 Always keep it in a mergeable state. 98 99 The non-technical side of this process is covered in [I opened a PR, how do I get it merged?](#i-opened-a-pr-how-do-i-get-it-merged). 100 101 The [ofborg](https://github.com/NixOS/ofborg) CI system will perform checks to ensure code quality. 102 You can see the results at the bottom of the PR. 103 See [the ofborg Readme](https://github.com/NixOS/ofborg#readme) for more details. 104 105 - To add new commits, repeat steps 3-4 and push the result: 106 ``` 107 git push 108 ``` 109 110 - To change existing commits, [rewrite the Git history](https://git-scm.com/book/en/v2/Git-Tools-Rewriting-History). 111 Useful Git commands for this are `git commit --patch --amend` and `git rebase --interactive`. 112 With a rewritten history you need to force-push the commits: 113 ``` 114 git push --force-with-lease 115 ``` 116 117 - If there are merge conflicts, you will have to [rebase the branch](https://git-scm.com/book/en/v2/Git-Branching-Rebasing) onto the current **base branch**. 118 Sometimes this can be done [on GitHub directly](https://docs.github.com/en/pull-requests/collaborating-with-pull-requests/proposing-changes-to-your-work-with-pull-requests/keeping-your-pull-request-in-sync-with-the-base-branch#updating-your-pull-request-branch). 119 To rebase locally: 120 ``` 121 git fetch upstream 122 git rebase upstream/master 123 git push --force-with-lease 124 ``` 125 126 Use the base branch from step 1 instead of `upstream/master`. 127 128 - If you need to change the base branch, [rebase][rebase]. 129 1308. If your PR is merged and [acceptable for releases][release-acceptable], you may [backport][pr-backport] it. 131 132### Pull request template 133[pr-template]: #pull-request-template 134 135The pull request template helps to determine which steps have been taken so far. 136Details not covered by the title and links to existing related issues should go at the top. 137 138When a PR is created, it will be pre-populated with some checkboxes. 139 140#### Tested using sandboxing 141 142When sandbox builds are enabled, Nix will set up an isolated environment for each build process. 143It is used to remove further hidden dependencies set by the build environment, to improve reproducibility. 144This includes access to the network during the build outside of `fetch*` functions and files outside the Nix store. 145Depending on the operating system, access to other resources is blocked as well; see [sandbox](https://nixos.org/manual/nix/stable/command-ref/conf-file#conf-sandbox) in the Nix manual for details. 146 147Please test builds with sandboxing enabled, because it is also used in [Hydra](https://nixos.org/hydra). 148 149If you are on Linux, sandboxing is enabled by default. 150On other platforms, sandboxing is disabled by default due to a small performance hit on each build. 151 152Please enable sandboxing **before** building the package by adding the following to `/etc/nix/nix.conf`: 153 154 ```ini 155 sandbox = true 156 ``` 157 158#### Built on platform(s) 159 160Many Nix packages are designed to run on multiple platforms. 161As such, it’s important to let the maintainer know which platforms you have tested on. 162It’s not always practical to test all platforms, and it’s not required for a pull request to be merged. 163Only check the platforms you tested the build on in this section. 164 165#### Tested via one or more NixOS test(s) if existing and applicable for the change (look inside nixos/tests) 166 167Packages with automated tests are likely merged quicker, because they don’t require as much manual testing. 168If there are existing tests for the package, they should be run. 169NixOS tests can only be run on linux. 170For more details on writing and running tests, see the [section in the NixOS manual](https://nixos.org/nixos/manual/index.html#sec-nixos-tests). 171 172#### Tested compilation of all pkgs that depend on this change using `nixpkgs-review` 173 174If you are modifying a package, you can use `nixpkgs-review` to make sure all packages that depend on the updated package still build. 175It can work on uncommitted changes with the `wip` option or on a specific pull request. 176 177Review changes from pull request number 12345: 178 179```ShellSession 180nix-shell -p nixpkgs-review --run "nixpkgs-review pr 12345" 181``` 182 183Alternatively, with flakes (and analogously for the other commands below): 184 185```ShellSession 186nix run nixpkgs#nixpkgs-review -- pr 12345 187``` 188 189Review uncommitted changes: 190 191```ShellSession 192nix-shell -p nixpkgs-review --run "nixpkgs-review wip" 193``` 194 195Review changes from the last commit: 196 197```ShellSession 198nix-shell -p nixpkgs-review --run "nixpkgs-review rev HEAD" 199``` 200 201#### Tested execution of all binary files (usually in `./result/bin/`) 202 203It's important to test a modified package's executables. 204Look into `./result/bin` and run all files in there, or at a minimum, the main executable. 205For example, if you make a change to `texlive`, you probably would only check the binaries associated with the change you made, rather than testing all of them. 206 207#### Meets Nixpkgs contribution standards 208 209The last checkbox is about whether it fits the guidelines in this `CONTRIBUTING.md` file. 210This document details our standards for commit messages, reviews, licensing of contributions, etc... 211Everyone should read and understand these standards before submitting a pull request. 212 213### Rebasing between branches (i.e. from `master` to `staging`) 214[rebase]: #rebasing-between-branches-ie-from-master-to-staging 215 216Sometimes, changes must be rebased between branches. 217One example is, if the number of rebuilds caused is too large for the original target branch. 218 219In the following example, the current `feature` branch is based on `master`, and we rebase it to have the PR target `staging`. 220We rebase on the _merge base_ between `master` and `staging` to avoid too many local rebuilds. 221 222 223```console 224# Rebase your commits onto the common merge base 225git rebase --onto upstream/staging... upstream/master 226# Force push your changes 227git push origin feature --force-with-lease 228``` 229 230The syntax `upstream/staging...` is equivalent to `upstream/staging...HEAD` and stands for the merge base between `upstream/staging` and `HEAD` (hence between `upstream/staging` and `upstream/master`). 231 232Then use the *Edit* button in the upper right corner of the GitHub PR, and switch the base branch from `master` to `staging`. 233*After* the PR has been retargeted, a final rebase onto the target branch might be needed to resolve merge conflicts. 234 235```console 236# Rebase onto target branch 237git rebase upstream/staging 238# Review and fixup possible conflicts 239git status 240# Force push your changes 241git push origin feature --force-with-lease 242``` 243 244## How to backport pull requests 245[pr-backport]: #how-to-backport-pull-requests 246 247Once a PR has been merged, a backport to the corresponding `release-YY.MM` branch can be created. 248 249### Automatically backporting changes 250 251> [!Note] 252> You have to be a [Nixpkgs maintainer](./maintainers) to automatically create a backport pull request. 253 254Add the [`backport release-YY.MM` label](https://github.com/NixOS/nixpkgs/labels?q=backport) to the PR on the `master` branch. 255This will cause [a GitHub Action](.github/workflows/backport.yml) to open a new PR to the `release-YY.MM` branch a few minutes later. 256This can be done on both open or already merged pull requests. 257 258### Manually backporting changes 259 260To manually create a backport, follow [the standard pull request process][pr-create], but: 261 262- Use `release-YY.MM` for the base branch, both for the local branch and the pull request. 263 264> [!Warning] 265> Do not use the `nixos-YY.MM` branch. 266> It points to the latest _tested_ release channel commit. 267 268- Instead of manually making and committing the changes, use [`git cherry-pick -x`](https://git-scm.com/docs/git-cherry-pick) for each commit. 269 Use `git cherry-pick -x <commit>` when the reason is obvious, for example for minor version bumps and fixes. 270 Otherwise, use `git cherry-pick -xe <commit>` to add a reason for the backport. 271 Here is [an example](https://github.com/nixos/nixpkgs/commit/5688c39af5a6c5f3d646343443683da880eaefb8). 272 273> [!Warning] 274> Ensure the commits exist on the `master` branch. 275> In the case of squashed or rebased merges, the commit hash will change and the new commits can be found in the merge message at the bottom of the `master` pull request. 276 277- In the pull request description, link to the original pull request to `master`. 278 The pull request title should include `[YY.MM]` matching the release you're backporting to. 279 280## How to review pull requests 281[pr-review]: #how-to-review-pull-requests 282 283The Nixpkgs project receives a high number of pull requests. 284Anyone may review and approve PRs and it is an important contribution to the project. 285 286The high change rate makes any PR that remains open for too long subject to merge conflicts. 287To avoid extra work, reviewing PRs timely and being responsive is key. 288GitHub provides sort filters to see the [most recently updated](https://github.com/NixOS/nixpkgs/pulls?q=is%3Apr+is%3Aopen+sort%3Aupdated-desc) pull requests. 289We highly encourage looking at [this list of ready to merge, unreviewed pull requests](https://github.com/NixOS/nixpkgs/pulls?q=is%3Apr+is%3Aopen+review%3Anone+status%3Asuccess+no%3Aproject+no%3Aassignee+no%3Amilestone). 290 291Controversial changes can lead to controversial opinions, but it is important to respect every community member and their work. 292Always be nice and polite. 293 294GitHub provides reactions for quick feedback to pull requests or comments. 295The thumb-down reaction should be used with care and, if possible, accompanied with explanation for the submitter to improve their contribution. 296 297When doing a review: 298- Aim to drive the proposal to a timely conclusion. 299- Focus on the proposed changes and keep the scope narrow. 300- Help the contributor prioritise their efforts towards getting their change merged. 301 302If you find anything related that could be improved but is not immediately required for acceptance, consider: 303- Implementing the changes yourself in a follow-up pull request, 304- Tracking your idea in an issue, 305- Offering to review a follow-up pull request, 306- Making concrete [suggestions](https://docs.github.com/en/pull-requests/collaborating-with-pull-requests/reviewing-changes-in-pull-requests/incorporating-feedback-in-your-pull-request) in the same pull request. 307 308For example, follow-up changes could involve refactoring code in the affected files. 309 310But please remember not to make such additional considerations a blocker, and communicate that to the contributor, for example by following the [conventional comments](https://conventionalcomments.org) pattern. 311If the related change is essential for the contribution at hand, make clear why you think it is important to address that first. 312 313Pull request reviews should include a list of what has been reviewed in a comment, so other reviewers and mergers can know the state of the review. 314 315All the review templates provided are generic examples. 316Their usage is optional and the reviewer is free to adapt them. 317 318To get more information about how to review specific parts of Nixpkgs, refer to the documents linked to in the [overview section][overview]. 319 320If a pull request contains documentation changes that might require feedback from the documentation team, ping [@NixOS/documentation-team](https://github.com/orgs/nixos/teams/documentation-team) on the pull request. 321 322If you have enough knowledge and experience in a topic and would like to be a long-term reviewer for related submissions, please contact the current reviewers for that topic. 323The main reviewers for a topic can be hard to find as there is no list, but checking past pull requests or git-blaming the code can give some hints. 324 325## How to merge pull requests yourself 326[pr-merge]: #how-to-merge-pull-requests 327 328You can invoke the nixpkgs-merge-bot by commenting `@NixOS/nixpkgs-merge-bot merge`. 329The bot will verify the following conditions, refusing to merge otherwise: 330 331- the PR author should be @r-ryantm or a Nixpkgs committer; 332- the invoker should be among the package maintainers; 333- the package should reside in `pkgs/by-name`. 334 335Further, nixpkgs-merge-bot will ensure all CI checks and the ofborg builds for Linux have successfully completed before merging the pull request. 336Should the checks still be underway, the bot will wait for them to finish before attempting the merge again. 337 338For other pull requests, please see [I opened a PR, how do I get it merged?](#i-opened-a-pr-how-do-i-get-it-merged). 339 340In case the PR is stuck waiting for the author to apply a trivial change and the author allowed members to modify the PR, consider applying it yourself. 341You should pay extra attention to make sure the addition doesn't go against the idea of the original PR and would not be opposed by the author. 342 343Please see the discussion in [GitHub nixpkgs issue #321665](https://github.com/NixOS/nixpkgs/issues/321665) for information on how to proceed to be granted this level of access. 344 345As a maintainer, when you leave the Nix community, please create an issue or post on [Discourse](https://discourse.nixos.org) with references to the packages and modules you maintained, so they can be taken over by other contributors. 346 347# Flow of merged pull requests 348 349After a pull request is merged, it eventually makes it to [Hydra](https://hydra.nixos.org). 350Hydra regularly evaluates and builds Nixpkgs, updating [the official channels](https://channels.nixos.org) when their jobs succeed. 351See [Nix Channel Status](https://status.nixos.org) for the current channel states. 352 353Our primary development branches and their related channels are: 354 355- `master`: The main branch, used for the unstable channels `nixos-unstable`, `nixos-unstable-small` and `nixpkgs-unstable`. 356- `release-YY.MM`: The release branches, used for the stable channels `nixos-YY.MM`, `nixos-YY.MM-small` and `nixpkgs-YY.MM-darwin`. 357 358When a channel is updated, its corresponding branch is also updated to the same commit. 359Example: The [`nixpkgs-unstable` branch](https://github.com/nixos/nixpkgs/tree/nixpkgs-unstable) corresponds to the commit from the [`nixpkgs-unstable` channel](https://channels.nixos.org/nixpkgs-unstable). 360 361Nixpkgs is tied to the NixOS release process, which is documented in the [NixOS Release Wiki](https://nixos.github.io/release-wiki/). 362 363See [this section][branch] to know when to use the release branches. 364 365## Staging 366[staging]: #staging 367 368The staging workflow exists to batch Hydra builds of many packages together. 369It is coordinated in the [Staging room](https://matrix.to/#/#staging:nixos.org) on Matrix. 370 371It works by directing commits that cause [mass rebuilds][mass-rebuild] to a separate `staging` branch that isn't directly built by Hydra. 372Regularly, the `staging` branch is _manually_ merged into a `staging-next` branch to be built by Hydra using the [`nixpkgs:staging-next` jobset](https://hydra.nixos.org/jobset/nixpkgs/staging-next). 373The `staging-next` branch should then only receive changes that fix Hydra builds; **for anything else, ask the [Staging room](https://matrix.to/#/#staging:nixos.org) first**. 374Once it is verified that there are no major regressions, `staging-next` is merged into `master` using [a pull request](https://github.com/NixOS/nixpkgs/issues?q=label%3A%224.workflow%3A+staging%22). 375This is done manually to ensure it's a good use of Hydra's computing resources. 376Since `staging-next` is separate from `staging`, you may merge changes into `staging` at any time. 377 378In order for the `staging` and `staging-next` branches to be up-to-date with the latest commits on `master`, there are regular _automated_ merges from `master` into `staging-next`, and from `staging-next` into `staging`. 379This is implemented using GitHub workflows [here](.github/workflows/periodic-merge-6h.yml) and [here](.github/workflows/periodic-merge-24h.yml). 380 381> [!Note] 382> Changes must be well tested before being merged into any branch. 383> Hydra builds should not be used as a testing platform. 384 385Here is a Git history diagram showing the flow of commits between the three branches: 386```mermaid 387%%{init: { 388 'theme': 'base', 389 'themeVariables': { 390 'gitInv0': '#ff0000', 391 'gitInv1': '#ff0000', 392 'git2': '#ff4444', 393 'commitLabelFontSize': '15px' 394 }, 395 'gitGraph': { 396 'showCommitLabel':true, 397 'mainBranchName': 'master', 398 'rotateCommitLabel': true 399 } 400} }%% 401gitGraph 402 commit id:" " 403 branch staging 404 commit id:" " 405 branch staging-next 406 407 merge master id:"automatic" 408 checkout staging 409 merge staging-next id:"automatic " 410 411 checkout staging-next 412 merge staging type:HIGHLIGHT id:"manual" 413 commit id:"fixup" 414 415 checkout master 416 checkout staging 417 checkout master 418 commit id:" " 419 checkout staging-next 420 merge master id:"automatic " 421 checkout staging 422 merge staging-next id:"automatic " 423 424 checkout staging-next 425 commit id:"fixup " 426 checkout master 427 merge staging-next type:HIGHLIGHT id:"manual (PR)" 428``` 429 430 431Here's an overview of the different branches: 432 433| branch | `master` | `staging-next` | `staging` | 434| --- | --- | --- | --- | 435| Used for development | ✔️ | ❌ | ✔️ | 436| Built by Hydra | ✔️ | ✔️ | ❌ | 437| [Mass rebuilds][mass-rebuild] | ❌ | ⚠️ Only to fix Hydra builds | ✔️ | 438| Critical security fixes | ✔️ for non-mass-rebuilds | ✔️ for mass-rebuilds | ❌ | 439| Automatically merged into | `staging-next` | `staging` | - | 440| Manually merged into | - | `master` | `staging-next` | 441 442The staging workflow is used for all stable branches with corresponding names: 443- `master`/`release-YY.MM` 444- `staging`/`staging-YY.MM` 445- `staging-next`/`staging-next-YY.MM` 446 447# Conventions 448 449## Branch conventions 450<!-- This section is relevant to both contributors and reviewers --> 451[branch]: #branch-conventions 452 453Most changes should go to `master`, but sometimes other branches should be used instead. 454Use the following decision process to figure out the right branch: 455 456Is the change [acceptable for releases][release-acceptable] and do you wish to have the change in the release? 457- No: Use the `master` branch, do not backport the pull request. 458- Yes: Can the change be implemented the same way on the `master` and release branches? 459 For example, a package's major version might differ between the `master` and release branches, such that separate security patches are required. 460 - Yes: Use the `master` branch and [backport the pull request](#how-to-backport-pull-requests). 461 - No: Create separate pull requests to the `master` and `release-YY.MM` branches. 462 463If the change causes a [mass rebuild][mass-rebuild], use the `staging` branch instead: 464- Mass rebuilds to `master` should go to `staging` instead. 465- Mass rebuilds to `release-YY.MM` should go to `staging-YY.MM` instead. 466 467See [this section][staging] for how such changes propagate between the branches. 468 469### Changes acceptable for releases 470[release-acceptable]: #changes-acceptable-for-releases 471 472Only changes to _supported_ releases may be accepted. 473The oldest supported release (`YYMM`) can be found using 474``` 475nix-instantiate --eval -A lib.trivial.oldestSupportedRelease 476``` 477 478The release branches should generally only receive backwards-compatible changes, both for the Nix expressions and derivations. 479Here are some examples of changes that are okay to backport: 480- ✔️ New packages, modules and functions 481- ✔️ Security fixes 482- ✔️ Package version updates 483 - ✔️ Patch versions with fixes 484 - ✔️ Minor versions with new functionality, but no breaking changes 485 486In addition, major package version updates with breaking changes are also acceptable for: 487- ✔️ Services that would fail without up-to-date client software, such as `spotify`, `steam`, and `discord` 488- ✔️ Security critical applications, such as `firefox` and `chromium` 489 490### Changes causing mass rebuilds 491[mass-rebuild]: #changes-causing-mass-rebuilds 492 493Which changes cause mass rebuilds is not formally defined. 494In order to help the decision, CI automatically assigns [`rebuild` labels](https://github.com/NixOS/nixpkgs/labels?q=rebuild) to pull requests based on the number of packages they cause rebuilds for. 495As a rule of thumb, if the number of rebuilds is **500 or more**, consider targeting the `staging` branch instead of `master`; if the number is **1000 or more**, the pull request causes a mass rebuild, and should target the `staging` branch. 496See [previously merged pull requests to the staging branches](https://github.com/NixOS/nixpkgs/issues?q=base%3Astaging+-base%3Astaging-next+is%3Amerged) to get a sense for what changes are considered mass rebuilds. 497 498## Commit conventions 499[commit-conventions]: #commit-conventions 500 501- Create one commit for each logical unit. 502 503- If you have commits `pkg-name: oh, forgot to insert whitespace`: squash commits in this case. 504 Use `git rebase -i`. 505 See [Squashing Commits](https://git-scm.com/book/en/v2/Git-Tools-Rewriting-History#_squashing) for additional information. 506 507- For consistency, there should not be a period at the end of the commit message's summary line (the first line of the commit message). 508 509- When adding yourself to `maintainer-list.nix`, make a separate commit with the message `maintainers: add <handle>`. 510 Add the commit before those making changes to the package or module. 511 See [Nixpkgs Maintainers](./maintainers/README.md) for details. 512 513- Make sure you read about any commit conventions specific to the area you're touching. 514 See: 515 - [Commit conventions](./doc/README.md#commit-conventions) for changes to `doc`, the Nixpkgs manual. 516 - [Commit conventions](./lib/README.md#commit-conventions) for changes to `lib`. 517 - [Commit conventions](./nixos/README.md#commit-conventions) for changes to `nixos`. 518 - [Commit conventions](./pkgs/README.md#commit-conventions) for changes to `pkgs`. 519 520### Writing good commit messages 521[writing-good-commit-messages]: #writing-good-commit-messages 522 523It's important to include relevant information in the *commit message*, so others can later understand *why* a change was made. 524While this potentially can be understood by reading code, PR discussion or upstream changes, doing so often requires a lot of work. 525 526Simple package version updates need to include the attribute name, old and new versions, as well as a reference to the release notes or changelog. 527Package upgrades with more extensive changes require more verbose commit messages. 528 529## Review and Merge conventions 530 531Comments on Pull Requests are considered non-blocking by default. 532Every blocking comment must be explicitly marked as such by using GitHub's "Request Changes" review type. 533A reviewer who submits a blocking review should be available for discussion and re-review. 534An abandoned review may be dismissed after reasonable time was given at the discretion of the merger. 535 536All suggestions for change, blocking or not, should be acknowledged before merge. 537This can happen implicitly by applying the suggestion, or explicitly by rejecting it. 538 539To make changes on commit structure and commit messages or apply simple suggestions, committers are encouraged to [checkout the PR](https://cli.github.com/manual/gh_pr_checkout) and push directly to the contributor's branch before merging. 540Committers will carefully weigh the cost of another review cycle against the feelings of the contributor when pushing to their branch. 541They should also transparently communicate which changes they made. 542If a contributor does not want committers to push to their branch, they must uncheck the "Allow edits and access to secrets by maintainers" box explicitly. 543 544> [!WARNING] 545> Committers: Branches created via `gh pr checkout` can't be pushed with `--force-with-lease`, so do a sanity check before pushing. 546 547## Code conventions 548[code-conventions]: #code-conventions 549 550### Release notes 551 552If you removed packages or made some major NixOS changes, write about it in the next release notes in [`nixos/doc/manual/release-notes`](./nixos/doc/manual/release-notes). 553 554### File naming and organisation 555 556Names of files and directories should be in lowercase, with dashes between words — kebab case, not camel case. 557For instance, it should be `all-packages.nix`, not `allPackages.nix` or `AllPackages.nix`. 558 559### Formatting 560 561CI [enforces](./.github/workflows/lint.yml) all Nix files to be formatted using the [official Nix formatter](https://github.com/NixOS/nixfmt). 562 563You can ensure this locally using either of these commands: 564``` 565nix-shell --run treefmt 566nix develop --command treefmt 567nix fmt 568``` 569 570If you're starting your editor in `nix-shell` or `nix develop`, you can also set it up to automatically run `treefmt` on save. 571 572If you have any problems with formatting, please ping the [formatting team](https://nixos.org/community/teams/formatting/) via [@NixOS/nix-formatting](https://github.com/orgs/NixOS/teams/nix-formatting). 573 574### Syntax 575 576- Set up [editorconfig](https://editorconfig.org) for your editor, such that [the settings](./.editorconfig) are automatically applied. 577 578- Use `lowerCamelCase` for variable names, not `UpperCamelCase`. 579 Note, this rule does not apply to package attribute names, which instead follow the rules in [package naming](./pkgs/README.md#package-naming). 580 581- Functions should list their expected arguments as precisely as possible. 582 That is, write 583 584 ```nix 585 { 586 stdenv, 587 fetchurl, 588 perl, 589 }: 590 <...> 591 ``` 592 593 instead of 594 595 ```nix 596 args: with args; <...> 597 ``` 598 599 or 600 601 ```nix 602 { 603 stdenv, 604 fetchurl, 605 perl, 606 ... 607 }: 608 <...> 609 ``` 610 611 For functions that are truly generic in the number of arguments, but have some required arguments, you should write them using an `@`-pattern: 612 613 ```nix 614 { 615 stdenv, 616 doCoverageAnalysis ? false, 617 ... 618 }@args: 619 620 stdenv.mkDerivation (args // { foo = if doCoverageAnalysis then "bla" else ""; }) 621 ``` 622 623 instead of 624 625 ```nix 626 args: 627 628 args.stdenv.mkDerivation ( 629 args 630 // { 631 foo = if args ? doCoverageAnalysis && args.doCoverageAnalysis then "bla" else ""; 632 } 633 ) 634 ``` 635 636- Unnecessary string conversions should be avoided. 637 Do 638 639 ```nix 640 { rev = version; } 641 ``` 642 643 instead of 644 645 ```nix 646 { rev = "${version}"; } 647 ``` 648 649- Building lists conditionally _should_ be done with `lib.optional(s)` instead of using `if cond then [ ... ] else null` or `if cond then [ ... ] else [ ]`. 650 651 ```nix 652 { buildInputs = lib.optional stdenv.hostPlatform.isDarwin iconv; } 653 ``` 654 655 instead of 656 657 ```nix 658 { buildInputs = if stdenv.hostPlatform.isDarwin then [ iconv ] else null; } 659 ``` 660 661 As an exception, an explicit conditional expression with null can be used when fixing a important bug without triggering a mass rebuild. 662 If this is done a follow up pull request _should_ be created to change the code to `lib.optional(s)`. 663 664- Any style choices not covered here but that can be expressed as general rules should be left at the discretion of the authors of changes and _not_ commented in reviews. 665 The purpose of this is: 666 - to avoid churn as contributors with different style preferences undo each other's changes, 667 - to ensure that style rules are written down and consistent (and can thus be followed when authoring changes, reducing review cycles), 668 - and to encourage reviews to focus on more impactful considerations. 669 670# Practical contributing advice 671 672To contribute effectively and efficiently, you need to be aware of how the process generally works. 673This section aims to document the process as we live it in Nixpkgs to set the right expectations and give practical tips on how to work with it. 674 675## I opened a PR, how do I get it merged? 676[i-opened-a-pr-how-do-i-get-it-merged]:#i-opened-a-pr-how-do-i-get-it-merged 677 678In order for your PR to be merged, a committer needs to review and merge it. 679Because committers are mostly independent, unpaid volunteers, this can take time. 680It is entirely normal for your PR to sit around without any feedback for days, weeks or sometimes even months. 681We strive to avoid this, but the reality is that it happens frequently. 682Even when you get feedback, follow-ups may take just as long. 683Don't be intimidated and kindly ask for feedback again every so often. 684If your change is good, it will eventually be merged. 685 686You can often speed up the process by understanding the committer's perspective and preparing your PR with reviewing in mind. 687 688### The committer's perspective 689 690PRs have varying quality and even the best people make mistakes. 691Committers need to assess whether a PR's changes are good or not. 692To merge, at least one committer has to be confident about its quality. 693 694Committers typically assess three aspects: 695 6961. Whether the change's intention is necessary and desirable. 6972. Whether the code quality of your changes is good. 6983. Whether the produced artifacts are good. 699 700To get your PR merged quickly and smoothly, you should help convince committers in these aspects. 701 702### How to help committers assess your PR 703 704It's best to explain *why* you've made your change, because guessing the intention is not always possible. 705This does not apply to trivial changes like version updates, because the intention is obvious. 706For more nuanced changes or even major version upgrades, it helps if you explain the background behind your change. 707For example, if you're adding a package, explain what it is and why it should be in Nixpkgs. 708This goes hand in hand with [Writing good commit messages](#writing-good-commit-messages). 709 710To show the quality of your code, you should focus on making it *reviewable*. 711First, take a look at your code changes yourself and try to put yourself into the shoes of someone who didn't just write that code. 712Would you immediately know what the code does or why it is needed by glancing at it? 713If not, reviewers will notice this and will ask you to clarify the code by refactoring it and/or adding code comments. 714Doing this preemptively can save a lot of time. 715Doing multiple unrelated changes in a single commit can become hard to review quickly. 716Thus, consider multiple atomic commits to tell the story of your change. 717There is a balance to strike however: over-fragmentation causes friction. 718 719The artifacts are the hardest to assess because PRs touch all sorts of components: applications, libraries, NixOS modules, editor plugins and many other things. 720Any individual committer can only really assess components that they themselves know how to use. 721Yet, they must still be convinced somehow. 722There isn't a good generic solution to this but there are some ways to ease it: 723 724- Provide smoke tests that can be run without much research or setup. 725 726 Committers usually don't have the time or interest to learn how your component works and how they could test its functionality. 727 Try to provide a quick guide on how to use it in a meaningful way or a ready-made command that demonstrates that it works as expected. 728 The committer can use this to convince themselves that your change is good. 729 If it can be automated, you could even turn this into an automated NixOS test which reviewers could simply run. 730 731- Invite other users of the component to try it out and report their findings. 732 733 Seeing other users testing the changes and having it work for them can convince committers, too. 734 735- Describe what you have done to test your PR. 736 737 It also helps, if you can additionally show that you have done sufficient quality assurance on your changes. 738 739- Become a maintainer of the component. 740 741 Listed maintainers generally receive more trust when it comes to changes to their maintained components. 742 743Even if you adhere to all of these recommendations, it is still quite possible for your PR to be forgotten or abandoned by any given committer. 744Please remain mindful of them doing this work on their own volition and unpaid in their free time and therefore [owing you nothing](https://mikemcquaid.com/open-source-maintainers-owe-you-nothing/). 745Causing a stink in such a situation is a surefire way to get any other potential committer to not want to look at your PR either. 746Ask them nicely whether they still intend to review your PR and find yourself another committer to look at your PR if not. 747 748### How can I get a committer to look at my PR? 749 750- Improve skimmability: use a simple descriptive PR title outlining _what_ is done and _why_. 751 Details go in commit messages. 752- Improve discoverability: apply all relevant labels, tick all relevant PR body checkboxes. 753- Wait. 754 Reviewers frequently browse open PRs and may happen to run across yours and take a look. 755- Get non-committers to review/approve. 756 Many committers filter open PRs for low-hanging fruit that have already been reviewed. 757- [@-mention](https://github.blog/news-insights/mention-somebody-they-re-notified/) someone and ask them nicely. 758- Post in one of the channels made for this purpose if there has been no activity for at least one week: 759 - The current "PRs ready for review" or "PRs already reviewed" threads in the [NixOS Discourse](https://discourse.nixos.org/c/dev/14). 760 - The [Nixpkgs Review Requests Matrix room](https://matrix.to/#/#review-requests:nixos.org). 761 - Similar threads/rooms in unofficial NixOS spaces, such as Discord. 762 763### CI failed or got stuck on my PR, what do I do? 764 765First, ensure that the failure is actually related to your change. 766Sometimes, the CI system simply has a hiccup or the check was broken by someone else before. 767Read through the error message; it's usually quite easy to tell whether it is caused by changes to the component you touched. 768If it is indeed caused by your change, try to fix it. 769Don't be afraid of asking for advice if you're uncertain how to do that, others might have fixed such issues already and can help you out. 770Your PR will not be merged while CI is still failing. 771 772ofborg builds can often get stuck, particularly in PRs targeting `staging` and in builders for the Darwin platform. 773Reviewers will know how to handle them or when to ignore them. 774Don't worry about it. 775However, if there is a build failure and it was caused by your change, you need to investigate it. 776If ofborg reveals the build to be broken on a platform that you don't have access to, consider setting your package's `meta.broken`, `meta.badPlatforms` or `meta.platforms` accordingly. 777 778When in any doubt, please ask via comments or through one of the help channels. 779 780## I received a review, how do I get it over the finish line? 781 782Most likely, a reviewer wants you to change a few things or requires further input. 783 784A reviewer may have taken a look at the code and it looked good to them ("Diff LGTM"), but they still need to be convinced of the artifact's quality. 785They might also be waiting on input from other users or maintainers on whether the intention and direction of your PR makes sense. 786If you know of people who could help clarify any of this, please bring the PR to their attention. 787The current state of the PR is frequently not clearly communicated, so please don't hesitate to ask about it if it's unclear to you. 788 789It's also possible for the reviewer to not be convinced that your PR is necessary or that the method you've chosen is the right one. 790 791Please explain your intentions and reasoning to the committer in such a case. 792There may be constraints you had to work with which they're not aware of or qualities of your approach that they didn't immediately notice. 793If these weren't clear to the reviewer, that's a good sign you should explain them in your commit message or code comments! 794 795There are some further pitfalls and realities to be aware of: 796 797### Aim to reduce cycles 798 799Be prepared for it to take a while for the reviewer to get back to you after you respond. 800This is simply the reality of projects at the scale of Nixpkgs. 801As such, make sure to respond to _all_ feedback at once. 802It wastes everyone's time to wait for a couple of days just to have the reviewer need to remind you to address something they asked for. 803 804### A reviewer requested a bunch of insubstantial changes 805 806The people involved in Nixpkgs care about code quality. 807Once in Nixpkgs, the code needs to be maintained for many years to come. 808Therefore, you will likely be asked to do something different or adhere to a standard. 809 810Sometimes however, they also care a bit too much and may ask you to adhere to a personal preference of theirs. 811It's not always easy to tell whether or not the requested changes must be addressed. 812Sometimes, another reviewer may even have a _conflicting_ opinion on some points. 813 814It is convention to mark review comments that are not required to merge as nitpicks, but this is not always followed. 815As the author, you should still take a look at these, as they will often reveal best practices and unwritten rules. 816Those usually have good reasons behind them and you may want to pick them up as well. 817 818Please keep in mind that reviewers always mean well. 819Their intent is not to denounce your code, they want your code to be as good as it can be. 820Through their experience, they may also take notice of a seemingly insignificant issue that has caused problems before. 821 822Sometimes however, they can also get a bit carried away and become too perfectionistic. 823If you feel some of the requests are unreasonable, out of scope, or merely a matter of personal preference, try to nicely ask the reviewers whether these requests are *critical* to the PR's success. 824 825While we do have a set of [official standards for the Nix community](https://github.com/NixOS/rfcs), we don't have standards for everything and there are often multiple valid ways to achieve the same goal. 826Unless there are standards forbidding the patterns used in your code or there are serious technical, maintainability or readability issues with your code, you can disregard these requests. 827Please communicate this clearly though; a simple "I prefer it this way and see no major issue maintaining it" can save a lot of arguing. 828 829If you are unsure about some change requests, please ask reviewers *why* they requested them. 830This will usually reveal how important they deem it to be and will help educate you about standards, best practices, unwritten rules as well as preferences people have and why. 831 832Some committers have stronger opinions on some things and may not want to merge your PR if you don't follow their requests. 833It is totally fine to get yourself a second or third opinion in such a case. 834 835### Committers work on a push-basis 836 837It's possible for you to get a review but nothing happens afterwards, even if you respond to review comments. 838A committer not following up on your PR does not necessarily mean they're disinterested, they may have simply had other circumstances preventing them from doing so. 839 840Committers typically handle many PRs at the same time and it is not realistic for them to keep up with all of them immediately. 841If someone approved and didn't merge a few days later, they most likely just forgot. 842 843Please see it as your responsibility to actively remind reviewers of your open PRs. 844 845The easiest way to do so is to notify them via GitHub. 846Github notifies people involved, whenever you add a comment or push to your PR or re-request their review. 847Doing any of that will get their attention again. 848Everyone deserves proper attention, and yes, that includes you! 849However, please be mindful that committers can sadly not always give everyone the attention they deserve. 850 851It may very well be the case that you have to do this every time you need the committer to follow up upon your PR. 852Again, this is a community project so please be mindful of people's circumstances here; be nice when requesting reviews again. 853 854It may also be the case that the committer has lost interest or isn't familiar enough with the component you're touching to be comfortable to merge. 855They will likely not immediately state that fact, so please ask for clarification and don't hesitate to find yourself another committer to take a look. 856 857### Nothing helped 858 859If you followed these guidelines but still got no results or if you feel that you have been wronged, please explicitly reach out to the greater community. 860 861The [NixOS Discourse](https://discourse.nixos.org) is a great place to do this, as it has historically been the asynchronous medium with the greatest concentration of committers and other people who are involved in Nixpkgs. 862There is a dedicated discourse thread [PRs in distress](https://discourse.nixos.org/t/prs-in-distress/3604) where you can link your PR, if everything else fails. 863The [Nixpkgs / NixOS contributions Matrix channel](https://matrix.to/#/#dev:nixos.org) is the best synchronous channel with the same qualities. 864 865Please reserve these for cases where you've made a serious effort in trying to get the attention of multiple active committers and provided realistic means for them to assess your PR's quality. 866As mentioned previously, it is unfortunately perfectly normal for a PR to sit around for weeks. 867 868Please don't blow up situations where progress is happening but is merely not going fast enough for your tastes. 869Honking in a traffic jam will not make you go any faster.