DEV Community

Ran Isenberg for AWS Community Builders

Posted on • Originally published at ranthebuilder.cloud

12 1 1 1 2

Python Cookiecutter: Streamline Template Projects for Enhanced Developer Experience

Photo by Pavel Danilyuk: [https://www.pexels.com/photo/close-up-shot-of-unbaked-cookies-6996345/](https://www.pexels.com/photo/close-up-shot-of-unbaked-cookies-6996345/)

The Python Cookiecutter library revolutionizes project development by offering streamlined approach to creating template projects and improving developer experience.

cookiecutter allows developers to quickly scaffold their projects with pre-defined structures, configurations, and best practices.

By abstracting away the intricacies of environment setup, cookiecutter enables developers to dive straight into coding, significantly reducing the time and effort required to get up and running. Embracing cookiecutter as a valuable tool can revolutionize the onboarding experience, enhance productivity, and empower developers to focus on what they do best: building innovative and high-quality software.

In this post, you will learn how to generate Python templates with cookiecutter and build new template projects from scratch. We will review usage examples and follow precise steps to help you develop your cookiecutter Python template project.

[https://www.ranthebuilder.cloud/](https://www.ranthebuilder.cloud/)

This blog post was originally published on my website, “Ran The Builder.”

Developer Experience and Template Projects Matter

I will say that template projects are critical to the success of your organization.

In my organization, CyberArk, we were in the process of adopting a new technology: Serverless.

I was on a pioneering group working furiously to learn the ropes of this new technology. Eventually, we built a new Serverless service with numerous best practices such as infrastructure as code (AWS CDK), a dedicated CI/CD pipeline, and observability built in.

It was an excellent service and worked well.

However, now the team faced new challenges: building the second service and another challenge, even harder: helping news teams at CyberArk create the same levels of serverless services with the same tools. That’s where template projects come in handy.

We turned the first service, that state-of-the-art service into a template project — a simple yet generic enough service so any team can use it as a starting point. It was a fully working repository with all the bells and whistles that help the development teams focus on what matters most — the business domain. Read more about it here.

In one instance, a team developing a serverless service with the template got to the design partners’ stages (when an actual customer uses the service) in just four months. For an enterprise such as CyberArk, this was unheard of and very quick.

Now that we understand the incentive for template projects, let’s start with generating a new repository from a template with cookiecutter and then process to create our template.

Installing Cookiecutter

First, make sure you install Python 3.

Then, run the following command:

pip install cookiecutter

or on a mac:

brew install cookiecutter

If you require more assistance, read the official installation guidelines.

Cookicutter — The User Experience

Before creating templates, we should understand what kind of user experience our developers will get. We should strive to make it simple, fast and remove as many manual steps as possible.

Creating a new working developer environment is one of the most complex challenges developers often face. We can use cookiecutter to build our project and set up the developer environment — more on that in the ‘hooks’ section.

Let’s create a new Serverless service from my cookiecutter serverless project, based on my AWS Lambda handler cookbook project.

The serverless service template has numerous features:

  • CDK infrastructure with infrastructure tests and security tests.

  • CI/CD pipelines based on Github actions that deploys to AWS with python linters, complexity checks and style formatters.

  • Makefile for simple developer experience.

  • The AWS Lambda handler embodies Serverless best practices and has all the bells and whistles for a proper production ready handler.

  • AWS Lambda handler 3 layer architecture: handler layer, logic layer and data access layer.

  • Features flags and configuration based on AWS AppConfig.

  • Unit, infrastructure, security, integration and E2E tests.

And the architecture diagram:

service design

The Setup

Run this command:

cookiecutter gh:ran-isenberg/cookiecutter-serverless-python

Answer the following questions to scaffold the project:

answer the questions

Cookiecutter will start to scaffold the project and initialize a working environment.

Once completed, a message will appear: “Project successfully initialized.”

That’s it! Simple as that, you can start developing your shiny new serverless service and deploy it to AWS.

If you liked the project and the experience, don’t be a stranger and give it a star :)

Create Your Own Template

Now that we understand what we want to achieve, let’s build it.

Repository Structure

Start with an empty repository. You will need to add four files at the top level of the project:

  1. Readme file — explain the project’s purpose and provide setup instructions.

  2. cookiecutter.json — this file provides cookiecutter with the configuration and scaffold parameters to use when cloning a new project. More on that below.

  3. The root folder of the template project you wish to provide.

  4. Hooks folder — used for advanced use cases; see below.

So you end up with something that loos like this:

folder structure

Let’s go over the main folder, the hooks folder and cookiecutter.json files.

cookiecutter.json

In this file, you define cookiecutter’s scaffolding parameters which impact the questions presented to the user during the initial setup. You can choose whatever you want, but I’d go with the following example as a starting point:

{
"repo_name": "my_serverless_service",
"service_name": "service",
"email": "dummy@dummy.com",
"author": "Dr. Myfull Name",
"description": "My serverless service description",
"_copy_without_render": [
"*.html",
".github"
]
}

The provided values default if the user presses enter as an answer and provides no other input.

The “_copy_without_render” part helps add files and folders you don’t want cookiecutter to go over and copy as is.

The author, email, and descriptions parts can be injected into the template readme file and poetry.toml descriptions section.

See an example here and here.

There’s also support for multi choice options as described here.

Main Template Folder

This is the entry folder for your template project. You must rename it to a name that cookiecutter can recognize and scaffold. All files beneath it will be added and scaffolded.

In my template, I renamed it to ‘{ { cookiecutter.repo_name } }’. Notice that the ‘repo_name’ is the parameter defined in the cookiecutter.json file.

Usually, under the main folder, you’d add another folder for the service name, which I defined as ‘{ { cookiecutter.service_name } }’

So it can look like this:

inner project structure

Not all folders require scaffolding support, and you can decide what folders remain constant and which are renamed.

Notice how the inner folder has its .toml file for poetry, .github for CI/CD workflows, CDK folder for deployment, tests folder, and other required folders. You can add whatever you wish.

One crucial issue you must know is that scaffolding breaks Python’s import path. Since the main folder is dynamic and determined by the user when they answer cookiecutter questions, it means that are Python import path declarations require us to edit them too.

Here’s an example how you overcome it:

from http import HTTPStatus
from typing import Any, Dict
from aws_lambda_powertools.metrics import MetricUnit
from aws_lambda_powertools.utilities.feature_flags.exceptions import ConfigurationStoreError, SchemaValidationError
from aws_lambda_powertools.utilities.parser import ValidationError, parse
from aws_lambda_powertools.utilities.parser.envelopes import ApiGatewayEnvelope
from aws_lambda_powertools.utilities.typing import LambdaContext
from {{cookiecutter.service_name}}.handlers.schemas.dynamic_configuration import MyConfiguration
from {{cookiecutter.service_name}}.handlers.schemas.env_vars import MyHandlerEnvVars
from {{cookiecutter.service_name}}.handlers.schemas.input import Input
view raw imports.py hosted with ❤ by GitHub

Notice lines 10–12 and how we import the files with scaffolding support.

The complete file can be found here.

You can use to method to rename any values in any file.

Hooks Folder

The hooks folder is optional, but I suggest you implement it too.

Hooks are code run either before the cookiecutter rename process or after it is finished.

You can use it to conduct input validation on the questions and set up an entire developer environment once the project is scaffolded.

Pre Hook

You can use the pre-hook for input validation on the user’s strings. If you recall, some of these strings are used as service or repository names. They are used in the import path of the generated repository files, so they must conform to Python’s naming convention.

You can use the following example:

import re
import sys
MODULE_REGEX = r"^[a-zA-Z][_a-zA-Z0-9]+$"
module_name = "{{ cookiecutter.repo_name }}"
if not re.match(MODULE_REGEX, module_name):
print("ERROR: %s is not a valid Python module name!" % module_name)
sys.exit(1)
SERVICE_REGEX = r"^[a-zA-Z][_a-zA-Z0-9]+$"
service_name = "{{ cookiecutter.service_name }}"
if not re.match(SERVICE_REGEX, service_name):
print("ERROR: %s is not a valid serverless service name!" % service_name)
sys.exit(1)
view raw pre_hook.py hosted with ❤ by GitHub

Post Hook

This is where the magic happens. We’ve talked a lot about the importance of developer experience, and the post hook is where you can make a difference.

You can write scripts and initialize the developer environment so the developer can get into writing new code instead of wasting time on manual and tedious environment setups.

Here’s an example of my post-hook script where I initialize git, install all poetry dependencies, and install pre-commit so all checks run before a PR can work.

import subprocess
def main():
print("Running 'git init'")
subprocess.run(['git', 'init'], check=True)
print("Running 'pip install --upgrade pip pre-commit poetry'")
subprocess.run(['pip', 'install', '--upgrade', 'pip', 'pre-commit', 'poetry'], check=True)
print("Running 'pre-commit install'")
subprocess.run(['pre-commit', 'install'], check=True)
print("Running 'poetry config --local virtualenvs.in-project true'")
subprocess.run(['poetry', 'config', '--local', 'virtualenvs.in-project', 'true'], check=True)
print("Running 'poetry install'")
subprocess.run(['poetry', 'install'], check=True)
print("Project successfully initialized")
return
if __name__ == "__main__":
main()
view raw post_hook.py hosted with ❤ by GitHub

Testing & Debugging Locally

It sounds pretty simple, right?

However, it won’t work for you the first time. It never does. So you need to debug, fix and debug again.

cookiecutter supports running from a local folder, not just a GitHub repository URL.

You can develop the new template repository locally, make changes and then run cookiecutter from the terminal and see if it works:

cookiecutter {path-to-project-on-local-disk}
Enter fullscreen mode Exit fullscreen mode

For more tips & tricks, read the official docs: https://cookiecutter.readthedocs.io/en/stable/

Billboard image

Never miss a downtime

With Checkly, you can use Playwright tests and Javascript to monitor end-to-end scenarios in your NextJS, Astro, Remix, or other application.

Get started now!

Top comments (6)

Collapse
 
vincentgna profile image
Vincent • Edited

given you mention CDK in the tech used, why choose cookiecutter over projen?

How does cookiecutter keep projects generated from a template, updated with new features added to the template? (rolling out new standards across all projects, can it be driven through the template they were generated from?)

Collapse
 
ranisenberg profile image
Ran Isenberg

cookiecutter is generic, simple and easy to use. You can use it with whatever IaC you want, it does not have to be CDK.

Collapse
 
vincentgna profile image
Vincent

understood, but the main problem Projen addresses (and highlights as a problem in Cookiecutter / yeoman / ...) - is updating projects after the template has been updated. Is that possible with Cookiecutter?

Thread Thread
 
ranisenberg profile image
Ran Isenberg

no, that's not possible but that's not really the purpose here.
The way I see it, you create template for quick start, in this case, in the serverless world.
The team will change the CDK constructs, write their own business domain, change the tests etc. They will have zero resemblance to the original project, even the CI/CD github actions part might change.
In addition, I'd like to only one version control and that's with poetry. Do you want to share constructs and update them? great, publish them as a python library, add it to the toml file and manage the version there, all dependancies from one place.

Thread Thread
 
vincentgna profile image
Vincent • Edited

my concern is with divergence across projects and maintaining boilerplate github workflows / security alerting / ...

Concerns I have as to why I want to avoid divergence:

  • communication across teams / easily move between teams and onboard resources
  • maintainability / security / reduce errors

Having worked in several companies with poly repos, I've struggled many times with having to push team managed repositories to adopt company wide standards (given company wide standards always change over time).

I've seen many approaches to this (including an OpenFAAS based approach using "templates" (github.com/openfaas/templates) which prescribes where the Microservice code will be injected into)

Allowing the security team to maintain the base docker image and ensuring compliance across the organization.

at the other end of the spectrum, seems to be as you suggest full team autonomy and perhaps a type of "Backstage" tracking company wide project adoption across teams?

Thread Thread
 
ranisenberg profile image
Ran Isenberg

I fully understand your point, it's a real struggle.
I dont suggest full automony, but a balance.
We manage our devtools config in a shared repo (pylint, flake8 etc.), same for security rules, waf rules as CDK constructs. We have cdk contructs for a secure SNS/SQS pattern etc.
So it's possible.
The only issue I have is the github actions, where all teams start from the shared template but they might change over time, and then each team is required to update it manually.
But it's kinda the same with the shared repos, the team needs to change the version or make some changes to support a new (maybe breaking) version.

My rule of thumb is: if it's not business domain oriented and i can put it in a shared repo, you should do it and have everybody use it from there.
but versions should be managed in one place, the toml file. That's also why i dont like lambda layers, but that's another story :)