Modules
A module is a class decorated with @module
decorator. They are used to organize the application
grouping related resources that provide a distinct business value together into reusable
self-contained units.
👋 @module has an alias
In the python world, the word "module" is often used to refer to a python file. As this library took inspiration from nest.js, we decided to use the word module anyway, but we also added some aliases, just in case you prefer to use a different word to refer to the same concept and avoid confusion.
All the following decorators are equivalent:
@module
@domain
@dom
We took the word domain from DDD (Domain Driven Design) (opens in a new tab). In DDD, a domain is collection of closely related concepts that have a clear business value.
All pest applications have at least one module, the root module. The root module is the
starting point of the application and it's used by pest
to build the application dependency graph.
The dependency graph is a collection of internal metadata that tells pest
how to build
and organize the application.
Nevertheless, pest
aims at mid to large scale APIs, so most applications will tipically
have more than one module, each of them containing and encapsulating closely related components
and services that when put together provides a distinct business value to the application.
👋 Tip: You can actually see the dependency graph!
If you print the string representation of your application instance, you will see how pest
created the dependency graph of your application as an actual ascii graph.
>>> from pest import Pest
>>> from .app_module import AppModule
>>> app = Pest.create(AppModule)
>>> print(app)
AppModule
│
├─ AuthModule
│ │ ○ AuthService
│ │ □ AuthController
│
├─ HauntedHouseModule
│ ○ HauntedHouseService
│ □ HauntedHouseController
from pest import module
from .haunted_house_controller import HauntedHouseController
@module(
controllers=[HauntedHouseController],
providers=[HauntedHouseService],
)
class HauntedHouseModule:
pass
The @module
decorator
The @module
decorator can take the following arguments:
imports | A list of imported modules that export the providers which are required in this module. |
providers | The providers that will be contained and can be injected for other componentsin the module. Also called dependencies or services. |
exports | The providers that will be exported and can be injected for other modules that import this module. They can be a subset of the providers in the providers list. This means that exports define the public interface of the module. |
controllers | The controllers that will be contained in the module. |
Feature modules
A feature module organizes different pieces of the application that are closely related and belongs to a specific domain.
When applications grow, it's common to have a lot of different components and services that are not closely related to each other. That's when applying the separation of concerns principle by creating feature modules containing related components and services becomes handy and helps to keep the application organized and maintainable over time.
Feature modules are just regular modules, but they are usually created in a different directory
and are imported by the root module or other feature modules using the imports
argument of the
@module
decorator.
Let's say we have some authentication related components and services that we want to group together in a feature module:
from pest import module
@module(
providers=[AuthService, AndOtherServices, RelatedToAuth],
exports=[AuthService], # we export the AuthService so other modules importing this
# feature module can use it as a dependency
)
class AuthModule:
pass
Notice that we are only exporting the AuthService
. Using the exports
argument of the
@module
decorator we can define the public interface of the module, that is, the
providers that will be available for other modules that import this one.
Now we can import the AuthModule
in the root module or any other feature module that needs
to use the AuthService
:
from pest import module
@module(
imports=[AuthModule, HauntedHouseModule], # we import the feature modules we want to use
providers=[AppService],
controllers=[AppController],
)
class AppModule:
pass
This way, AuthService
is available for the modules that import the AuthModule
(or by
all of them, if it's imported by the root module)
This allows us to create isolated and self-contained feature modules that can be imported by others while, at the same time, keeping the public interface of the module clear and explicit.
This opens the door to create reusable feature modules that can even be its own package, published and developed independently from the rest of the application (even by a different team), making our feature modules completely encapsulated (opens in a new tab).