The Clean Architecture can be very cool while you implement your domain layer. Your code is pretty, your business logic is Clear, your dependencies are amazingly abstract. You even learn about Domain Events and fascinated to make some. Everything is perfect.
Then you go on to implement the Application layer. You start making your Use Cases and find yourself quite puzzled. You need to implement transactions, event dispatching, loading of dependencies, loading of dependencies with transactions... You understand that you would be very happy if all of that was implemented by someone else. After all, its just technical nuances that distract you from real business values of your project.
Once upon a time a hard-working beaver named Bobr faced the same problems and made a solution that you could find suitable for your needs. Let's see.
We will start from Use Cases.
The type of the DUseCase
is quite tricky but you only need to know about the three constructor parameters. Let's see an example:
const createUserUC = new DUseCase(
"create user",
Type.Object({
username: Type.String(),
password: Type.String(),
email: Type.String(),
age: Type.Optional(Type.Number()),
status: Type.Optional(Type.String()),
}),
async(params, deps: Dependencies, useCaseName, handleDEvent) => {
console.log(`${useCaseName}...`);
const user = new User(
params.username,
deps.passwordService.hashPassword(params.password),
params.age ?? null,
params.status ?? null,
);
deps.userRepo.add(user);
handleDEvent([new UserAdded(params.username)]);
}
);
The first parameter is the name of the Use Case. It will be remembered by Typescript and will help us in the future.
The second one is the validator of the Use Case parameters from Typebox library. This is my decision for an old dilemma about DTOs: if you declare them in the Application layer you lose its support by features of the Presentation layer frameworks like JSON validation, if you declare them in the Presentation layer you can't use them in Application layer because the Application layer can't depend on the Presentation layer. So I decided to use Typebox: it's simple, statically typed and easily converted to JSON Schema which is quite common and could be used by many Presentation layer frameworks, so this third-party library shouldn't corrupt our Application layer.
The third parameter is the Use Case function itself. It takes four parameters:
- The Use Case parameters which has static type from the Typebox schema.
- Then the Use Case dependencies - the explicit type, as you see, is required here, because the Use Case can't know what you want.
- Then the Use Case name - the name you put for your Use Case so you don't write it twice if there is a need for it.
- The last one is eventHandler function. If you have events just give them to this handler.
Now let's see how events look. The basic type of events in DDDam-App is:
interface DEvent<DEName extends string> {
eventName: DEName;
}
To implement your Event you only need to write something like:
class UserAdded {
eventName = "USER ADDED" as const;
constructor(
readonly username: string,
readonly email: string,
) {
this.username = username;
this.email = email;
}
}
You can easily declare your Events in the Domain layer any way you wish,
you don't even need to bring the DEvent
there because it's only an interface and Typescript is structurally typed.
But I beg you, don't forget about as const
for eventName
.
An Event will be given to corresponding Event Processors. For example:
const sendConfirmationEmailEP = new DEventProcessor(
"USER ADDED",
async(dEvent: UserAdded, deps: EventDependencies) => {
const confirmationCode = deps.confirmation.createConfirmationCode(dEvent.username);
deps.sendEmail(dEvent.email, `Your confirmation code: ${confirmationCode}`);
}
);
DEventProcessor
takes two parameters:
The first one is the Event name. The only reason this parameter exists is because
typescript can't "leak" an information from the compile-time to the runtime.
The name is already known from dEvent: UserAdded
parameter but Typescript can't help us,
so we must write it ourselves. But Typescript can check that
the name in the first parameter matches the name in Event type (like dEvent: UserAdded
),
and will inform us if we make any typo. That's why as const
for eventName
was important.
The second one is the processing function itself. The function takes two parameters:
the Event and EventProcessor
dependencies. You must write types for both.
Context Container is very important part of the DDDam-App
and a bit tricky. Maybe it's easier to show its type:
export type ContextContainer<UCDeps, EPDeps> = <R>(
action: (ucDeps: UCDeps, epDeps: EPDeps) => Promise<R>
) => Promise<R>;
You most implement Context Container by yourself.
It's a function which only parameter is an action - a function that takes two arguments:
Use Case dependencies and Event Processor dependencies. When Context Container is called,
it most build those dependencies and then call the action
with dependencies in its parameters.
The action itself is not the concern of the Context Container
(is's made by DApp
and actually runs use cases and events). You free to use closures or anything you want.
For example let's implement a Context Container for Repositories made by Kysely query builder:
// application layer
type UCDependencies = {
accountRepository: AccountRepository,
}
type EPDependencies = {
serverEmail: string,
}
// infrastructure layer
function getContextContainer(
mainConfig: MainConfig,
): ContextContainer<UCDependencies, EPDependencies> {
const db = new Kysely<Db>({
dialect: new PostgresDialect({
pool: new Pool({
host: mainConfig.dbHost,
port: mainConfig.dbPort,
database: mainConfig.dbName,
user: mainConfig.dbUsername,
password: mainConfig.dbPassword,
}),
}),
});
const contextContainer = async<R>(
action: (ucDeps: UCDependencies, epDeps: EPDependencies) => Promise<R>
) => {
return await db.transaction().execute(async t => {
const ucDeps = {
accountRepository: new KyselyAccountRepository(t),
};
const epDeps = {
serverEmail: mainConfig.serverEmail,
};
return await action(ucDeps, epDeps);
});
};
return contextContainer;
}
Here we made getContextContainer
function which returns our Context Container.
contextContainer
will always have an access to db
by javascript closure,
it builds KyselyAccountRepository
and executes the action
in the same Kysely transaction.
You can implement in Context Container anything you want: UnitOfWork
, IdentityMap
, etc.
And now let's combine everything!
To do it we need a simple DApp
instance. Maybe even not one if your needs demand that.
const app = new DApp(
getContextContainer(mainConfig),
[createUserUC, deleteUserUC, postUC],
[sendConfirmationEmailEP,
takeDiskSpaceEP,
emailModeratorForSuspectedPostEP,
freeDiskSpaceEP],
);
DApp
constructor takes three parameters.
The first one is the Context Container.
The second one is an array of all Use Cases we have. Their Dependencies types most match.
The third one is an array of all our Event Processors. Their Dependencies types most match too, but are different from those of Use Cases.
If you want to have different Dependencies types for Use Cases or Event Processors
you're free to have more than one DApp
instance:
const [authCC, postsCC] = getManyContextContainers(mainConfig);
const authApp = new DApp(authCC, authUseCases, authEventProcessors);
const postsApp = new DApp(postsCC, postsUseCases, postsEventProcessors);
And let's see the magic!!!
To run a Use Case we need to call app.run
method. And it has a LOT of type magic.
As you see, Typescript made IDE suggest Use Case names so we won't make a mistake. And it will inform if you write a wrong name.
Then we write Use Case parameters:
Typescript surprizes us again and makes IDE suggest the type of the parameters of the specified Use Case. But the parameters will be validated anyway so you can pass any not validated data.
Then the Use Case and Events will just work.