This .NET library provides base classes useful in projects that aims to conform to Domain Driven Design principles or projects that just wants to use the tiny types pattern.
The code is heavily inspired by the book Patterns, Principles, and Practices of Domain-Driven Design by Scott Millett.
The library defines three different base classes.
- Entity
- Value
- TinyValue
The Entity
class is the base class for entities and aggregate roots.
An entity is defined by having a unique identifier. This identifier alone makes an entity unique.
Two entities of the same type are considered to be equal if and only if they have the same identifier value. This is also true even when the other data contained within the two entities are different.
public sealed class Basket : Entity<Guid>
{
public Address InvoiceAddress { get; }
public Address? DeliveryAddress { get; }
public Basket(Guid identifier, Address invoiceAddress, Address? deliveryAddress) : base(identifier)
{
InvoiceAddress = invoiceAddress;
DeliveryAddress = deliveryAddress;
}
}
From outside the class the entity identifier is accessible via the Identifier
property.
var basket = new Basket(identifier, invoiceAddress, deliveryAddress);
var basketIdentifier = basket.Identifier;
The Equals()
method is implemented to compare only the identifiers of two entity classes of the same type.
No other members are taken into consideration when comparing entity classes.
The ==
and !=
operators are also available for comparison operations.
The Value
class is the base class for value types.
By default all fields and properties (from now on referred to a value members) in a value type, are evaluated when comparing two values of the same type. This is true no matter which access modifier the value members are decorated with.
Two values of the same type are considered to be equal if and only if all corresponding value members in each instance have the same value.
public sealed class Address : Value<Address>
{
public readonly string Street;
public readonly string HouseNumber;
public readonly string Floor;
public readonly string PostalCode;
public Address(string street, string houseNumber, string floor, string postalCode)
{
Street = street;
HouseNumber = houseNumber;
Floor = floor;
PostalCode = postalCode;
}
}
The Equals()
method is implemented to compare only included value members of two instances.
All excluded value members are skipped. More on that in the next sections.
The ==
and !=
operators are also available for comparison operations.
Annotate a value member with ExcludeAttribute to exclude it from equality operations.
public class Sushi : Value<Sushi>
{
public virtual string Name { get; }
[Exclude]
public virtual double Price { get; }
public Sushi(string name, double price)
{
Name = name;
Price = price;
}
}
In this case the property Price
is ignored in all equality operations.
var firstSalmonNirigi = new Sushi("Salmon Nigiri", 1.99);
var secondSalmonNirigi = new Sushi("Salmon Nigiri", 0.99);
var salmonNigiriPiecesAreEqual = firstSalmonNirigi == secondSalmonNirigi; // true
An excluded virtual property can be included in a child class by overriding the property and annotating it with IncludeAttribute.
public class PriceConsciousSushi : Sushi
{
[Include]
public override double Price { get; }
public PriceConsciousSushi(string name, string price) : base(name, price)
{
Price = price;
}
}
PriceConsciousSushi
will in this case behave as if the Price
property had never been excluded.
var firstSalmonNirigi = new PriceConsciousSushi("Salmon Nigiri", 1.99);
var secondSalmonNirigi = new PriceConsciousSushi("Salmon Nigiri", 0.99);
var salmonNigiriPiecesAreEqual = firstSalmonNirigi.Equals(secondSalmonNirigi); // false
If the Price
property had not been explicitly annotated with IncludeAttribute then the exclusion would have been inherited and the price would still have been ignored.
The state of value members are inherited by child classes. If a value member is excluded then it's also excluded in all classes inheriting from this class.
Properties are special value member in that they can be overridden if marked as virtual. If a property is overridden, the properties checked one by one starting from the outermost class in which the property is declared. This first explicit inclusion or exclusion encountered defines the state of that property. If non of the properties are explicitly annotated then the property will be regarded as included.
The login that applies to virtual properties also apply to abstract properties and their implementations.
Note that if a value member is annotated with both ExcludeAttribute and IncludeAttribute then the value member will be excluded since an exclude trumps an include.
Manually implemented backing fields for properties should be excluded manually. Otherwise they'll be included in equality operation.
public class Sushi : Value<Sushi>
{
[Exclude]
private readonly string _name;
public virtual string Name => _name;
public Sushi(string name)
{
_name = name;
}
}
Technically it wouldn't matter whether or not the backing field in this example was excluded, but it's important to keep in mind that all fields and properties are included in equality operations by default.
Backing fields automatically created by the compiler for properties are always ignored.
Coming soon.
Copyright © 2022 Michel Gammelgaard, Acidus. All rights reserved. Provided under the MIT license.