Partilhar via


Seedwork (classes base reutilizáveis e interfaces para o seu modelo de domínio)

Gorjeta

Este conteúdo é um trecho do eBook, .NET Microservices Architecture for Containerized .NET Applications, disponível no .NET Docs ou como um PDF para download gratuito que pode ser lido offline.

.NET Microservices Architecture for Containerized .NET Applications eBook cover thumbnail.

A pasta da solução contém uma pasta SeedWork . Esta pasta contém classes base personalizadas que você pode usar como base para suas entidades de domínio e objetos de valor. Use essas classes base para não ter código redundante na classe de objeto de cada domínio. A pasta para esses tipos de classes é chamada SeedWork e não algo como Framework. É chamado SeedWork porque a pasta contém apenas um pequeno subconjunto de classes reutilizáveis que não podem realmente ser consideradas uma estrutura. Seedwork é um termo introduzido por Michael Feathers e popularizado por Martin Fowler , mas você também pode nomear essa pasta como Common, SharedKernel, ou similar.

A Figura 7-12 mostra as classes que formam o seedwork do modelo de domínio no microsserviço de ordenação. Ele tem algumas classes base personalizadas como Entity, ValueObject e Enumeration, além de algumas interfaces. Essas interfaces (IRepository e IUnitOfWork) informam a camada de infraestrutura sobre o que precisa ser implementado. Essas interfaces também são usadas por meio da injeção de dependência da camada de aplicativo.

Screenshot of the classes contained in the SeedWork folder.

O conteúdo detalhado da pasta SeedWork, contendo classes base e interfaces: Entity.cs, Enumeration.cs, IAggregateRoot.cs, IRepository.cs, IUnitOfWork.cs e ValueObject.cs.

Figura 7-12. Um conjunto de exemplo de classes base e interfaces "seedwork" do modelo de domínio

Este é o tipo de reutilização de copiar e colar que muitos desenvolvedores compartilham entre projetos, não uma estrutura formal. Você pode ter seedworks em qualquer camada ou biblioteca. No entanto, se o conjunto de classes e interfaces ficar grande o suficiente, convém criar uma única biblioteca de classes.

A classe base Entity personalizada

O código a seguir é um exemplo de uma classe base Entity onde você pode colocar código que pode ser usado da mesma maneira por qualquer entidade de domínio, como o ID da entidade, operadores de igualdade, uma lista de eventos de domínio por entidade, etc.

// COMPATIBLE WITH ENTITY FRAMEWORK CORE (1.1 and later)
public abstract class Entity
{
    int? _requestedHashCode;
    int _Id;
    private List<INotification> _domainEvents;
    public virtual int Id
    {
        get
        {
            return _Id;
        }
        protected set
        {
            _Id = value;
        }
    }

    public List<INotification> DomainEvents => _domainEvents;
    public void AddDomainEvent(INotification eventItem)
    {
        _domainEvents = _domainEvents ?? new List<INotification>();
        _domainEvents.Add(eventItem);
    }
    public void RemoveDomainEvent(INotification eventItem)
    {
        if (_domainEvents is null) return;
        _domainEvents.Remove(eventItem);
    }

    public bool IsTransient()
    {
        return this.Id == default(Int32);
    }

    public override bool Equals(object obj)
    {
        if (obj == null || !(obj is Entity))
            return false;
        if (Object.ReferenceEquals(this, obj))
            return true;
        if (this.GetType() != obj.GetType())
            return false;
        Entity item = (Entity)obj;
        if (item.IsTransient() || this.IsTransient())
            return false;
        else
            return item.Id == this.Id;
    }

    public override int GetHashCode()
    {
        if (!IsTransient())
        {
            if (!_requestedHashCode.HasValue)
                _requestedHashCode = this.Id.GetHashCode() ^ 31;
            // XOR for random distribution. See:
            // https://learn.microsoft.com/archive/blogs/ericlippert/guidelines-and-rules-for-gethashcode
            return _requestedHashCode.Value;
        }
        else
            return base.GetHashCode();
    }
    public static bool operator ==(Entity left, Entity right)
    {
        if (Object.Equals(left, null))
            return (Object.Equals(right, null));
        else
            return left.Equals(right);
    }
    public static bool operator !=(Entity left, Entity right)
    {
        return !(left == right);
    }
}

O código anterior usando uma lista de eventos de domínio por entidade será explicado nas próximas seções ao se concentrar em eventos de domínio.

Contratos de repositório (interfaces) na camada de modelo de domínio

Os contratos de repositório são simplesmente interfaces .NET que expressam os requisitos contratuais dos repositórios a serem usados para cada agregado.

Os repositórios em si, com código EF Core ou quaisquer outras dependências de infraestrutura e código (Linq, SQL, etc.), não devem ser implementados dentro do modelo de domínio; Os repositórios só devem implementar as interfaces definidas no modelo de domínio.

Um padrão relacionado a essa prática (colocar as interfaces do repositório na camada do modelo de domínio) é o padrão Interface separada. Como explicado por Martin Fowler, "Use a interface separada para definir uma interface em um pacote, mas implementá-la em outro. Desta forma, um cliente que precisa da dependência da interface pode ignorar completamente a implementação."

Seguir o padrão Interface separada permite que a camada de aplicativo (neste caso, o projeto de API da Web para o microsserviço) tenha uma dependência dos requisitos definidos no modelo de domínio, mas não uma dependência direta com a camada de infraestrutura/persistência. Além disso, você pode usar a injeção de dependência para isolar a implementação, que é implementada na camada de infraestrutura/persistência usando repositórios.

Por exemplo, o exemplo a seguir com a interface IOrderRepository define quais operações a classe OrderRepository precisará implementar na camada de infraestrutura. Na implementação atual do aplicativo, o código só precisa adicionar ou atualizar ordens ao banco de dados, uma vez que as consultas são divididas seguindo a abordagem CQRS simplificada.

// Defined at IOrderRepository.cs
public interface IOrderRepository : IRepository<Order>
{
    Order Add(Order order);

    void Update(Order order);

    Task<Order> GetAsync(int orderId);
}

// Defined at IRepository.cs (Part of the Domain Seedwork)
public interface IRepository<T> where T : IAggregateRoot
{
    IUnitOfWork UnitOfWork { get; }
}

Recursos adicionais