It feels like—after three decades—the Python world is experiencing a renaissance, both in tooling and in libraries. It’s an exciting time to be a Python developer.
We’ve seen exciting libraries like Pydantic and FastAPI picking up steam, which is fantastic.
That said, there are also some smaller utilities that are gems in their own right. Some, like pytest-sugar, can be dropped into a project with minimal fuss. Others, like ptpython, can be used by an individual on a team without impacting the team.
Here are four can’t-live-without Python tools that I’ve recently added to my dev utility belt because they make me a faster developer (and they’re doggone fun to use):
4. ptpython
Website: https://github.com/prompt-toolkit/ptpython
Installation: python -m pip install ptpython
, Python 3.6+
Ptpython is what the built-in Python REPL ought to be. Just take a look at these features!
Syntax highlighting
data:image/s3,"s3://crabby-images/3e9f0/3e9f037a7706a4e09ad01b4bebdd9bb7b25a46a3" alt="Screenshot of the syntax highlighting in ptpython"
Autocompletion
data:image/s3,"s3://crabby-images/fe75f/fe75fc4f3148fe33d25673483380c88f17f112d0" alt="Screen capture of autocompleting the capitalize function on a string in ptpython"
Autoindentation
Watch ptpython automatically indent the body of this class; I didn’t have to manually add tabs or spaces!
data:image/s3,"s3://crabby-images/63986/639860c104533f3d5972527ba1fe4bb8d4ab3fe3" alt="Screen capture of automatic identation for a class block in ptpython"
Multiline editing
In this example, I forgot to include the age
attribute when I initially defined the class. I hit the up arrow once and now I can edit the entire multiline block of that particular class.
data:image/s3,"s3://crabby-images/e9e7b/e9e7bd823d6a9b27b0bad4d34f2ce73f994ab8ca" alt="Screen capture of hitting the up arrow once to edit an entire class block at once in ptpython"
Syntax errors
Ptpython will catch syntax errors before you execute the line, giving you a chance to go back and fix them before execution. And yes, it works with multiline blocks.
data:image/s3,"s3://crabby-images/94a3e/94a3ef2939c0ad13058451f91e5032a320ffc96f" alt="Screen capture of being able to fix a syntax error before executing a line in ptpython"
I haven’t even gotten into Emacs/Vi modes, IPython compatibility, displaying docstrings and function signatures, running shell commands without leaving the REPL, and host of other features. The next two tools are the peanut butter to ptpython’s chocolate, so install it and get ready for even more fun!
3. devtools, rich, and icecream
Website: https://python-devtools.helpmanual.io/
Installation: python -m pip install devtools
, Python 3.7+
Website: https://github.com/Textualize/rich
Installation: python -m pip install rich
, Python 3.7+
Website: https://github.com/gruns/icecream
Installation: python -m pip install icecream
, Python 2, 3
The next tool is a three-fer: devtools, rich, and IceCream are all great libraries and which one I reach for just depends on how I’m feeling at the moment. They all make printing complex data structures so much nicer than Python’s pprint
. They also have more tricks up their sleaves than just pretty printing. Let’s dig in…
devtools.debug()
debug
is like
The devtools Usage page has a colorful-but-apt description of what debug
does. It will pretty print as many variables as you can throw at it, along with the file name, line number, and function where debug
is being invoked.
data:image/s3,"s3://crabby-images/951f6/951f67463b7cd1ddea17b895f5e54299a36144ce" alt="Screen capture of printing out both a complex—nested dictionaries and lists—data structure as well as the returned value from a function call using devtools.debug()"
Additionally, debug
returns what you pass into it, meaning you can insert it into your code without changing functionality:
data:image/s3,"s3://crabby-images/f5ed6/f5ed60c85ef90e8a8118d44e24211dcc32afaab7" alt="Screen capture of devtools.debug() returning the class passed into it, allowing the variable assignment to happen as expected"
devtools.debug.timer()
You can also use the timer
function to do basic timings on operations:
data:image/s3,"s3://crabby-images/8f9a6/8f9a634b0fea6962514ca6e3d8710d59d7c99155" alt="Screen capture of devtools.debug.timer() timing a longer-running operation"
devtools.debug.breakpoint()
Finally, you can use debug.breakpoint()
to invoke pdb
:
data:image/s3,"s3://crabby-images/143aa/143aa220d44242854d4e402678d584abd98ba64c" alt="Screen capture of devtools.debug.breakpoint() invoking pdb"
rich.print()
Similar to devtools.debug()
, rich.print()
is a better pretty printer. Unlike devtools.debug()
, which is solely focused on debugging, rich.print()
supports a slew of formatting options, such as shortcodes (including emojis).
data:image/s3,"s3://crabby-images/c6647/c6647458612a38bc282918b53f10a28084581e48" alt="Screen capture of rich.print() formatting shortcodes and emojis"
rich.inspect()
rich.inspect()
is a nice tool for digging into variables; it shows the type, the contained values, and optionally methods, private, and dunder attributes. Note: if “dunder” is a new term for you, it refers to attributes with double underscores around them, like __init__
; “double underscore” gets tiresome to say, so you’ll frequently see that shortened to “dunder” in the Python world.
data:image/s3,"s3://crabby-images/48b71/48b71e866f3fe472a19e06c7dc806cb81b37e7a2" alt="Screen capture of rich.inspect() showing more details about an object"
icecream.ic()
ic()
is pretty similar to the other two libraries but with the added advantage that it’s even faster to type!
data:image/s3,"s3://crabby-images/4d32f/4d32fdb4ca44c0c45ecf0328c8552d922edaf7ca" alt="Screen capture of icecream.ic() printing out basic data"
It will print both the variable name and its value, with syntax highlighting.
data:image/s3,"s3://crabby-images/15181/15181a65af817ba6b5735799ed3121831565207c" alt="Screen capture of icecream.ic() printing out a complex dictionary"
Like devtools.debug()
it’ll return what you passed in, so you can inline it in your code.
data:image/s3,"s3://crabby-images/5a019/5a0192ef80049de27d6d8c55850ab81f8318e410" alt="Screen capture of icecream.ic() printing out a class; the instance is also passed to a variable, which is then passed to another ic() call to print the name attribute"
One additional neat trick is that you can scatter it throughout your code like debugging breadcrumbs and it’ll inspect itself, leaving you with a trail of exactly which lines in the code were triggered:
data:image/s3,"s3://crabby-images/acb6e/acb6efb76e9d2027d771104ffc5fa36c82db389e" alt="Screen capture of icecream.ic() printing out a trail through multiple functions and conditionals"
2. pytest-sugar
Website: https://github.com/Teemu/pytest-sugar
Installation: python -m pip install pytest-sugar
, Python 3.8+, pytest 6.2+
pytest-sugar is a drop-in (automatically activated) plugin for pytest. It does several things to make your tests look and work better:
- Errors are reported immediately; no waiting to see what broke
- A progress bar visualizes how far you’ve progressed through the suite
- Colored summary report
Here’s a test suite without pytest-sugar:
data:image/s3,"s3://crabby-images/002a7/002a76680d34c8fed9a9ec2dab90fe4c12758ffc" alt="Screen capture of a test suite, with some failures, using default pytest formatting"
And the same test suite with pytest-sugar:
data:image/s3,"s3://crabby-images/e1dc1/e1dc1c9fc4bc24c7a3a4f47487f70426062d8969" alt="Screen capture of a test suite, with some failures, using pytest-sugar's formatting"
1. Ruff
Website: https://docs.astral.sh/ruff/
Installation: python -m pip install pytest-sugar
, Python 3.7+
Ruff is an extremely fast Python linter. In addition to its speed, I also appreciate its sensible default settings and large ruleset.
Linting in Python is worth its own blog post, so I’ll try to summarize here. The question, “what linter should I use for Python?” always felt fraught with uncertainty and complications. I’ve seen many projects end up with autopep8, flake8, pylint, bandit, and isort and nobody knows which are still used … or even what they’re used for.
Ruff replaces them all.
It’s hard to understate just how little work we’ve had to do to switch to Ruff at my current client: we just swapped the dependencies out, deleted all the non-Ruff configuration, and were good to go. We started migrating to Ruff when we upgraded to SQLAlchemy 2.0 and ran into a bug that caused our pylint runs to be unbearably slow. The bug is now fixed as of pylint 3; however, our switch to Ruff was so easy that we didn’t need to wait around for that fix to land.
As Ruff matures, we continue to replace tools. The most recent swap was pulling bandit for Ruff’s flake8-bandit ruleset. We’re keeping an eye on the Ruff Formatter, and it seems likely that we’ll be replacing Black soon. (Black is another tool I love, and it would be on this list if it weren’t for Ruff moving towards feature parity.)
BONUS: VS Code Extensions
Because you made it this far, I’ll throw in two recently-discovered, favorite VS Code extensions:
indent-rainbow
Makes your indent levels both pretty and easy to read!
data:image/s3,"s3://crabby-images/44694/44694d3c0757119e970b8e1cbacbebdc81ba98ba" alt="Screenshot of indent-rainbow shading different indent levels different colors in a JSON document"
gitmoji
Spice up your commit messages with gitmojis, because every commit log should be as beautiful as FastAPI’s.
data:image/s3,"s3://crabby-images/b27f3/b27f3424748d98eccd9769b80c4a35deca3af535" alt="Screenshot of FastAPI's commit log with gitmoji commit messages"
Want to continue the conversation about Python tooling? Join our N.E.A.T. community or if you’re already registered, head on over to the forum to talk Python tool tips!