• Stars
    star
    166
  • Rank 227,748 (Top 5 %)
  • Language
    Python
  • License
    MIT License
  • Created over 4 years ago
  • Updated 2 months ago

Reviews

There are no reviews yet. Be the first to send feedback to the community and the maintainers!

Repository Details

Use Poetry inside Nox sessions

nox-poetry

PyPI Python Version License

Read the documentation at https://nox-poetry.readthedocs.io/ Tests Codecov

Use Poetry inside Nox sessions

This package provides a drop-in replacement for the nox.session decorator, and for the nox.Session object passed to user-defined session functions. This enables session.install to install packages at the versions specified in the Poetry lock file.

from nox_poetry import session

@session(python=["3.10", "3.9"])
def tests(session):
    session.install("pytest", ".")
    session.run("pytest")

Disclaimer: This project is not affiliated with Nox, and not an official Nox plugin.

Installation

Install nox-poetry from the Python Package Index:

$ pip install nox-poetry

Important: This package must be installed into the same environment that Nox is run from. If you installed Nox using pipx, use the following command to install this package into the same environment:

$ pipx inject nox nox-poetry

Requirements

  • Python 3.7+
  • Poetry >= 1.0.0

You need to have a Poetry installation on your system. nox-poetry uses Poetry via its command-line interface.

Usage

Import the @session decorator from nox_poetry instead of nox. There is nothing else you need to do. The session.install method automatically honors the Poetry lock file when installing dependencies. This allows you to manage packages used in Nox sessions as development dependencies in Poetry.

This works because session functions are passed instances of nox_poetry.Session, a proxy for nox.Session adding Poetry-related functionality. Behind the scenes, nox-poetry uses Poetry to export a constraints file and build the package.

For more fine-grained control, additional utilities are available under the session.poetry attribute:

  • session.poetry.installroot(distribution_format=["wheel"|"sdist"])
  • session.poetry.build_package(distribution_format=["wheel"|"sdist"])
  • session.poetry.export_requirements()

Note that distribution_format is a keyword-only parameter.

Here is a comparison of the different installation methods:

  • Use session.install(...) to install specific development dependencies, e.g. session.install("pytest").
  • Use session.install(".") (or session.poetry.installroot()) to install your own package.
  • Use session.run_always("poetry", "install", external=True) to install your package with all development dependencies.

Please read the next section for the tradeoffs of each method.

Why?

Let's look at an example:

from nox_poetry import session

@session(python=["3.10", "3.9"])
def tests(session):
    session.install("pytest", ".")
    session.run("pytest")

This session performs the following steps:

  • Build a wheel from the local package.
  • Install the wheel as well as the pytest package.
  • Invoke pytest to run the test suite against the installation.

Consider what would happen in this session if we had imported @session from nox instead of nox_poetry:

  • Package dependencies would only be constrained by the wheel metadata, not by the lock file. In other words, their versions would not be pinned.
  • The pytest dependency would not be constrained at all.
  • Poetry would be installed as a build backend every time.

Unpinned dependencies mean that your checks are not reproducible and deterministic, which can lead to surprises in Continuous Integration and when collaborating with others. You can solve these issues by declaring pytest as a development dependency, and installing your package and its dependencies using poetry install:

@nox.session
def tests(session: Session) -> None:
    """Run the test suite."""
    session.run_always("poetry", "install", external=True)
    session.run("pytest")

Unfortunately, this approach comes with its own set of problems:

  • Checks run against an editable installation of your package, i.e. your local copy of the code, instead of the installed wheel your users see. In the best case, any mistakes will still be caught during Continuous Integration. In the worst case, you publish a buggy release because you forgot to commit some changes.
  • The package is installed, as well as all of its core and development dependencies, no matter which tools a session actually runs. Code formatters or linters, for example, don't need your package installed at all. Besides being wasteful, it goes against the idea of running checks in isolated environments.

nox-poetry uses a third approach:

  • Installations are performed by pip, via the session.install method.
  • When installing your own package, Poetry is used to build a wheel, which is passed to pip.
  • When installing third-party packages, Poetry is used to export a constraints file, which is passed to pip along with the packages. The constraints file ensures that package versions are pinned by the lock file, without forcing an installation of every listed dependency and sub-dependency.

In summary, this approach brings the following advantages:

  • You can manage tools like pytest as development dependencies in Poetry.
  • Dependencies are pinned by Poetry's lock file, making checks predictable and deterministic.
  • You can run checks against an installed wheel, instead of your local copy of the code.
  • Every tool can run in an isolated environment with minimal dependencies.
  • No need to install your package with all its dependencies if all you need is some linter.

Contributing

Contributions are very welcome. To learn more, see the Contributor Guide.

License

Distributed under the terms of the MIT license, nox-poetry is free and open source software.

Issues

If you encounter any problems, please file an issue along with a detailed description.

Credits

This project was generated from @cjolowicz's Hypermodern Python Cookiecutter template.

More Repositories

1

cookiecutter-hypermodern-python

Hypermodern Python Cookiecutter
Python
1,780
star
2

hypermodern-python

Hypermodern Python
Python
556
star
3

retrocookie

Retrocookie updates Cookiecutter templates with changes from generated projects
Python
30
star
4

docker-incremental-build-example

Incremental Docker builds for monolithic code bases
CMake
16
star
5

cookiecutter-hypermodern-python-instance

Instance of the Hypermodern Python Cookiecutter
Python
11
star
6

sc101-nbd

This software allows Linux to access the Netgear Storage Central 101 (sc101) as a network block device.
C
7
star
7

poetry-up

Command-line tool for upgrading Python dependencies using Poetry.
Python
6
star
8

lojban-etymology

Lojban Etymology
Python
5
star
9

cutty

An experimental Cookiecutter clone ๐Ÿšง
Python
5
star
10

cjolowicz.github.io

HTML
5
star
11

poetry-merge-lock

Merge the lock file of a Poetry project
Python
4
star
12

scripts

My scripts
Shell
4
star
13

lvm-duplicity

Backup logical volumes using duplicity.
4
star
14

iso9

ISO 9:1995 transliteration for Cyrillic text
Python
4
star
15

docker-buildbot

Docker image for buildbot
Shell
3
star
16

blog

Hugo source for my blog
Shell
2
star
17

docker-buildbot-worker

Docker images for buildbot worker
Dockerfile
2
star
18

covid19

Simulating COVID-19 transmission for Germany
Python
2
star
19

lazysequence

Make iterators look like immutable lists
Python
2
star
20

docker-centos5

Docker Images for Centos 5 with OpenSSL 1.1.0j
M4
1
star
21

lojban-alice-in-wonderland

The Lojban translation of โ€œAlice in Wonderlandโ€, in alternate orthography
1
star
22

hadiya

Hadiya is a web application for exchanging things for free.
Ruby
1
star
23

cjolowicz

1
star
24

railsdemo

Ruby on Rails Tutorial - Demo application
Ruby
1
star
25

railssample

Ruby on Rails tutorial - Sample application
Ruby
1
star
26

lojban-cmavo-table

Lojban cmavo tables
Python
1
star
27

microblog

The example project from Miguel Grinberg's Flask Mega Tutorial
Python
1
star
28

wordpress-configure

Configure a WordPress installation on Debian.
1
star
29

vmail-admin

Manage virtual mailboxes and aliases in a MySQL database.
Shell
1
star