Purpose

One of the main problems the Python community faces is complex packaging. Despite being one of the easier languages to get started and go to production with, Python packaging is rather convoluted. There’s not a lot of quality tutorials out there that cover building your own package using setup.py, and the ones that do exist leave out some critical information like the concept of sub-modules and how to import them.

Fortunately, there is an easier solution to all of this! It’s called Poetry and it makes building Python packages incredibly easy.

Introducing Poetry

If you’re a web developer or somebody that works with modern Javascript a lot, you’re probably familiar with npm. npm is the Node Package Manager - it lets you manage your project and the dependencies for it. It does this by installing the packages you specify into the node_modules directory in your project, so that each project gets its own clean copy of the dependencies it uses.

In Python, the package manager is called pip. pip…installs packages. That’s about it. Coming from JS dev, it isn’t immediately obvious how to create a new Python project and add dependencies to it. Conventional wisdom is to create a setup.py file and new “virtual environment” using the virtualenv tool. That creates an isolated environment for your Python project and stores the dependencies there without having to mess with your system Python environment.

With Poetry, things look a lot more like using npm (or cargo, for Rust developers) - it’s one tool that manages your project, your virtual environment, and your dependencies. In this post, I aim to teach you how to use Poetry effectively.

A quick note on virtual environments

Virtual environments still remain the most effective way to isolate project dependencies from your system’s Python packages. For that reason, Poetry (under the hood) does still use virtual environments to isolate your dependencies. Poetry handles them transparently and automatically for you. They’re placed in $HOME/.cache/pypoetry/virtualenvs by default, if you ever want to explore them.

poetry run will run a command or Python script inside of this virtualenv, poetry add adds dependencies inside of it, and so on. It’s a huge quality of life improvement over manually juggling them.

Installation

Installing Poetry is as simple as running


$ pip install --user poetry

in your terminal. After that, you should have the poetry command available to you. If you don’t, you may need to add $HOME/.local/bin to your PATH.


export PATH="$PATH:$HOME/.local/bin"

Starting a new project

Ready for this one?


$ poetry new project-name

Revolutionary. For the purposes of this article, we’ll call our project first-steps, which changes our command above to


$ poetry new first-steps
Created package first_steps in first-steps

Great! Let’s take a look at what’s in this directory.


$ cd first-steps && ls
first_steps  pyproject.toml  README.rst  tests

first_steps is the directory where all of our code is going to be stored, and the name of our package. If somebody installs this package, they refer to it in their code by that name. This typically looks like from first_steps import something.

pyproject.toml is the package configuration file. The vast majority of the time, you’re not going to be editing this manually.

Building our package

For simplicity’s sake, we’re going to be making a package that does exactly one thing and does it well: it increments a number. We’ll be providing both a CLI utility and a programmatic interface so that developers of all creeds can use our revolutionary module.

We’ll start with the programmatic interface! Create first_steps/increment.py and fill it out as follows


def increment(num: int):
  return num + 1

If you have no idea how to read that code, I would strongly recommend checking out my post on learning to code before this one.

Right now, users of your package would have to import this by typing from first_steps.increment import increment. That’s a bit too wordy for my tastes, so let’s go change that.


from .increment import increment

You can test this by dropping into your virtual environment’s REPL with poetry run python:


>>> from first_steps import increment
>>> increment(1)
2

Easy! Now all we need is a commandline interface. Let’s over-engineer the shit out of it so we can tour more of Poetry’s features.

Over-Engineering a CLI

click is an excellent package for building fully featured and complex commandline interfaces with minimal overhead and an easy learning curve. For this package, we do not need it at all. We’re going to use it anyways though, because we need to learn two things about working with Poetry:

  1. Managing dependencies
  2. Exporting a command from your package

Let’s start by initializing our project’s virtual environment and adding a dependency on click. Virtual environments are handled automatically and transparently with poetry, so there’s no need to worry about juggling them yourself.


$ poetry add click
Using version ^8.0.1 for click

Updating dependencies
Resolving dependencies... (0.2s)

Writing lock file

Package operations: 1 install, 0 updates, 0 removals

  • Installing click (8.0.1)

While we’re at it, let’s add the “Black” code formatter as a dev dependency. A dev dependency is a dependency that is only used when developing the package, and does not need to be shipped to the user.


$ poetry add --dev black
Using version ^21.5b1 for black

Updating dependencies
Resolving dependencies... (1.1s)

Writing lock file

Package operations: 6 installs, 0 updates, 0 removals

  • Installing appdirs (1.4.4)
  • Installing mypy-extensions (0.4.3)
  • Installing pathspec (0.8.1)
  • Installing regex (2021.4.4)
  • Installing toml (0.10.2)
  • Installing black (21.5b1)

Now we need to configure black for our project. We’re writing in modern (3.8+) Python here, so let’s tell it that we want that syntax. We can also increase maximum line length by a little bit while we’re here.

To do this, we’ll add a section to our pyproject.toml file. This section can be located anywhere within the file, but I like to put it at the bottom.


[tool.black]
line-length = 95
target-version = [ "py39",]

Now let’s make the file that will handle our user interaction.


import click
from .increment import increment


@click.command()
@click.argument("number", type=int)
def cli(number: int):
    click.echo(
        "{} has become {}!".format(
            click.style(number, bold=True),
            click.style(increment(number), bold=True, fg="green"),
        )
    )

Now we have a Python function registered as a command through click, and all we have to do is give it a snazzy name. We’ll call it inc because I’m not particularly creative.

To make sure that the user can run inc as a command when they install our package, we need to go back into our project configuration and register a script.


[tool.poetry.scripts]
inc = "first_steps.cli:cli"

You can test this out in your terminal now using poetry!


$ poetry run inc 5
5 has become 6!

Publishing our package

This is the easiest part. Simply run poetry publish --build to build and publish your package in one step. You will be prompted for your PyPi credentials, and then you can upload your package!


$ poetry publish --build
Building first-steps (0.1.0)
  - Building sdist
  - Built first-steps-0.1.0.tar.gz
  - Building wheel
  - Built first_steps-0.1.0-py3-none-any.whl

Publishing first-steps (0.1.0) to PyPI
 - Uploading first-steps-0.1.0.tar.gz 100%

The name first-steps is reserved by PyPi, so you’ll have to use your imagination on your next package.

Pretending it did go through and get published successfully, we could install this package like any other by running pip install first-steps.