Vogen 1.0.1
Vogen
What is the package?
This is a semi-opinionated library which generates Value Objects that wrap simple primitives such as int, string, double etc. The main goal of this project is to have almost the same speed and memory performance as using primitives.
Primitive Obsession means being obsessed with primitives. It is a Code Smell that degrades the quality of software.
"Primitive Obsession is using primitive data types to represent domain ideas" #
Some examples:
- instead of
int age- we'd haveAge age.Agemight have validation that it couldn't be negative - instead of
string postcode- we'd havePostcode postcode.Postcodemight have validation on the format of the text
The opinions are expressed as:
- A Value Object (VO) is constructed via a factory method named
From, e.g.Age.From(12) - A VO is equatable (
Age.From(12) == Age.From(12)) - A VO, if validated, is validated with a private static method named
Validatethat returns aValidationresult - Any validation that is not
Validation.Okresults in aValueObjectValidationExceptionbeing thrown
Instead of
int customerId = 42;
... we'd have
var customerId = CustomerId.From(42);
CustomerId is declared as:
[ValueObject(typeof(int))]
public partial struct CustomerId
{
}
That's all you need to do to switch from a primitive to a Value Object.
Here it is again with some validation
[ValueObject(typeof(int))]
public partial struct CustomerId
{
private static Validation Validate(int value) =>
value > 0 ? Validation.Ok : Validation.Invalid();
}
This allows us to have more strongly typed domain objects instead of primitives, which makes the code easier to read and enforces better method signatures, so instead of:
public void DoSomething(int customerId, int supplierId, int amount)
we can have:
public void DoSomething(CustomerId customerId, SupplierId supplierId, Amount amount)
Now, callers can't mess up the ordering of parameters and accidentally pass us a Supplier ID in place of a Customer ID.
It also means that validation is in just one place. You can't introduce bad objects into your domain, therefore you can assume that in your domain every ValueObject is valid. Handy.
How does it compare to using native types?
Here's the benchmarks comparing a native int to a ValueObject:
| Method | Mean | Error | StdDev | Ratio | Gen 0 | Gen 1 | Gen 2 | Allocated |
|------------------------ |---------:|---------:|---------:|------:|-------:|------:|------:|----------:|
| UsingIntNatively | 17.04 ns | 0.253 ns | 0.014 ns | 1.00 | - | - | - | - |
| UsingValueObjectStruct | 19.76 ns | 2.463 ns | 0.135 ns | 1.16 | - | - | - | - |
There's hardly any speed overhead, and no memory overhead.
The next most common scenario is using a VO class to represent a native `String`. These results are:
```ini
| Method | Mean | Error | StdDev | Ratio | Gen 0 | Gen 1 | Gen 2 | Allocated |
|------------------------- |---------:|---------:|--------:|------:|-------:|------:|------:|----------:|
| UsingStringNatively | 204.4 ns | 8.09 ns | 0.44 ns | 1.00 | 0.0153 | - | - | 256 B |
| UsingValueObjectAsClass | 250.7 ns | 29.97 ns | 1.64 ns | 1.23 | 0.0196 | - | - | 328 B |
| UsingValueObjectAsStruct | 248.9 ns | 18.82 ns | 1.03 ns | 1.22 | 0.0181 | - | - | 304 B |
No packages depend on Vogen.
.NET Standard 2.0
- No dependencies.
| Version | Downloads | Last updated |
|---|---|---|
| 8.0.5-beta.1 | 4 | 02/03/2026 |
| 8.0.4 | 4 | 02/03/2026 |
| 8.0.4-beta.2 | 3 | 02/03/2026 |
| 8.0.4-beta.1 | 5 | 02/03/2026 |
| 8.0.3 | 4 | 02/03/2026 |
| 8.0.3-beta.2 | 4 | 02/03/2026 |
| 8.0.3-beta.1 | 3 | 02/03/2026 |
| 8.0.2 | 1 | 02/03/2026 |
| 8.0.1 | 4 | 02/03/2026 |
| 8.0.0-beta.2 | 4 | 02/03/2026 |
| 8.0.0-beta.1 | 4 | 02/03/2026 |
| 7.0.5-beta.1 | 4 | 02/03/2026 |
| 7.0.4 | 4 | 02/03/2026 |
| 7.0.4-beta.1 | 4 | 02/03/2026 |
| 7.0.3 | 5 | 02/02/2026 |
| 7.0.2 | 2 | 02/03/2026 |
| 7.0.1 | 3 | 02/03/2026 |
| 7.0.0 | 4 | 02/03/2026 |
| 7.0.0-beta.1 | 4 | 02/03/2026 |
| 6.0.0 | 4 | 02/03/2026 |
| 6.0.0-beta.4 | 4 | 02/03/2026 |
| 6.0.0-beta.3 | 4 | 02/03/2026 |
| 6.0.0-beta.2 | 4 | 02/03/2026 |
| 6.0.0-beta.1 | 4 | 02/03/2026 |
| 5.0.6 | 3 | 02/03/2026 |
| 5.0.6-beta.3 | 3 | 02/03/2026 |
| 5.0.6-beta.2 | 2 | 02/03/2026 |
| 5.0.6-beta.1 | 2 | 02/03/2026 |
| 5.0.5 | 3 | 02/03/2026 |
| 5.0.5-rc.2 | 2 | 02/03/2026 |
| 5.0.5-rc.1 | 3 | 02/03/2026 |
| 5.0.5-beta.3 | 3 | 02/03/2026 |
| 5.0.5-beta.2 | 3 | 02/03/2026 |
| 5.0.4-beta.1 | 5 | 02/03/2026 |
| 5.0.3 | 2 | 02/03/2026 |
| 5.0.2 | 4 | 02/03/2026 |
| 5.0.1 | 4 | 02/03/2026 |
| 5.0.0-beta.1 | 4 | 02/03/2026 |
| 4.0.19 | 4 | 02/03/2026 |
| 4.0.18 | 2 | 02/03/2026 |
| 4.0.17 | 3 | 02/03/2026 |
| 4.0.16 | 4 | 02/03/2026 |
| 4.0.15 | 4 | 02/03/2026 |
| 4.0.14 | 4 | 02/03/2026 |
| 4.0.13 | 4 | 02/03/2026 |
| 4.0.12 | 3 | 02/03/2026 |
| 4.0.11 | 4 | 02/03/2026 |
| 4.0.10 | 2 | 02/03/2026 |
| 4.0.9 | 4 | 02/03/2026 |
| 4.0.8 | 2 | 02/03/2026 |
| 4.0.7 | 4 | 02/03/2026 |
| 4.0.6 | 2 | 02/03/2026 |
| 4.0.5 | 2 | 02/03/2026 |
| 4.0.4 | 4 | 02/03/2026 |
| 4.0.3 | 4 | 02/03/2026 |
| 4.0.2 | 2 | 02/03/2026 |
| 4.0.1 | 4 | 02/03/2026 |
| 4.0.0 | 4 | 02/03/2026 |
| 3.0.25-beta.1 | 2 | 02/03/2026 |
| 3.0.24 | 4 | 02/03/2026 |
| 3.0.23 | 4 | 02/03/2026 |
| 3.0.23-beta.2 | 4 | 02/03/2026 |
| 3.0.23-beta.1 | 3 | 02/03/2026 |
| 3.0.22 | 4 | 02/03/2026 |
| 3.0.21 | 4 | 02/03/2026 |
| 3.0.20 | 4 | 02/03/2026 |
| 3.0.19 | 4 | 02/03/2026 |
| 3.0.18 | 4 | 02/03/2026 |
| 3.0.17 | 2 | 02/03/2026 |
| 3.0.16 | 3 | 02/03/2026 |
| 3.0.15 | 1 | 02/03/2026 |
| 3.0.14 | 4 | 02/03/2026 |
| 3.0.13 | 4 | 02/03/2026 |
| 3.0.12 | 3 | 02/03/2026 |
| 3.0.11 | 4 | 02/03/2026 |
| 3.0.10 | 2 | 02/03/2026 |
| 3.0.9 | 5 | 02/03/2026 |
| 3.0.8 | 3 | 02/03/2026 |
| 3.0.7 | 2 | 02/03/2026 |
| 3.0.6 | 3 | 02/03/2026 |
| 3.0.5 | 3 | 02/03/2026 |
| 3.0.4 | 2 | 02/03/2026 |
| 3.0.3 | 2 | 02/03/2026 |
| 3.0.2-alpha.2 | 4 | 02/03/2026 |
| 3.0.2-alpha | 4 | 02/03/2026 |
| 3.0.1 | 4 | 02/03/2026 |
| 2.0.5 | 2 | 02/03/2026 |
| 2.0.4 | 4 | 02/03/2026 |
| 2.0.3 | 4 | 02/03/2026 |
| 2.0.2 | 2 | 02/03/2026 |
| 1.0.25 | 3 | 02/03/2026 |
| 1.0.24 | 3 | 02/03/2026 |
| 1.0.23 | 4 | 02/03/2026 |
| 1.0.22 | 3 | 02/03/2026 |
| 1.0.21 | 4 | 02/03/2026 |
| 1.0.20 | 4 | 02/03/2026 |
| 1.0.19 | 3 | 02/03/2026 |
| 1.0.18 | 4 | 02/03/2026 |
| 1.0.17 | 2 | 02/03/2026 |
| 1.0.16-alpha.1 | 4 | 02/03/2026 |
| 1.0.15 | 2 | 02/03/2026 |
| 1.0.12 | 1 | 02/03/2026 |
| 1.0.11 | 3 | 02/03/2026 |
| 1.0.9 | 4 | 02/03/2026 |
| 1.0.8 | 4 | 02/03/2026 |
| 1.0.7 | 4 | 02/03/2026 |
| 1.0.6 | 4 | 02/03/2026 |
| 1.0.5 | 3 | 02/03/2026 |
| 1.0.4 | 4 | 02/03/2026 |
| 1.0.3 | 3 | 02/03/2026 |
| 1.0.2 | 4 | 02/03/2026 |
| 1.0.1 | 4 | 02/03/2026 |
| 1.0.1-alpha.0.2 | 4 | 02/03/2026 |
| 0.0.0-alpha.0 | 2 | 02/03/2026 |