The ServiceCollection Extension Pattern

Apr 08 2024

 

Many thanks to the sponsors who make it possible for this newsletter to be free for readers.

 

• Streamline your API development with Postman's REST Client a powerful tool for sending requests, inspecting responses, and debugging REST APIs with ease. Discover a more efficient way to build and test APIs at link.

 

• Backend design like you've never seen it before - a guide to building SOLID ASP.NET Core web apps that stand the test of time. Featuring more Minimal APIs, more testing, a new e-commerce project, and the modular monolith! Explore the contents in depth here.

 

Background

 
 

In the .NET environment, Dependency Injection (DI) is a fundamental technique that helps us to create flexible, testable, and loosely coupled code.

 

It's an instance of the Inversion of Control (IoC) principle, where the control of creating objects is handled by a container or a framework, rather than by the classes themselves.

 

The IServiceCollection interface represents a contract for a collection of service descriptors, providing an abstraction to add, remove, and retrieve services.

 

This interface becomes very important in the Program.cs class.

 

Here, we inject all of the necessary services that our application requires.

 
 

The ServiceCollection Extension Pattern

 
 

In larger applications, Program.cs can become bloated and messy.

 

To avoid this, we can utilize a pattern called the ServiceCollection Extension Pattern.

 

This pattern lets us encapsulate the service registration logic into separate static classes, providing a more organized and readable structure.

 

The pattern involves creating extension methods on the IServiceCollection interface.

 

 

Let's see an example:

public static class ServiceCollectionExtensions
{
    public static IServiceCollection AddCustomServices(
        this IServiceCollection services)
    {
        services.AddScoped<IMyService, MyService>();
        services.AddSingleton<IOtherService, OtherService>();

        // ... other services

        return services;
    }
}
And then in your Program.cs file, you can leverage these extensions like this:

builder.Services.AddCustomServices();

 
 

The Real-world Example

 
 

We can further categorize our services using this pattern, for example, by creating different extension methods for different types or layers of services.

 

Let's consider an application that uses Entity Framework Core for data access, Identity for authentication and authorization, and also needs to configure CORS (Cross-Origin Resource Sharing).

 

Here's how you might apply the ServiceCollection Extension Pattern in such a scenario.

 

Database:

public static IServiceCollection AddDatabase(
    this IServiceCollection services,
    string connectionString)
{
    services.AddDbContext<MyDbContext>(options => options.UseSqlServer(connectionString));

    return services;
}
Identity:

public static IServiceCollection AddIdentityServices(
    this IServiceCollection services)
{
    services.AddIdentity<ApplicationUser, IdentityRole>()
        .AddEntityFrameworkStores<MyDbContext>()
        .AddDefaultTokenProviders();

    return services;
}
JWT Authentication:

public static IServiceCollection AddJwtAuthentication(
    this IServiceCollection services, IConfiguration configuration)
{
    var jwtSettings = configuration.GetSection("JwtSettings");
    var key = Encoding.ASCII.GetBytes(jwtSettings.GetValue<string>("Secret"));

    services.AddAuthentication(options =>
    {
        options.DefaultAuthenticateScheme = JwtBearerDefaults.AuthenticationScheme;
        options.DefaultChallengeScheme = JwtBearerDefaults.AuthenticationScheme;
    })
    .AddJwtBearer(options =>
    {
        options.RequireHttpsMetadata = false;
        options.SaveToken = true;
        options.TokenValidationParameters = new TokenValidationParameters
        {
            ValidateIssuerSigningKey = true,
            IssuerSigningKey = new SymmetricSecurityKey(key),
            ValidateIssuer = false,
            ValidateAudience = false
        };
    });

    return services;
}
Cors Policy:

public static IServiceCollection AddCorsPolicy(
    this IServiceCollection services,
    string policyName,
    string[] allowedOrigins)
{
    services.AddCors(options =>
    {
        options.AddPolicy(policyName,
            builder => builder.WithOrigins(allowedOrigins)
                     .AllowAnyMethod()
                     .AllowAnyHeader()
                     .AllowCredentials());
    });

    return services;
}
And then in your Program.cs file, you can leverage these extensions like this:

builder.Services.AddDatabase(Configuration.GetConnectionString("DefaultConnection"));
builder.Services.AddIdentityServices();
builder.Services.AddJwtAuthentication(Configuration);
builder.Services.AddCorsPolicy("MyPolicy", new[] { "http://example.com" });

// ... other configurations

 
 

Why should we consider using it?

 
 

1. Organization and Readability

 

In a large application with a myriad of services, the Program.cs can quickly become bloated and messy.
This makes it difficult for a developer to comprehend what's going on at a glance.

 

This improves the readability of your code and makes it easier to understand, which is especially valuable in team environments.

 

2. Encapsulation

 

The ServiceCollection Extension Pattern follows the principle of Encapsulation, a core tenet of object-oriented programming.

 

Encapsulation enables us to hide the complexity of service registrations behind a set of methods.
This shields other parts of your application from the intricacies of these operations, and provides a clean and simple interface to register services.

 

3. Maintainability

 

Applications are bound to evolve over time.

 

The services you start with may not be the ones you end with. With the ServiceCollection Extension Pattern, it's easier to modify, add, or remove service registrations.

 

Since you have logically separated your service registrations, you can find and alter the specific group you need without hunting through a potentially large Program.cs.

 

4. Reusability

 

If you have common sets of services used across multiple projects, you can reuse your extension methods in different applications. This cuts down on duplicated code, saving you time and reducing the potential for errors.

 
 

Wrapping up

 
 

The ServiceCollection Extension Pattern is a valuable tool in keeping your Program.cs tidy and maintainable, particularly for larger applications.

 

By encapsulating the service registration logic into separate methods, we can improve the organization and readability of our service registration code.

 

That's all from me today.

Join 13,250+ subscribers to improve your .NET Knowledge.

There are 3 ways I can help you:

Design Patterns Simplified ebook

Go-to resource for understanding the core concepts of design patterns without the overwhelming complexity. In this concise and affordable ebook, I've distilled the essence of design patterns into an easy-to-digest format. It is a Beginner level. Check out it here.


Sponsorship

Promote yourself to 13,250+ subscribers by sponsoring this newsletter.


Join TheCodeMan.net Newsletter

Every Monday morning, I share 1 actionable tip on C#, .NET & Arcitecture topic, that you can use right away.


Subscribe to
TheCodeMan.net

Subscribe to the TheCodeMan.net and be among the 13,250+ subscribers gaining practical tips and resources to enhance your .NET expertise.