AttributeAutoDI 1.0.1

dotnet add package AttributeAutoDI --version 1.0.1
                    
NuGet\Install-Package AttributeAutoDI -Version 1.0.1
                    
This command is intended to be used within the Package Manager Console in Visual Studio, as it uses the NuGet module's version of Install-Package.
<PackageReference Include="AttributeAutoDI" Version="1.0.1" />
                    
For projects that support PackageReference, copy this XML node into the project file to reference the package.
<PackageVersion Include="AttributeAutoDI" Version="1.0.1" />
                    
Directory.Packages.props
<PackageReference Include="AttributeAutoDI" />
                    
Project file
For projects that support Central Package Management (CPM), copy this XML node into the solution Directory.Packages.props file to version the package.
paket add AttributeAutoDI --version 1.0.1
                    
#r "nuget: AttributeAutoDI, 1.0.1"
                    
#r directive can be used in F# Interactive and Polyglot Notebooks. Copy this into the interactive tool or source code of the script to reference the package.
#addin nuget:?package=AttributeAutoDI&version=1.0.1
                    
Install AttributeAutoDI as a Cake Addin
#tool nuget:?package=AttributeAutoDI&version=1.0.1
                    
Install AttributeAutoDI as a Cake Tool

AttributeAutoDI

๐ŸŒŸ Attribute-based automatic dependency injection library for .NET


README-KR

README-KR

Nuget

NUGET

dotnet add package AttributeAutoDI --version 1.0.0


Sample

Sample


๐Ÿ’ก Motivation

The default DI container in ASP.NET Core is simple and powerful, but having to call builder.Services.AddXXX<>() repeatedly introduces a few pain points:

โŒ The more services you implement, the longer and harder your Program.cs becomes to manage

โŒ Handling multiple implementations of the same interface requires verbose, error-prone registration

โŒ Service intent is not self-explanatory โ€“ registration and class definition are separated

๐Ÿ‘‰ What if... classes could declare how they want to be injected? AttributeAutoDI was born from this idea.

It is built on the following principles: โœ… Services should declare their own lifecycle and purpose

โœ… Multiple implementations should be clearly handled via [Primary] or [Named]

โœ… We should reduce boilerplate code in Program.cs without sacrificing readability

Our goal is to simplify Program.cs and Startup.cs as much as possible so you can trust the system to "just work."


๐Ÿงพ Requirements

  • .NET 6.0 or later
  • Compatible with Microsoft.Extensions.DependencyInjection (ASP.NET Core, Console apps, etc.)

๐Ÿ”ฅ Features

โœ… Automatic registration via [Singleton], [Scoped], and [Transient]

โญ [Primary] support for selecting default implementation

๐Ÿท๏ธ [Named] support for explicit injection via constructor parameters

๐Ÿ› ๏ธ [Options("Section")] for automatic configuration binding

๐Ÿ‘พ Executes configuration methods marked with [Execute] inside classes decorated with [PreConfiguration] or [PostConfiguration].


๐Ÿš€ Usage

1. Register the extension

builder.Services.AddAttributeDependencyInjection(builder.Configuration);

or

builder.Services.AddAttributeDependencyInjection(builder.Configuration,typeof(Program).Assembly);

Internally, it expands to:

public static class AttributeInjectionExtension
{
    public static void AddAttributeDependencyInjection(
    this IServiceCollection services,
    IConfiguration configuration,
    Assembly? assembly = null
)
{
    assembly ??= Assembly.GetEntryAssembly();

    services.UsePreConfiguration(assembly!);

    services.UseOptionsBindingInjection(configuration, assembly!);
    services.UseAttributeInjection(assembly!);
    services.UsePrimaryInjection();
    services.UseNameParameterInjection(assembly!);
    services.Replace(ServiceDescriptor.Transient<IControllerActivator, NamedControllerActivator>());

    services.UsePostConfiguration(assembly!);
}

Execution Flow

  1. Execute methods under [PreConfiguration] classes

  2. Register [Options] bindings

  3. Register class dependencies marked with [Singleton], [Transient], or [Scoped]

  4. Register [Primary] dependencies

  5. Register constructor parameter classes and controller activators with [Named]

