Tests | |
Examples | |
Performance |
Supports .NET starting with .NET Framework 2.0, released 2005-10-27, and all newer versions.
-
.NET SDK 6.0.4 or later is installed. At the same time, you can develop .NET projects even for older versions like .NET Framework 2.0
-
C# 8 or later. This requirement only needs to be met for projects that reference the Pure.DI source code generator, other projects can use any version of C#.
Pure.DI is not a framework or library, but a C# source code generator for creating object compositions in the Pure DI paradigm. To make them accurate, the developer uses a set of intuitive hints from the Pure.DI API. At the compilation stage, Pure.DI determines the optimal graph structure, checks its correctness, and generates partial class code to create object compositions using only basic language constructs. The resulting code is robust, works everywhere, does not create exceptions, does not depend on .NET library calls or .NET reflections, is efficient in terms of performance and memory consumption, and is ready for all optimizations. This code can be easily integrated into an application as it does not use unnecessary delegates, additional calls to any methods, type conversions, boxing/unboxing, etc.
- DI without any IoC/DI containers, frameworks, dependencies and hence no performance impact or side effects.
Pure.DI is actually a .NET code generator. It uses basic language constructs to create simple code as well as if you were doing it yourself: de facto it's just a bunch of nested constructor calls. This code can be viewed, analyzed at any time, and debugged.
- A predictable and verified dependency graph is built and validated on the fly while writing code.
All logic for analyzing the graph of objects, constructors and methods takes place at compile time. Pure.DI notifies the developer at compile time of missing or cyclic dependencies, cases when some dependencies are not suitable for injection, etc. The developer has no chance to get a program that will crash at runtime because of some exception related to incorrect object graph construction. All this magic happens at the same time as the code is written, so you have instant feedback between the fact that you have made changes to your code and the fact that your code is already tested and ready to use.
- Does not add any dependencies to other assemblies.
When using pure DI, no dependencies are added to assemblies because only basic language constructs and nothing more are used.
- Highest performance, including compiler and JIT optimization and minimal memory consumption.
All generated code runs as fast as your own, in pure DI style, including compile-time and run-time optimization. As mentioned above, graph analysis is done at compile time, and at runtime there are only a bunch of nested constructors, and that's it. Memory is spent only on the object graph being created.
- It works everywhere.
Since the pure DI approach does not use any dependencies or .NET reflection at runtime, it does not prevent the code from running as expected on any platform: Full .NET Framework 2.0+, .NET Core, .NET, UWP/XBOX, .NET IoT, Xamarin, Native AOT, etc.
- Ease of Use.
The Pure.DI API is very similar to the API of most IoC/DI libraries. And this was a conscious decision: the main reason is that programmers don't need to learn a new API.
- Superfine customization of generic types.
In Pure.DI it is proposed to use special marker types instead of using open generic types. This allows you to build the object graph more accurately and take full advantage of generic types.
- Supports the major .NET BCL types out of the box.
Pure.DI already supports many of BCL types like
Array
,IEnumerable<T>
,IList<T>
,IReadOnlyCollection<T>
,IReadOnlyList<T>
,ISet<T>
,IProducerConsumerCollection<T>
,ConcurrentBag<T>
,Func<T>
,ThreadLocal
,ValueTask<T>
,Task<T>
,MemoryPool<T>
,ArrayPool<T>
,ReadOnlyMemory<T>
,Memory<T>
,ReadOnlySpan<T>
,Span<T>
,IComparer<T>
,IEqualityComparer<T>
and etc. without any extra effort. - Good for building libraries or frameworks where resource consumption is particularly critical.
Its high performance, zero memory consumption/preparation overhead, and lack of dependencies make it ideal for building libraries and frameworks.
interface IBox<out T>
{
T Content { get; }
}
interface ICat
{
State State { get; }
}
enum State { Alive, Dead }
record CardboardBox<T>(T Content): IBox<T>;
class ShroedingersCat(Lazy<State> superposition): ICat
{
// The decoherence of the superposition
// at the time of observation via an irreversible process
public State State => superposition.Value;
}
Important
Our abstraction and implementation knows nothing about the magic of DI or any frameworks.
Add the Pure.DI package to your project:
Let's bind the abstractions to their implementations and set up the creation of the object graph:
DI.Setup(nameof(Composition))
// Models a random subatomic event that may or may not occur
.Bind().As(Singleton).To<Random>()
// Quantum superposition of two states: Alive or Dead
.Bind().To((Random random) => (State)random.Next(2))
.Bind().To<ShroedingersCat>()
// Cardboard box with any contents
.Bind().To<CardboardBox<TT>>()
// Composition Root
.Root<Program>("Root");
Note
In fact, the Bind().As(Singleton).To<Random>()
binding is unnecessary since Pure.DI supports many .NET BCL types out of the box, including Random. It was added just for the example of using the Singleton lifetime.
The above code specifies the generation of a partial class named Composition, this name is defined in the DI.Setup(nameof(Composition))
call. This class contains a Root property that returns a graph of objects with an object of type Program as the root. The type and name of the property is defined by calling Root<Program>("Root")
. The code of the generated class looks as follows:
partial class Composition
{
private Lock _lock = new Lock();
private Random? _random;
public Program Root
{
[MethodImpl(MethodImplOptions.AggressiveInlining)]
get
{
var stateFunc = new Func<State>(() => {
if (_random is null)
using (_lock.EnterScope())
if (_random is null)
_random = new Random();
return (State)_random.Next(2)
});
return new Program(
new CardboardBox<ICat>(
new ShroedingersCat(
new Lazy<State>(
stateFunc))));
}
}
public T Resolve<T>() { ... }
public object Resolve(Type type) { ... }
}
Obviously, this code does not depend on other libraries, does not use type reflection or any other tricks that can negatively affect performance and memory consumption. It looks like an efficient code written by hand. At any given time, you can study it and understand how it works.
The public Program Root { get; }
property here is a Composition Root, the only place in the application where the composition of the object graph for the application takes place. Each instance is created by only basic language constructs, which compiles with all optimizations with minimal impact on performance and memory consumption. In general, applications may have multiple composition roots and thus such properties. Each composition root must have its own unique name, which is defined when the Root<T>(string name)
method is called, as shown in the above code.
class Program(IBox<ICat> box)
{
// Composition Root, a single place in an application
// where the composition of the object graphs
// for an application take place
static void Main() => new Composition().Root.Run();
private void Run() => Console.WriteLine(box);
}
Pure.DI creates efficient code in a pure DI paradigm, using only basic language constructs as if you were writing code by hand. This allows you to take full advantage of Dependency Injection everywhere and always, without any compromise!
The full analog of this application with top-level statements can be found here.
Just try creating a project from scratch!
Install the projects template
dotnet new install Pure.DI.Templates
In some directory, create a console application
dotnet new di
And run it
dotnet run
- Auto-bindings
- Injections of abstractions
- Composition roots
- Resolve methods
- Simplified binding
- Factory
- Simplified factory
- Class arguments
- Root arguments
- Tags
- Smart tags
- Build up of an existing object
- Builders
- Builders with arguments
- Field injection
- Method injection
- Property injection
- Default values
- Required properties or fields
- Root binding
- Async Root
- Consumer types
- Transient
- Singleton
- PerResolve
- PerBlock
- Scope
- Auto scoped
- Default lifetime
- Default lifetime for a type
- Default lifetime for a type and a tag
- Disposable singleton
- Async disposable singleton
- Async disposable scope
- Func
- Enumerable
- Enumerable generics
- Array
- Lazy
- Task
- ValueTask
- Manually started tasks
- Span and ReadOnlySpan
- Tuple
- Weak Reference
- Async Enumerable
- Service collection
- Func with arguments
- Func with tag
- Keyed service provider
- Service provider
- Service provider with scope
- Overriding the BCL binding
- Generics
- Generic composition roots
- Complex generics
- Generic composition roots with constraints
- Generic async composition roots with constraints
- Custom generic argument
- Build up of an existing generic object
- Generic root arguments
- Complex generic root arguments
- Generic builders
- Constructor ordinal attribute
- Dependency attribute
- Member ordinal attribute
- Tag attribute
- Type attribute
- Inject attribute
- Custom attributes
- Custom universal attribute
- Custom generic argument attribute
- Bind attribute
- Bind attribute with lifetime and tag
- Bind attribute for a generic type
- Resolve hint
- ThreadSafe hint
- OnDependencyInjection regular expression hint
- OnDependencyInjection wildcard hint
- OnCannotResolve regular expression hint
- OnCannotResolve wildcard hint
- OnNewInstance regular expression hint
- OnNewInstance wildcard hint
- ToString hint
- Check for a root
- Composition root kinds
- Tag Type
- Tag Unique
- Tag on injection site
- Tag on a constructor argument
- Tag on a member
- Tag on a method argument
- Tag on injection site with wildcards
- Dependent compositions
- Accumulators
- Global compositions
- Partial class
- A few partial classes
- Tracking disposable instances per a composition root
- Tracking disposable instances in delegates
- Tracking disposable instances using pre-built classes
- Tracking disposable instances with different lifetimes
- Tracking async disposable instances per a composition root
- Tracking async disposable instances in delegates
- Exposed roots
- Exposed roots with tags
- Exposed roots via arg
- Exposed roots via root arg
- Exposed generic roots
- Exposed generic roots with args
- DI tracing via serilog
- Console
- Unity
- UI
- Web
- Git repo with examples
Each generated class, hereafter called a composition, must be customized. Setup starts with a call to the Setup(string compositionTypeName)
method:
DI.Setup("Composition")
.Bind<IDependency>().To<Dependency>()
.Bind<IService>().To<Service>()
.Root<IService>("Root");
The following class will be generated
partial class Composition
{
// Default constructor
public Composition() { }
// Scope constructor
internal Composition(Composition parentScope) { }
// Composition root
public IService Root
{
get
{
return new Service(new Dependency());
}
}
public T Resolve<T>() { ... }
public T Resolve<T>(object? tag) { ... }
public object Resolve(Type type) { ... }
public object Resolve(Type type, object? tag) { ... }
}
The compositionTypeName parameter can be omitted
- if the setup is performed inside a partial class, then the composition will be created for this partial class
- for the case of a class with composition kind
CompositionKind.Global
, see this example
Setup arguments
The first parameter is used to specify the name of the composition class. All sets with the same name will be combined to create one composition class. Alternatively, this name may contain a namespace, e.g. a composition class is generated for Sample.Composition
:
namespace Sample
{
partial class Composition
{
...
}
}
The second optional parameter may have multiple values to determine the kind of composition.
This value is used by default. If this value is specified, a normal composition class will be created.
If you specify this value, the class will not be generated, but this setup can be used by others as a base setup. For example:
DI.Setup("BaseComposition", CompositionKind.Internal)
.Bind().To<Dependency>();
DI.Setup("Composition").DependsOn("BaseComposition")
.Bind().To<Service>();
If the CompositionKind.Public flag is set in the composition setup, it can also be the base for other compositions, as in the example above.
No composition class will be created when this value is specified, but this setup is the base setup for all setups in the current project, and DependsOn(...)
is not required.
Constructors
It's quite trivial, this constructor simply initializes the internal state.
It replaces the default constructor and is only created if at least one argument is specified. For example:
DI.Setup("Composition")
.Arg<string>("name")
.Arg<int>("id")
...
In this case, the constructor with arguments is as follows:
public Composition(string name, int id) { ... }
and there is no default constructor. It is important to remember that only those arguments that are used in the object graph will appear in the constructor. Arguments that are not involved cannot be defined, as they are omitted from the constructor parameters to save resources.
This constructor creates a composition instance for the new scope. This allows Lifetime.Scoped
to be applied. See this example for details.
Composition Roots
To create an object graph quickly and conveniently, a set of properties (or a methods) is formed. These properties/methods are here called roots of compositions. The type of a property/method is the type of the root object created by the composition. Accordingly, each invocation of a property/method leads to the creation of a composition with a root element of this type.
DI.Setup("Composition")
.Bind<IService>().To<Service>()
.Root<IService>("MyService");
In this case, the property for the IService type will be named MyService and will be available for direct use. The result of its use will be the creation of a composition of objects with the root of IService type:
public IService MyService
{
get
{
...
return new Service(...);
}
}
This is recommended way to create a composition root. A composition class can contain any number of roots.
If the root name is empty, a private composition root with a random name is created:
private IService RootM07D16di_0001
{
get { ... }
}
This root is available in Resolve methods in the same way as public roots. For example:
DI.Setup("Composition")
.Bind<IService>().To<Service>()
.Root<IService>();
These properties have an arbitrary name and access modifier private and cannot be used directly from the code. Do not attempt to use them, as their names are arbitrarily changed. Private composition roots can be resolved by Resolve methods.
Methods "Resolve"
By default, a set of four Resolve methods is generated:
public T Resolve<T>() { ... }
public T Resolve<T>(object? tag) { ... }
public object Resolve(Type type) { ... }
public object Resolve(Type type, object? tag) { ... }
These methods can resolve both public and private composition roots that do not depend on any arguments of the composition roots. They are useful when using the Service Locator approach, where the code resolves composition roots in place:
var composition = new Composition();
composition.Resolve<IService>();
This is a not recommended way to create composition roots because Resolve methods have a number of disadvantages:
- They provide access to an unlimited set of dependencies.
- Their use can potentially lead to runtime exceptions, for example, when the corresponding root has not been defined.
- Lead to performance degradation because they search for the root of a composition based on its type.
To control the generation of these methods, see the Resolve hint.
Provides a mechanism to release unmanaged resources. These methods are generated only if the composition contains at least one singleton/scoped instance that implements either the IDisposable and/or DisposeAsync interface. The Dispose()
or DisposeAsync()
method of the composition should be called to dispose of all created singleton/scoped objects:
using var composition = new Composition();
or
await using var composition = new Composition();
To dispose objects of other lifetimes please see this or this examples.
Setup hints
Hints are used to fine-tune code generation. Setup hints can be used as shown in the following example:
DI.Setup("Composition")
.Hint(Hint.Resolve, "Off")
.Hint(Hint.ThreadSafe, "Off")
.Hint(Hint.ToString, "On")
...
In addition, setup hints can be commented out before the Setup method as hint = value
. For example:
// Resolve = Off
// ThreadSafe = Off
DI.Setup("Composition")
...
Both approaches can be mixed:
// Resolve = Off
DI.Setup("Composition")
.Hint(Hint.ThreadSafe, "Off")
...
The list of hints will be gradually expanded to meet the needs and desires for fine-tuning code generation. Please feel free to add your ideas.
Determines whether to generate Resolve methods. By default, a set of four Resolve methods are generated. Set this hint to Off to disable the generation of resolve methods. This will reduce the generation time of the class composition, and in this case no private composition roots will be generated. The class composition will be smaller and will only have public roots. When the Resolve hint is disabled, only the public roots properties are available, so be sure to explicitly define them using the Root<T>(string name)
method with an explicit composition root name.
Determines whether to use the OnNewInstance partial method. By default, this partial method is not generated. This can be useful, for example, for logging purposes:
internal partial class Composition
{
partial void OnNewInstance<T>(ref T value, object? tag, object lifetime) =>
Console.WriteLine($"'{typeof(T)}'('{tag}') created.");
}
You can also replace the created instance with a T
type, where T
is the actual type of the created instance. To minimize performance loss when calling OnNewInstance, use the three hints below.
Determines whether to generate the OnNewInstance partial method. By default, this partial method is generated when the OnNewInstance hint is On
.
This is a regular expression for filtering by instance type name. This hint is useful when OnNewInstance is in On state and it is necessary to limit the set of types for which the OnNewInstance method will be called.
This is a Wildcard for filtering by instance type name. This hint is useful when OnNewInstance is in On state and it is necessary to limit the set of types for which the OnNewInstance method will be called.
This is a regular expression for filtering by tag. This hint is also useful when OnNewInstance is in On state and it is necessary to limit the set of tags for which the OnNewInstance method will be called.
This is a wildcard for filtering by tag. This hint is also useful when OnNewInstance is in On state and it is necessary to limit the set of tags for which the OnNewInstance method will be called.
This is a regular expression for filtering by lifetime. This hint is also useful when OnNewInstance is in On state and it is necessary to restrict the set of life times for which the OnNewInstance method will be called.
This is a wildcard for filtering by lifetime. This hint is also useful when OnNewInstance is in On state and it is necessary to restrict the set of life times for which the OnNewInstance method will be called.
Determines whether to use the OnDependencyInjection partial method when the OnDependencyInjection hint is On
to control dependency injection. By default it is On
.
// OnDependencyInjection = On
// OnDependencyInjectionPartial = Off
// OnDependencyInjectionContractTypeNameRegularExpression = ICalculator[\d]{1}
// OnDependencyInjectionTagRegularExpression = Abc
DI.Setup("Composition")
...
Determines whether to generate the OnDependencyInjection partial method to control dependency injection. By default, this partial method is not generated. It cannot have an empty body because of the return value. It must be overridden when it is generated. This may be useful, for example, for Interception Scenario.
// OnDependencyInjection = On
// OnDependencyInjectionContractTypeNameRegularExpression = ICalculator[\d]{1}
// OnDependencyInjectionTagRegularExpression = Abc
DI.Setup("Composition")
...
To minimize performance loss when calling OnDependencyInjection, use the three tips below.
This is a regular expression for filtering by instance type name. This hint is useful when OnDependencyInjection is in On state and it is necessary to restrict the set of types for which the OnDependencyInjection method will be called.
This is a wildcard for filtering by instance type name. This hint is useful when OnDependencyInjection is in On state and it is necessary to restrict the set of types for which the OnDependencyInjection method will be called.
This is a regular expression for filtering by the name of the resolving type. This hint is also useful when OnDependencyInjection is in On state and it is necessary to limit the set of permissive types for which the OnDependencyInjection method will be called.
This is a wildcard for filtering by the name of the resolving type. This hint is also useful when OnDependencyInjection is in On state and it is necessary to limit the set of permissive types for which the OnDependencyInjection method will be called.
This is a regular expression for filtering by tag. This hint is also useful when OnDependencyInjection is in the On state and you want to limit the set of tags for which the OnDependencyInjection method will be called.
This is a wildcard for filtering by tag. This hint is also useful when OnDependencyInjection is in the On state and you want to limit the set of tags for which the OnDependencyInjection method will be called.
This is a regular expression for filtering by lifetime. This hint is also useful when OnDependencyInjection is in On state and it is necessary to restrict the set of lifetime for which the OnDependencyInjection method will be called.
This is a wildcard for filtering by lifetime. This hint is also useful when OnDependencyInjection is in On state and it is necessary to restrict the set of lifetime for which the OnDependencyInjection method will be called.
Determines whether to use the OnCannotResolve<T>(...)
partial method to handle a scenario in which an instance cannot be resolved. By default, this partial method is not generated. Because of the return value, it cannot have an empty body and must be overridden at creation.
// OnCannotResolve = On
// OnCannotResolveContractTypeNameRegularExpression = string|DateTime
// OnDependencyInjectionTagRegularExpression = null
DI.Setup("Composition")
...
To avoid missing failed bindings by mistake, use the two relevant hints below.
Determines whether to generate the OnCannotResolve<T>(...)
partial method when the OnCannotResolve hint is On to handle a scenario in which an instance cannot be resolved. By default it is On
.
// OnCannotResolve = On
// OnCannotResolvePartial = Off
// OnCannotResolveContractTypeNameRegularExpression = string|DateTime
// OnDependencyInjectionTagRegularExpression = null
DI.Setup("Composition")
...
To avoid missing failed bindings by mistake, use the two relevant hints below.
Determines whether to use a static partial method OnNewRoot<TContract, T>(...)
to handle the new composition root registration event.
// OnNewRoot = On
DI.Setup("Composition")
...
Be careful, this hint disables checks for the ability to resolve dependencies!
Determines whether to generate a static partial method OnNewRoot<TContract, T>(...)
when the OnNewRoot hint is On
to handle the new composition root registration event.
// OnNewRootPartial = Off
DI.Setup("Composition")
...
This is a regular expression for filtering by the name of the resolving type. This hint is also useful when OnCannotResolve is in On state and it is necessary to limit the set of resolving types for which the OnCannotResolve method will be called.
This is a wildcard for filtering by the name of the resolving type. This hint is also useful when OnCannotResolve is in On state and it is necessary to limit the set of resolving types for which the OnCannotResolve method will be called.
This is a regular expression for filtering by tag. This hint is also useful when OnCannotResolve is in On state and it is necessary to limit the set of tags for which the OnCannotResolve method will be called.
This is a wildcard for filtering by tag. This hint is also useful when OnCannotResolve is in On state and it is necessary to limit the set of tags for which the OnCannotResolve method will be called.
This is a regular expression for filtering by lifetime. This hint is also useful when OnCannotResolve is in the On state and it is necessary to restrict the set of lives for which the OnCannotResolve method will be called.
This is a wildcard for filtering by lifetime. This hint is also useful when OnCannotResolve is in the On state and it is necessary to restrict the set of lives for which the OnCannotResolve method will be called.
Determines whether to generate the ToString() method. This method provides a class diagram in mermaid format. To see this diagram, just call the ToString method and copy the text to this site.
// ToString = On
DI.Setup("Composition")
.Bind<IService>().To<Service>()
.Root<IService>("MyService");
var composition = new Composition();
string classDiagram = composition.ToString();
This hint determines whether the composition of objects will be created in a thread-safe way. The default value of this hint is On. It is a good practice not to use threads when creating an object graph, in this case the hint can be disabled, which will result in a small performance gain. For example:
// ThreadSafe = Off
DI.Setup("Composition")
.Bind<IService>().To<Service>()
.Root<IService>("MyService");
Overrides the modifiers of the public T Resolve<T>()
method.
Overrides the method name for public T Resolve<T>()
.
Overrides the modifiers of the public T Resolve<T>(object? tag)
method.
Overrides the method name for public T Resolve<T>(object? tag)
.
Overrides the modifiers of the public object Resolve(Type type)
method.
Overrides the method name for public object Resolve(Type type)
.
Overrides the modifiers of the public object Resolve(Type type, object? tag)
method.
Overrides the method name for public object Resolve(Type type, object? tag)
.
Overrides the modifiers of the public void Dispose()
method.
Overrides the modifiers of the public ValueTask DisposeAsync()
method.
Specifies whether the generated code should be formatted. This option consumes a lot of CPU resources. This hint may be useful when studying the generated code or, for example, when making presentations.
Indicates the severity level of the situation when, in the binding, an implementation does not implement a contract. Possible values:
- "Error", it is default value.
- "Warning" - something suspicious but allowed.
- "Info" - information that does not indicate a problem.
- "Hidden" - what's not a problem.
Specifies whether the generated code should be commented.
// Represents the composition class
DI.Setup(nameof(Composition))
.Bind<IService>().To<Service>()
// Provides a composition root of my service
.Root<IService>("MyService");
Appropriate comments will be added to the generated Composition
class and the documentation for the class, depending on the IDE used, will look something like this:
Indicates whether System.Threading.Lock
should be used whenever possible instead of the classic approach of synchronizing object access using System.Threading.Monitor1.
On` by default.
DI.Setup(nameof(Composition))
.Hint(Hint.SystemThreadingLock, "Off")
.Bind().To<Service>()
.Root<Service>("MyService");
Then documentation for the composition root:
Code generation workflow
flowchart TD
start@{ shape: circle, label: "Start" }
setups[fa:fa-search DI setups analysis]
types["`fa:fa-search Types analysis
constructors/methods/properties/fields`"]
subgraph dep[Dependency graph]
option[fa:fa-search Selecting a next dependency set]
creating[fa:fa-cog Creating a dependency graph variant]
verification{fa:fa-check-circle Verification}
end
codeGeneration[fa:fa-code Code generation]
compilation[fa:fa-cog Compilation]
failed@{ shape: dbl-circ, label: "fa:fa-thumbs-down Compilation failed" }
success@{ shape: dbl-circ, label: "fa:fa-thumbs-up Success" }
start ==> setups
setups -.->|Has problems| failed
setups ==> types
types -.-> |Has problems| failed
types ==> option
option ==> creating
option -.-> |There are no other options| failed
creating ==> verification
verification -->|Has problems| option
verification ==>|Correct| codeGeneration
codeGeneration ==> compilation
compilation -.-> |Has problems| failed
compilation ==> success
Pure.DI | DI Source code generator |
Pure.DI.Abstractions | Abstractions for Pure.DI |
Pure.DI.Templates | Template Package you can call from the shell/command line. |
Pure.DI.MS | Tools for working with Microsoft DI |
Install the DI template Pure.DI.Templates
dotnet new install Pure.DI.Templates
Create a "Sample" console application from the template di
dotnet new di -o ./Sample
And run it
dotnet run --project Sample
For more information about the template, please see this page.
Version update
When updating the version, it is possible that the previous version of the code generator remains active and is used by compilation services. In this case, the old and new versions of the generator may conflict. For a project where the code generator is used, it is recommended to do the following:
- After updating the version, close the IDE if it is open
- Delete the obj and bin directories
- Execute the following commands one by one
dotnet build-server shutdown
dotnet restore
dotnet build
Disabling API generation
Pure.DI automatically generates its API. If an assembly already has the Pure.DI API, for example, from another assembly, it is sometimes necessary to disable its automatic generation to avoid ambiguity. To do this, you need to add a DefineConstants element to the project files of these modules. For example:
<PropertyGroup>
<DefineConstants>$(DefineConstants);PUREDI_API_SUPPRESSION</DefineConstants>
</PropertyGroup>
Display generated files
You can set project properties to save generated files and control their storage location. In the project file, add the <EmitCompilerGeneratedFiles>
element to the <PropertyGroup>
group and set its value to true
. Build the project again. The generated files are now created in the obj/Debug/netX.X/generated/Pure.DI/Pure.DI/Pure.DI.SourceGenerator directory. The path components correspond to the build configuration, the target framework, the source generator project name, and the full name of the generator type. You can choose a more convenient output folder by adding the <CompilerGeneratedFilesOutputPath>
element to the application project file. For example:
<Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup>
<EmitCompilerGeneratedFiles>true</EmitCompilerGeneratedFiles>
<CompilerGeneratedFilesOutputPath>$(BaseIntermediateOutputPath)Generated</CompilerGeneratedFilesOutputPath>
</PropertyGroup>
</Project>
Performance profiling
Please install the JetBrains.dotTrace.GlobalTools dotnet tool globally, for example:
dotnet tool install --global JetBrains.dotTrace.GlobalTools --version 2024.3.3
Or make sure it is installed. Add the following sections to the project:
<Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup>
<PureDIProfilePath>c:\profiling</PureDIProfilePath>
</PropertyGroup>
<ItemGroup>
<CompilerVisibleProperty Include="PureDIProfilePath" />
</ItemGroup>
</Project>
Replace the path like c:\profiling with the path where the profiling results will be saved.
Start the project build and wait until a file like c:\profiling\pure_di_????.dtt appears in the directory.
Examples of how to set up a composition
Articles
- RU New in Pure.DI by the end of 2024
- RU New in Pure.DI
- RU Pure.DI v2.1
- RU Pure.DI next step
- RU Pure.DI for .NET
RU DotNext video
Thank you for your interest in contributing to the Pure.DI project! First of all, if you are going to make a big change or feature, please open a problem first. That way, we can coordinate and understand if the change you're going to work on fits with current priorities and if we can commit to reviewing and merging it within a reasonable timeframe. We don't want you to waste a lot of your valuable time on something that may not align with what we want for Pure.DI.
Contribution prerequisites: .NET SDK 9.0 or later is installed.
This repository contains the following directories and files:
π .github GitHub related files and main.yml for building using GitGub actions
π .logs temporary files for generating the README.md file
π .run configuration files for the Rider IDE
π benchmarks projects for performance measurement
π build application for building locally and using CI/CD
π docs resources for the README.md file
π readme sample scripts and examples of application implementations
π samples sample projects
π src source codes of the code generator and all libraries
|- π Pure.DI source code generator project
|- π Pure.DI.Abstractions abstraction library for Pure.DI
|- π Pure.DI.Core basic implementation of the source code generator
|- π Pure.DI.MS project for integration with Microsoft DI
|- π Pure.DI.Templates project templates for creating .NET projects using Pure.DI
|- π Directory.Build.props common MSBUILD properties for all source code generator projects
|- π Library.props common MSBUILD properties for library projects such as Pure.DI.Abstractions
π tests contains projects for testing
|- π Pure.DI.Example project for testing some integration scenarios
|- π Pure.DI.IntegrationTests integration tests
|- π Pure.DI.Tests unit tests for basic functionality
|- π Pure.DI.UsageTests usage tests, used for examples in README.md
|- π Directory.Build.props common MSBUILD properties for all test projects
π LICENSE license file
π build.cmd Windows script file to run one of the build steps, see description below
π build.sh Linux/Mac OS script file to run one of the build steps, see description below
π .space.kts build file using JetBrains space actions
π README.md this README.md file
π SECURITY.md policy file for handling security bugs and vulnerabilities
π Directory.Build.props basic MSBUILD properties for all projects
π Pure.DI.sln .NET solution file
The entire build logic is a regular console .NET application. You can use the build.cmd and build.sh files with the appropriate command in the parameters to perform all basic actions on the project, e.g:
Commands | Description |
---|---|
bm, benchmarks, benchmarks | Run benchmarks |
c, check, check | Compatibility checks |
dp, deploy, deploy | Package deployment |
e, example, example | Create examples |
g, generator, generator | Build and test the source code generator |
i, install, install | Install templates |
l, libs, libs | Build and test libraries |
p, pack, pack | Create NuGet packages |
perf, performance, performance | Performance tests |
pb, publish, publish | Publish the balazor web sssembly example |
r, readme, readme | Generate README.md |
t, template, template | Create and deploy templates |
te, testexamples, testexamples | Test examples |
u, upgrade, upgrade | Upgrading the internal version of DI to the latest public version |
For example, to build and test the source code generator:
./build.sh generator
or to run benchmarks:
./build.cmd benchmarks
If you are using the Rider IDE, it already has a set of configurations to run these commands. This project uses C# interactive build automation system for .NET. This tool helps to make .NET builds more efficient.
Thanks!
Array
Method | Mean | Error | StdDev | Median | Ratio | RatioSD | Gen0 | Gen1 | Allocated | Alloc Ratio |
---|---|---|---|---|---|---|---|---|---|---|
'Pure.DI composition root' | 88.33 ns | 1.075 ns | 1.006 ns | 88.58 ns | 0.94 | 0.03 | 0.0377 | - | 632 B | 1.00 |
'Pure.DI Resolve<T>()' | 91.94 ns | 1.832 ns | 2.445 ns | 91.66 ns | 0.98 | 0.04 | 0.0377 | - | 632 B | 1.00 |
'Pure.DI Resolve(Type)' | 92.65 ns | 2.244 ns | 6.366 ns | 89.18 ns | 0.98 | 0.07 | 0.0377 | - | 632 B | 1.00 |
'Hand Coded' | 94.24 ns | 1.883 ns | 2.578 ns | 94.06 ns | 1.00 | 0.04 | 0.0377 | - | 632 B | 1.00 |
DryIoc | 99.10 ns | 0.999 ns | 0.834 ns | 98.91 ns | 1.05 | 0.03 | 0.0377 | - | 632 B | 1.00 |
LightInject | 103.24 ns | 2.119 ns | 4.183 ns | 102.65 ns | 1.10 | 0.05 | 0.0377 | - | 632 B | 1.00 |
Unity | 4,510.09 ns | 74.628 ns | 66.155 ns | 4,488.31 ns | 47.89 | 1.47 | 0.8621 | 0.0076 | 14520 B | 22.97 |
Autofac | 15,134.66 ns | 110.608 ns | 86.355 ns | 15,131.24 ns | 160.72 | 4.45 | 1.7090 | 0.0610 | 28976 B | 45.85 |
Enum
Method | Mean | Error | StdDev | Median | Ratio | RatioSD | Gen0 | Gen1 | Allocated | Alloc Ratio |
---|---|---|---|---|---|---|---|---|---|---|
'Pure.DI composition root' | 64.36 ns | 1.282 ns | 2.919 ns | 62.75 ns | 1.00 | 0.05 | 0.0205 | - | 344 B | 1.00 |
'Hand Coded' | 64.49 ns | 0.791 ns | 0.661 ns | 64.47 ns | 1.00 | 0.01 | 0.0205 | - | 344 B | 1.00 |
'Pure.DI Resolve<T>()' | 65.26 ns | 0.965 ns | 0.903 ns | 65.56 ns | 1.01 | 0.02 | 0.0205 | - | 344 B | 1.00 |
'Pure.DI Resolve(Type)' | 65.43 ns | 1.004 ns | 0.784 ns | 65.23 ns | 1.01 | 0.02 | 0.0205 | - | 344 B | 1.00 |
'Microsoft DI' | 93.40 ns | 1.322 ns | 1.172 ns | 92.96 ns | 1.45 | 0.02 | 0.0281 | - | 472 B | 1.37 |
LightInject | 147.58 ns | 1.650 ns | 1.544 ns | 147.96 ns | 2.29 | 0.03 | 0.0510 | - | 856 B | 2.49 |
DryIoc | 147.61 ns | 1.110 ns | 0.984 ns | 147.54 ns | 2.29 | 0.03 | 0.0510 | - | 856 B | 2.49 |
Unity | 3,736.48 ns | 73.272 ns | 68.539 ns | 3,739.88 ns | 57.94 | 1.18 | 0.8202 | 0.0076 | 13752 B | 39.98 |
Autofac | 15,610.79 ns | 288.580 ns | 595.967 ns | 15,436.50 ns | 242.08 | 9.47 | 1.7395 | 0.0610 | 29104 B | 84.60 |
Func
Method | Mean | Error | StdDev | Ratio | RatioSD | Gen0 | Gen1 | Allocated | Alloc Ratio |
---|---|---|---|---|---|---|---|---|---|
'Pure.DI composition root' | 4.382 ns | 0.0561 ns | 0.0525 ns | 0.84 | 0.01 | 0.0014 | - | 24 B | 1.00 |
'Hand Coded' | 5.193 ns | 0.0649 ns | 0.0542 ns | 1.00 | 0.01 | 0.0014 | - | 24 B | 1.00 |
'Pure.DI Resolve<T>()' | 5.914 ns | 0.0753 ns | 0.0667 ns | 1.14 | 0.02 | 0.0014 | - | 24 B | 1.00 |
'Pure.DI Resolve(Type)' | 6.375 ns | 0.0465 ns | 0.0388 ns | 1.23 | 0.01 | 0.0014 | - | 24 B | 1.00 |
DryIoc | 28.979 ns | 0.3875 ns | 0.3435 ns | 5.58 | 0.08 | 0.0072 | - | 120 B | 5.00 |
LightInject | 155.185 ns | 2.9628 ns | 2.7714 ns | 29.89 | 0.60 | 0.0300 | - | 504 B | 21.00 |
Unity | 1,760.400 ns | 12.4774 ns | 11.6714 ns | 339.03 | 3.99 | 0.1507 | - | 2552 B | 106.33 |
Autofac | 5,745.474 ns | 36.1642 ns | 30.1988 ns | 1,106.51 | 12.28 | 0.8316 | 0.0076 | 14008 B | 583.67 |
Singleton
Method | Mean | Error | StdDev | Ratio | RatioSD | Gen0 | Gen1 | Allocated | Alloc Ratio |
---|---|---|---|---|---|---|---|---|---|
'Hand Coded' | 3.016 ns | 0.0721 ns | 0.0602 ns | 1.00 | 0.03 | 0.0014 | - | 24 B | 1.00 |
'Pure.DI composition root' | 3.337 ns | 0.0778 ns | 0.0728 ns | 1.11 | 0.03 | 0.0014 | - | 24 B | 1.00 |
'Pure.DI Resolve<T>()' | 3.517 ns | 0.0602 ns | 0.0533 ns | 1.17 | 0.03 | 0.0014 | - | 24 B | 1.00 |
'Pure.DI Resolve(Type)' | 4.649 ns | 0.1417 ns | 0.3832 ns | 1.54 | 0.13 | 0.0014 | - | 24 B | 1.00 |
DryIoc | 11.388 ns | 0.0389 ns | 0.0304 ns | 3.78 | 0.07 | 0.0014 | - | 24 B | 1.00 |
'Simple Injector' | 16.630 ns | 0.0782 ns | 0.0693 ns | 5.52 | 0.11 | 0.0014 | - | 24 B | 1.00 |
'Microsoft DI' | 19.551 ns | 0.0684 ns | 0.0606 ns | 6.49 | 0.12 | 0.0014 | - | 24 B | 1.00 |
LightInject | 426.861 ns | 1.0945 ns | 0.9702 ns | 141.59 | 2.70 | 0.0014 | - | 24 B | 1.00 |
Unity | 2,645.530 ns | 47.7345 ns | 66.9171 ns | 877.51 | 27.40 | 0.1869 | - | 3184 B | 132.67 |
Autofac | 9,723.452 ns | 60.9124 ns | 53.9972 ns | 3,225.24 | 63.41 | 1.4343 | 0.0458 | 24208 B | 1,008.67 |
'Castle Windsor' | 16,991.331 ns | 93.5701 ns | 87.5255 ns | 5,635.97 | 110.23 | 1.4038 | - | 23912 B | 996.33 |
Ninject | 67,297.995 ns | 1,045.7465 ns | 927.0271 ns | 22,322.54 | 516.33 | 4.2725 | 1.0986 | 73176 B | 3,049.00 |
Transient
Method | Mean | Error | StdDev | Ratio | RatioSD | Gen0 | Gen1 | Allocated | Alloc Ratio |
---|---|---|---|---|---|---|---|---|---|
'Pure.DI composition root' | 3.491 ns | 0.1203 ns | 0.2764 ns | 0.97 | 0.09 | 0.0014 | - | 24 B | 1.00 |
'Hand Coded' | 3.619 ns | 0.1132 ns | 0.1952 ns | 1.00 | 0.07 | 0.0014 | - | 24 B | 1.00 |
'Pure.DI Resolve<T>()' | 4.417 ns | 0.1377 ns | 0.3429 ns | 1.22 | 0.11 | 0.0014 | - | 24 B | 1.00 |
'Pure.DI Resolve(Type)' | 4.649 ns | 0.1431 ns | 0.1338 ns | 1.29 | 0.08 | 0.0014 | - | 24 B | 1.00 |
LightInject | 7.043 ns | 0.0902 ns | 0.0754 ns | 1.95 | 0.10 | 0.0014 | - | 24 B | 1.00 |
'Microsoft DI' | 10.357 ns | 0.1357 ns | 0.1133 ns | 2.87 | 0.15 | 0.0014 | - | 24 B | 1.00 |
DryIoc | 10.673 ns | 0.1197 ns | 0.1119 ns | 2.96 | 0.16 | 0.0014 | - | 24 B | 1.00 |
'Simple Injector' | 14.594 ns | 0.1373 ns | 0.1217 ns | 4.04 | 0.21 | 0.0014 | - | 24 B | 1.00 |
Unity | 4,320.577 ns | 85.0619 ns | 175.6675 ns | 1,197.13 | 78.58 | 0.3052 | - | 5176 B | 215.67 |
Autofac | 12,616.111 ns | 227.4971 ns | 201.6703 ns | 3,495.62 | 189.02 | 1.9836 | 0.0916 | 33224 B | 1,384.33 |
'Castle Windsor' | 27,766.858 ns | 312.7020 ns | 277.2021 ns | 7,693.52 | 405.54 | 3.2349 | 0.0305 | 54360 B | 2,265.00 |
Ninject | 148,286.554 ns | 3,309.2559 ns | 9,600.7521 ns | 41,086.58 | 3,398.59 | 7.5684 | 1.4648 | 128736 B | 5,364.00 |
Benchmarks environment
BenchmarkDotNet v0.14.0, Windows 10 (10.0.19045.4894/22H2/2022Update)
AMD Ryzen 9 5900X, 1 CPU, 24 logical and 12 physical cores
.NET SDK 9.0.100
[Host] : .NET 9.0.0 (9.0.24.52809), X64 RyuJIT AVX2
DefaultJob : .NET 9.0.0 (9.0.24.52809), X64 RyuJIT AVX2