Skip to main content

Minimal, spiritual fork of projen. No JSII. Python focused.

Project description

pyprojen

A minimal, Python-centric port of projen.

Apache 2.0 License Commit activity Projen debut demo


Overview

Project templates are great... except that they promote "code duplication at scale" and all the technical debt that come with that.

projen (the tool this project is based on) represents a new generation of project template tools. projen helps you manage the "day 2 problem" for project templates, to avoid "template drift".

projen also helps you treat project configuration as an abstraction, whose implementation details can be changed/improved, without disrupting the developer workflow. Elad Ben-Israel, creator of projen, AWS CDK, and Winglang gave a talk on this at CDK day 2020.

  1. ⚡️ Agility. Centrally push unit-tested updates to all existing projects created from your templates in minutes, not months.
  2. 🧱 Modularity. Define composable, parameterized "components" that you can add/remove from your project as needed.

Real Examples

pyprojen is a minimal port of projen's core functionality to Python.

Specifically, it gets you

  1. ⚡️ Agility
  • Examples:
    • 10x the speed of CI for all repos overnight by making CI steps run in parallel instead of serially
    • Update a team linting rule in pyproject.toml, ruff.toml, etc.
    • Add auth steps to publish to or install from a private PyPI registry in CI or Dockerfiles
    • Completely change CI systems with minimal disruption, e.g. switch from Bitbucket Pipelines to GitHub Actions to AWS CodeBuild and back
  1. 🧱 Modularity
  • Examples:
    1. define an opinionated PythonPackage component, and
    2. layer on top a
      • FastAPIApp,
      • StreamlitApp,
      • CdkApp,
      • PulumiApp,
      • AirflowDag,
      • DagsterDag,
      • BentoMLService,
      • AwsLambdaPythonFunction,
      • etc.
    3. Add (or remove) as many of these to your repo as you like, whenever you like, and find these packages instantly set up with CI, linting, formatting, tests, packaging, publishing, deploying, etc.
      • For example, you might incrementally develop a "mini data science app monorepo" with
        1. a MetaflowDag that trains a model
        2. served in a FastAPI app
        3. which you can interact with via a Streamlit app
        4. both of which are deployed by an AwdCdk app
      • all of these components would ideally be instantly set up with CI, linting, formatting, testing, packaging, publishing, deployment, etc. the moment they are added. And "un set up" with those if they are removed.

Q&A

How does (py)projen compare to copier or cookiecutter?

Click to expand

copier is a reaction to cookiecutter, built to allow template updates to be propagated to existing projects.

  1. Migration. The migration process for (py)projen is a single CLI command (python .pyprojenrc.py). Whereas the migration process for copier is a bit more manual and prone to errors.

  2. Composable components. copier is more like cookiecutter in that it uses Jinja templates to generate a certain set of files. (py)projen lets you define re-usable components. You can add arbitrary numbers of these components to your project with different parameters and remove them just as easily.

That said, although cookiecutter and copier are more limited, they are also simpler.

What is the difference between pyprojen and projen?

Click to expand

1. Fundamentally, projen offers a few things:

  1. A Project, Component, and Construct abstraction that lets you define reusable components that you can push updates to.
  2. Primitive components like TextFile, YamlFile, JsonFile, MarkdownFile, etc. that you can compose to build "higher-level components".
  3. A library of opinionated, higher-level components like PythonProject, TypescriptProject, DockerCompose, GithubWorkflow, ...
  4. An opinionated "task runner" system (think Makefile/Justfile, poetry scripts, etc.) to define project-related commands.
  5. A projen new command which creates the initial .projenrc.py config file for your project

2. pyprojen implements [1] and [2] from the list above (the unopinionated parts).

It is up to you to create your own components with your own opinions on things like

  • when, if, and how to manage virtual environments, e.g. uv, pip, conda, etc.
  • which linter/formatter to use, e.g. ruff, pre-commit, etc.
  • how to structure single- and multi-package repos (monorepos) and CI for them

Coming from tools like cookiecutter or copier, many people/teams prefer than using off-the-shelf templates or components.

3. pyprojen is not a drop-in replacement for projen, but it tries to get close.

If you write components in Python using pyprojen, it should be easy to move them over to the projen Python bindings if you decide to.

Should I use pyprojen or projen?

Click to expand

TL;DR Bias towards projen, unless you

  1. Want a Python-first dev experience, and
  2. Prefer to fully-define your own project template/components rather than using projen's existing project templates, higher-level components, or task runner system

projen is a larger project and is primarily maintained by developers at AWS. projen,

But to develop with projen, you either need to write TypeScript, or use generated Python bindings that invoke TypeScript.

If you are familiar with writing AWS CDK in Python, developing with projen in Python is a similar experience, because they both use Python bindings generated from TypeScript using the JSII project.

This means:

  1. Not all internals of the original TS/JS code is exposed in the TS bindings, e.g. private attributes. You can unexectedly hit dead ends when attributes or methods that are available in TS are simply not available in Python.
  2. Step debugging is limited. The bindings are thin wrappers around a tool that invokes the original TypeScript/JavaScript code
  3. Errors raised by python bindings are cryptic and difficult to parse.
  4. Autocompletion is poor
  5. You need to have NodeJS installed on your system and in CI
  6. The JSII is a bit slow. (seconds not milliseconds)

Quick start (TODO)

[!NOTE] Until this section is filled out, you can refer to this repo to get a sense of what projen can do. And the official projen docs contain many of the same concepts that this port uses.

pip install pyprojen
from pyprojen import ...

Developing/Contributing

System requirements

You will need the following installed on your machine to develop on this codebase

  • make AKA cmake, e.g. sudo apt-get update -y; sudo apt-get install cmake -y
  • Python 3.7+, ideally using pyenv to easily change between Python versions
  • git

# clone the repo
git clone https://github.com/<your github username>/pyprojen.git

# install the dev dependencies
make install

# run the tests
make test

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

pyprojen-0.0.3.tar.gz (27.0 kB view details)

Uploaded Source

Built Distribution

pyprojen-0.0.3-py3-none-any.whl (29.7 kB view details)

Uploaded Python 3

File details

Details for the file pyprojen-0.0.3.tar.gz.

File metadata

  • Download URL: pyprojen-0.0.3.tar.gz
  • Upload date:
  • Size: 27.0 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/5.1.1 CPython/3.11.9

File hashes

Hashes for pyprojen-0.0.3.tar.gz
Algorithm Hash digest
SHA256 4914d405f7e08f13ca6f6ac43b3813ea77cd74d14fe83f5ed753f41e46e93f7c
MD5 1aff5cb043e31051b07591f91ef155bf
BLAKE2b-256 bca89d23f8358b09937bde6bb12fe86d7bf7a912e29c97bda1e8610270a0fdad

See more details on using hashes here.

File details

Details for the file pyprojen-0.0.3-py3-none-any.whl.

File metadata

  • Download URL: pyprojen-0.0.3-py3-none-any.whl
  • Upload date:
  • Size: 29.7 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/5.1.1 CPython/3.11.9

File hashes

Hashes for pyprojen-0.0.3-py3-none-any.whl
Algorithm Hash digest
SHA256 e795fd11541ca695fbfcd71e822413edccab188a13eab8407b28b76fd1dffdc4
MD5 7301ea43c8e642f59681a4cd152acd5c
BLAKE2b-256 71187023856d99ea1d9a564eada6a99619d4c73b1e39bbf1e4e25a2c38de34c4

See more details on using hashes here.

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page