Command Bus Pattern
FancyCRUD is using the Command Bus Pattern to organize and manage the execution of commands within a form or table. This pattern is based on the principle of separation of concerns and promotes code modularity and scalability.
In the context of the Command Bus, a "command" typically refers to a request to perform a specific action in the application. Instead of executing these actions directly where they originate, they are sent through a command bus. The command bus acts as an intermediary that directs the command to the appropriate handler, responsible for executing the logic associated with that command.
The Command Bus consists of three main components:
Command: Represents the intention to perform an action in the application. It is an object that contains all the necessary information to carry out the operation.
Command Handler: Is responsible for executing the logic associated with a specific command. Each type of command has its own handler.
Command Bus: Is the component that receives the commands and directs them to the corresponding handler. It can also handle aspects such as command validation before sending it to the handler.
By using a Command Bus, a clearer separation is achieved between the code that generates commands and the code that executes the associated actions. This makes it easier to expand and maintain the system since it's simpler to add new commands and handlers without modifying existing code.
Let's see an example to execute a command:
<template>
<f-form v-bind="form" />
<pre>Form Data: {{ formData.jsonForm }}</pre>
</template>
<script lang="ts" setup>
import { useForm, FieldType, Bus, GenerateFormDataCommand } from '@fancy-crud/vue'
const bus = new Bus()
const form = useForm({
fields: {
firstName: {
type: FieldType.text,
label: 'First name',
}
}
})
const formData = computed(
() => bus.execute(
new GenerateFormDataCommand(form.fields)
)
)
</script>
The code above is creating an instance of the Bus
class, which it's the responsable to direct the command to the corresponding handler.
Then, we created a firstName
field inside a form. And last, a computed property to execute the command every time the form.fields.firstName
changes.
Command
You can use the FancyCrud
built-in tools to create your own commands. Let's create a command that greets a person each time the user presses the greet button on a form. First of all, you need to define the command class.
import type { IBaseCommand, Meta } from '@fancy-crud/bus'
export class GreetCommand implements IBaseCommand {
public readonly $meta: Meta<string>
constructor(
public readonly name: string,
) {}
}
We're creating a GreetCommand
class that implements the IBaseCommand
interface, which requires the implementation of the $meta
attribute, this attribute is a utility for the Bus
to be able to know the return type string
for this command.
Handler implementation
Now, let's create the handler for the GreetCommand
.
import type { GreetCommand } from '../commands/greet'
export class GreetHandler implements IBaseHandler<GreetCommand> {
execute(command: GreetCommand): string {
return `Hello ${command.name}`
}
}
As you can see, the GreetHandler
is implementing the IBaseHandler
, and we're passing the GreetCommand
as generic type value, so the handler is able to infer the correct command type parameter and return type for the execute
method.
Command handler registration
Last but not least, we need to register the command into the command bus. To accomplish that, just use the register()
function.
import { register } from '@fancy-crud/bus'
import { GreetCommand } from '../commands/greet'
import { GreetHandler } from '../handlers/greet'
register(GreetCommand, GreetHandler)
It is important to place the register
function in a place where you know that the functions will always be executed. Otherwise, you will receive an error message like the following:
Error
HandlerDoesNotExist: Handler not found for command: GreetCommand
INFO
For this example, we place the register
into the component, but you can place this register into the entry file.
Once you completed the steps above, you will be able to execute the command, let's try it:
<template>
<f-form v-bind="form" />
</template>
<script lang="ts" setup>
import { useForm, FieldType, Bus, register } from '@fancy-crud/vue'
import { GreetCommand } from '../commands/greet'
import { GreetHandler } from '../handlers/greet';
register(GreetCommand, GreetHandler)
const bus = new Bus()
const form = useForm({
fields: {
name: {
type: FieldType.text,
label: 'Name',
modelValue: ''
}
},
buttons: {
main: {
label: 'Greetings',
onClick: sayHello
}
},
settings: {
title: 'Say Hello to'
}
})
function sayHello() {
const greetings = bus.execute(
new GreetCommand(form.fields.name.modelValue)
)
alert(greetings)
}
</script>