Sometimes you need to build up an existing composition root and inject all of its dependencies, in which case the Builder
method will be useful, as in the example below:
using Shouldly;
using Pure.DI;
DI.Setup(nameof(Composition))
.Bind().To(_ => Guid.NewGuid())
.Bind().To<Dependency>()
.Builder<Service>("BuildUpService");
var composition = new Composition();
var service = composition.BuildUpService(new Service());
service.Id.ShouldNotBe(Guid.Empty);
service.Dependency.ShouldBeOfType<Dependency>();
interface IDependency;
class Dependency : IDependency;
interface IService
{
Guid Id { get; }
IDependency? Dependency { get; }
}
record Service: IService
{
public Guid Id { get; private set; } = Guid.Empty;
// The Dependency attribute specifies to perform an injection
[Dependency]
public IDependency? Dependency { get; set; }
[Dependency]
public void SetId(Guid id) => Id = id;
}
Running this code sample locally
- Make sure you have the .NET SDK 9.0 or later is installed
dotnet --list-sdk
- Create a net9.0 (or later) console application
dotnet new console -n Sample
dotnet add package Pure.DI
dotnet add package Shouldly
- Copy the example code into the Program.cs file
You are ready to run the example 🚀
dotnet run
Important Notes:
- The default builder method name is
BuildUp
- The first argument to the builder method is always the instance to be built
Advantages:
- Allows working with pre-existing objects
- Provides flexibility in dependency injection
- Supports partial injection of dependencies
- Can be used with legacy code
Use Cases:
- When objects are created outside the DI container
- For working with third-party libraries
- When migrating existing code to DI
- For complex object graphs where full construction is not feasible
The following partial class will be generated:
partial class Composition
{
private readonly Composition _root;
[OrdinalAttribute(128)]
public Composition()
{
_root = this;
}
internal Composition(Composition parentScope)
{
_root = (parentScope ?? throw new ArgumentNullException(nameof(parentScope)))._root;
}
[MethodImpl(MethodImplOptions.AggressiveInlining)]
public Service BuildUpService(Service buildingInstance)
{
if (buildingInstance is null) throw new ArgumentNullException(nameof(buildingInstance));
Guid transientGuid2 = Guid.NewGuid();
Service transientService0;
Service localBuildingInstance86 = buildingInstance;
localBuildingInstance86.Dependency = new Dependency();
localBuildingInstance86.SetId(transientGuid2);
transientService0 = localBuildingInstance86;
return transientService0;
}
}
Class diagram:
---
config:
class:
hideEmptyMembersBox: true
---
classDiagram
Dependency --|> IDependency
Composition ..> Service : Service BuildUpService(Pure.DI.UsageTests.Basics.BuilderScenario.Service buildingInstance)
Service *-- Guid : Guid
Service *-- Dependency : IDependency
namespace Pure.DI.UsageTests.Basics.BuilderScenario {
class Composition {
<<partial>>
+Service BuildUpService(Pure.DI.UsageTests.Basics.BuilderScenario.Service buildingInstance)
}
class Dependency {
+Dependency()
}
class IDependency {
<<interface>>
}
class Service {
<<record>>
+IDependency Dependency
+SetId(Guid id) : Void
}
}
namespace System {
class Guid {
<<struct>>
}
}