Documentation
Learn
Getting Started

Getting Started with pest

In this chapter, we'll guide you through the process of setting up a new project and creating a basic API using pest.

Code

The code for this tutorial is available on Github.

Each branch of the repository represents a chapter of this tutorial.

Installing the CLI

The first step is to install the pest CLI. Open your terminal and execute the following command:

~/haunted-houses
 pip install pest-cli

While using the CLI is not mandatory, it simplifies the process of creating new projects. It doesn't need to be a dependency of your project; you can install it globally.

Creating a new project

Now that pest-cli is installed, let's kickstart our haunted-houses project:

~/haunted-houses
 pest generate application haunted-houses --dm poetry

This command creates a new pest project using poetry as the dependency manager. Ensure poetry is installed on your machine for this to work. If you don't have poetry installed or don't want to install it, you can omit the --dm poetry flag, and pest-cli will prompt you to choose a dependency manager. As of the time of writing this tutorial, pest-cli supports poetry and requirements.txt. If you prefer using a requirements.txt file, you can install the dependencies with pip install -r requirements.txt, and you're good to go.

Running the project

Now that the project is created, let's install the dependencies:

~/haunted-houses
 cd haunted-houses
 poetry install

Next, run the project:

~/haunted-houses
 poetry run uvicorn haunted_houses.main:app --reload
INFO:     Will watch for changes in these directories: ['~/haunted-houses']
INFO:     Uvicorn running on http://localhost:8000 (Press CTRL+C to quit)
INFO:     Started reloader process [21816] using StatReload
INFO:pest:Initializing haunted-houses
INFO:pest:Setting up HelloController
DEBUG:pest:haunted-houses initialized:
AppModule 🐀
    
    ├─ HelloModule
          GreeterService
          HelloController
 
INFO:     Started server process [24852]
INFO:     Waiting for application startup.
INFO:     Application startup complete.

Open your browser and navigate to http://localhost:8000 (opens in a new tab); you should see a JSON response similar to this:

{
    "hello": "pest!"
}

Project structure

Let's take a look at the project structure generated by pest-cli:

      • greeter_service.py
      • hello_controller.py
      • module.py
  • app_module.py
  • main.py
  • app_module.py

    This file contains the main/root module of the application.

    It instructs pest on how to locate the controllers and services within the application.

    The main module can directly import other controllers or services, or it can import other modules that define their own controllers and services. In this instance, the AppModule imports the HelloModule, a feature module that encapsulates the "greeting" feature of the application.

    📄 app_module.py

    Here's how the app_module.py file looks:

    from pest import module
     
    from .hello.module import HelloModule
     
     
    @module(
        imports=[HelloModule],
    )
    class AppModule:
        pass

    The @module decorator defines a module, and the imports argument is used to import other modules into the current module. In this case, we're importing the HelloModule into the AppModule.

    hello/module.py

    This is a feature module, encapsulating a specific feature of the application.

    In the case of HelloModule, it encapsulates the greeting feature, composed of a controller (HelloController) and a service (GreeterService).

    📄 hello/module.py

    Here's how the hello/module.py file looks:

    from pest import module
     
    from .greeter_service import GreeterService
    from .hello_controller import HelloController
     
     
    @module(
        controllers=[HelloController],
        providers=[GreeterService],
    )
    class HelloModule:
        pass

    The controller defines the routes of the application, and the service contains the business logic. pest injects the service, making it available for injection across all controllers and providers of the module. In this example, we have one controller and one service, but you can have as many as needed.

    hello/greeter_service.py

    This file represents a provider. Typically, a provider or service contains the business logic of the application.

    While a controller handles requests and returns responses on specific routes, a service manages the business logic.

    In the usual workflow, a controller calls a service to handle the business logic and return a response to the client.

    📄 hello/greeter_service.py

    Here's how the hello/greeter_service.py file looks:

    from pydantic import BaseModel
     
     
    class Greet(BaseModel):
        """A greeting"""
     
        hello: str
        '''The greeting'''
     
     
    class GreeterService:
        """The haunted house service"""
     
        def say_hello(self) -> Greet:
            """Say hello"""
            return Greet(hello="pest!")

    This is a simple service that returns a greeting.

    hello/hello_controller.py

    This file serves as a controller, responsible for handling requests and returning responses on specific routes.

    pest automatically invokes the appropriate route handler (a method of the controller) based on the HTTP method and path of the request.

    Typically, a controller delegates business logic handling to a service and returns a response to the client.

    📄 hello/hello_controller.py

    Here's how the hello/hello_controller.py file looks:

    from pest import controller, get
     
    from .greeter_service import Greet, GreeterService
     
     
    @controller("/")
    class HelloController:
        """say hello routes"""
        service: GreeterService  # 💉 injected
     
        @get("/")
        def say_hello(self) -> Greet:
            """Say hello"""
            return self.service.say_hello()

    This defines a controller with a single route handler. The handler is triggered when a GET request is made to the root path of the application. It calls the say_hello method of the GreeterService and returns its response.

    main.py

    This is the entrypoint of the application. It contains the main function that starts the application. Remember when we ran poetry run uvicorn haunted_houses.main:app --reload? The app.main:app part tells uvicorn to look for the app module and the app variable inside the main.py file and mount it as the root application.

    📄 main.py

    Here's how the main.py file looks:

    import logging
     
    from pest import Pest
     
    from .app_module import AppModule
     
    logging.basicConfig(level=logging.DEBUG)
     
     
    app = Pest.create(
        AppModule,
        title='haunted-houses'
    )

    The Pest.create method creates a new Pest application, passing the AppModule as the root module and the title of the application. The title will be then used as the application's title in the OpenAPI specification (e.g. by Swagger UI).

    OpenAPI

    Thanks to FastAPI (opens in a new tab), pest automatically generates an OpenAPI specification for your application and mounts a Swagger UI at /docs and a ReDoc UI at /redoc.

    Conclusion

    In this tutorial, we've learned how to create a new project using pest-cli, and we've delved into each file generated by it, learning how they collaborate to create a simple API.

    In the following chapters, we will modify this project to add more features and learn more about pest.