Today I Learned — 04/05/2022
Aggregates
Um Aggregate é um padrão em DDD onde temos um conjuntos de objetos de dominio e value objects podem ser tratados como uma unidade
Value Objects
Value objects são como o nome diz, um objeto que não tem operações associadas, ele serve somente para guardar valores, no exemplo acima o Address é um value object pois quando uma pessoa muda para a rua ao lado, mesmo que a unica parte que tenha mudado no endereço dela seja a rua, nós não falamos que ela ATUALIZOU a rua no endereço, e sim que ela MUDOU de endereço, o que no sistema se traduziria em um novo value object e não em um update dentro de um value object já existente
Services
Quando falamos em DDD o Service é uma operação sem estado que cumpre uma tarefa especifica do domínio. A melhor indicação que devemos criar um service é quando queremos fazer uma operação que não parece fazer sentido como o método em um Agreggate ou um Value Object.
Cuidados
- Caso existam muitos services no projeto isso pode indicar que os aggregates estão anêmicos
- Services são Stateless, O que significa que eles não carregam valores de uma chamada para a outra
Implementação
Para o mesmo sistema demonstrado acima podemos ter por exemplo uma entidade de orders que contem items como value object
export class Order{
private _id: string;
private _customerId: string;
private _items: OrderItem[];
private _total: number;
constructor(id: string, customerId: string, items: OrderItem[]){
this._id = id;
this._customerId = customerId;
this._items = items;
this._total = this.total();
}
total(): number{
return this._items.reduce((acc, item) => acc + item.price, 0)
}
}
Nessa classe de order temos um método para retornar o total dá order em especifico mas e se quisermos calcular o total de todas as orders? Não faz sentido ficar como método de uma order já q esse método vai precisar receber outras orders para fazer a conta, é nesse caso que devemos criar o service de Orders
export class OrderService{
static total(orders: Order[]): number {
return orders.reduce((acc, order) => acc + order.total(), 0);
}
}
O OrderService pode conter todas os métodos que não fazem sentido ficarem dentro de Order inclusive por exemplo criar uma nova order
export class OrderService{
static total(orders: Order[]): number {
return orders.reduce((acc, order) => acc + order.total(), 0);
}
static placeOrder(customer: Customer, items: OrderItem[]): Order{
if(items.length === 0){
throw new Error("Order must have at least one item")
}
const order = new Order(uuid(), customer.id, items);
return order
}
}
Ainda estou aprendendo sobre DDD e se quiser ver melhor o código pode dar uma olhada nesse Repositório
Top comments (0)