  6. Execute methods under [PostConfiguration] classes

โš ๏ธ Important: Register in correct order!

โœ”๏ธ This works as expected:

builder.Service.AddSingleton...
builder.Service.AddAttributeDependencyInjection 

โŒ This may not work correctly:

builder.Service.AddAttributeDependencyInjection
builder.Service.AddSingleton...

If you use both manual and attribute-based registration, always call AddAttributeDependencyInjection() after manual registrations.

2. Attribute-based Auto registration - [Singleton], [Transient], [Scoped]

[Singleton]
public class MyService : IMyService
{
    public string Get() => "hello";
}

This is equivalent to:

builder.Services.AddSingleton<IMyService, MyService>();

3. Primary service selection - [Primary]


[Singleton]
public class MyService : IMyService
{
    public string Get() => "hello";
}

[Primary]
[Singleton]
public class MyPrimaryService : IMyService
{
    public string Get() => "I am primary";
}

When injecting IMyService, the system will prefer MyPrimaryService.

4. Named injection - [Named]

[Singleton]
public class MyService : IMyService
{
    public string Get() => "hello";
}

This will register myService (default camel-case name). Or set custom name:

[Singleton("myNamedService")]
public class MyService : IMyService
{
    public string Get() => "hello";
}

Then inject with:

[Singleton]
public class TestFacade([Named("myNamedService")]IMyservice myservice)
{
...
}
// myservice == MyService

Named injection works in registered Class

5. Options binding via attribute - [Options]

[Options("Sample:Sample")]
public class SampleOption
{
    public string Sample { get; set; } = "";
}

This is equivalent to:

builder.Services.Configure<SampleOption>(configuration.GetSection("Sample:Sample"));

Then you can inject it with:

public class SampleController(
    IOptions<SampleOption> sampleOption) ...

6. Configuration Execution - [PreConfiguration], [PostConfiguration], [Execute]

You can run setup methods before or after AddAttributeDependencyInjection using [Execute] methods.

The execution order is:

PreConfiguration โ†’ AddAttributeDependencyInjection โ†’ PostConfiguration

[PreConfiguration] //or [PostConfiguration]
public static class Configuration{
    [Execute]
    public static void TestConfig1(IServiceCollection service){
        ...
    }

    [Execute]
    public static void TestConfig2(this IServiceCollection service){
    ...
    }
    
    [Execute]
    public static void TestConfig3(IServiceCollection service, IConfiguration configuration){
        ...
    }

    [Execute]
    public static void TestConfig4(this IServiceCollection service, IConfiguration configuration){
    ...
    }
}

โœ… Only these four method signatures are supported.

โ„น๏ธ Return types are ignored โ€” any return type is allowed but not used.


๐Ÿงช Examples

Check out the AttributeAutoDI.Sample project for practical examples.


๐Ÿ™Œ Contributing

Found a bug? Have an improvement idea? Feel free to open an issue or submit a pull request.


๐Ÿ“„ License

MIT License Copyright (c) 2025

Product Compatible and additional computed target framework versions.
.NET net6.0 is compatible.  net6.0-android was computed.  net6.0-ios was computed.  net6.0-maccatalyst was computed.  net6.0-macos was computed.  net6.0-tvos was computed.  net6.0-windows was computed.  net7.0 is compatible.  net7.0-android was computed.  net7.0-ios was computed.  net7.0-maccatalyst was computed.  net7.0-macos was computed.  net7.0-tvos was computed.  net7.0-windows was computed.  net8.0 is compatible.  net8.0-android was computed.  net8.0-browser was computed.  net8.0-ios was computed.  net8.0-maccatalyst was computed.  net8.0-macos was computed.  net8.0-tvos was computed.  net8.0-windows was computed.  net9.0 is compatible.  net9.0-android was computed.  net9.0-browser was computed.  net9.0-ios was computed.  net9.0-maccatalyst was computed.  net9.0-macos was computed.  net9.0-tvos was computed.  net9.0-windows was computed. 
Compatible target framework(s)
Included target framework(s) (in package)
Learn more about Target Frameworks and .NET Standard.
  • net6.0

    • No dependencies.
  • net7.0

    • No dependencies.
  • net8.0

    • No dependencies.
  • net9.0

    • No dependencies.

NuGet packages

This package is not used by any NuGet packages.

GitHub repositories

This package is not used by any popular GitHub repositories.

Version Downloads Last updated
1.0.1 41 4/2/2025
1.0.0 34 4/2/2025