/MapStrap

MapStrap is a simple set of interfaces for defining mappings between types in a strictly layered .NET application.

Primary LanguageC#MIT LicenseMIT

MapStrap

MapStrap is a simple set of interfaces for defining mappings between types in a strictly layered .NET application.

MapStrap uses AutoMapper to provide the automated mapping using a convention based approach. It simplifies the configuration of AutoMapper through a simple set of interfaces that move mapping code away from bootstrapping code and next to the associated Data Transfer Object (DTO).

Quickstart

  1. Install the MapStrap NuGet package.
  2. Use the MapStrap interface types on your Data Transfer Objects (DTOs) to define your AutoMapper mappings.
  3. Use the MapStrap AutoMapper extension methods during bootstrap to create your AutoMapper maps by convention.

Details

Configuring Mapping from Domain Models to Data Transfer Objects (DTOs)

In a strictly layered application, DTOs are used in responses from API or MVC controllers. For example, in an MVC application a view model is passed to a view, rather than exposing a domain model. In a Web API, an API model is serialized as part of the response to an HTTP request, rather than returning a domain model directly.

To configure your mapping between your domain model and DTO, use one of the following interfaces provided by MapStrap on your DTO types:

  • IMapFromDomain<TDomain> - use this for convention only based mappings
  • IHaveCustomMap<TSource, TDestination> - use this to add overriding configurations to your mappings (gets access to AutoMapper IMappingExpression)
  • IHaveCustomConfiguration - use this to have access to full AutoMapper configuration (gets access to AutoMapper IMapperConfigurationExpression)

Examples

Assume that we have the following User domain model that uses encapsulation to protect its invariants:

public class User
{
    public User(int id, string name)
    {
        this.Id = id;
        this.Name = name;
    }
    
    public int Id { get; }
        
    public string Name { get; }   
}

If we wish to return this type from our API controller, then we would have an equivalent UserApiModel or similar type. To map by convention only (map property values if their names match), then we can define our UserApiModel using the IMapFromDomain<TDomain> interface, where TDomain is the type of the domain model that we are mapping from (in this case User).

public class UserApiModel : IMapFromDomain<User>
{
   public int Id { get; set; }
   public string Name { get; set; }
}

You'll see later how we bootstrap using MapStrap so that the appropriate AutoMapper mapping is automatically created, and how we then perform a mapping from a User instance to a UserApiModel.

Customising Mappings

If you wish to add some futher configuration to your mapping, to override some of the AutoMapper conventions, then you can use the IHaveCustomMap<TSource, TDestination> where TSource is the domain type (in this case User) and TDestination is the DTO type (in this case UserApiModel).

This interface defines a Map method that takes an AutoMapper IMappingExpression<TSource, TDestination>:

public class UserApiModel : IHaveCustomMap<User, UserApiModel>
{
   public int Id { get; set; }
   public string Name { get; set; }
   
   public void Map(IMappingExpression<User, UserApiModel> mapping)
   {
      mapping.ForMember(
          u => u.Name,
          cfg => cfg.ResolveUsing(u => "User name is " + u.Id));
   }
}

If you want full control over AutoMapper, then you can use the IHaveCustomConfiguration interface. This defines a Configure method which takes the AutoMapper configuration instance:

public class UserApiModel : IHaveCustomConfiguration
{
   public int Id { get; set; }
   public string Name { get; set; }
   
   public void Configure(IMapperConfigurationExpression configuration)
   {
       // Access AutoMapper configuration instance here
       ...
   }
}

Bootstrapping

With all your mappings defined on your DTOs, you can automatically have MapStrap create all the appropriate AutoMapper mappings by placing the following code as part of your application bootstrapping process. You can use AutoMapper's instance or static based API. Here we use the instance based API:

var config = new MapperConfiguration(
    cfg =>
        {
            cfg.CreateMaps(
                new AssemblyTypeResolver(
                    new[]
                    {
                        typeof(MyApiModel).Assembly                        
                    }));
        });

var mapper = config.CreateMapper();

Note that if you prefer to use AutoMappers static API instead, the CreateMaps, CreateConventionMaps, CreateCustomMaps, and CreateCustomConfiguration extension methods can also be used with Mapper.Initialize().

MapStrap provides a CreateMaps extension method which takes an ITypeResolver. This returns the collection of types that we wish to be considered when scanning for mappings. There is one implementation out of the box - AssemblyTypeResolver which takes a collection of assemblies to scan.

Note that you can implement your own ITypeResolver if you wish to have more control over what types are scanned for mappings.

The CreateMaps extension then automatically creates AutoMapper maps for each type returned from the type resolver, based on convention, custom maps, or configuration.

If you only want to scan for a subset of convention, custom maps, and configuration maps then use a combination of the CreateConventionMaps, CreateCustomMaps, and CreateCustomConfiguration extension methods instead.

Mapping from Domain Models to Data Transfer Objects (DTOs)

With MapStrap successfully bootstrapped, all of our mappings are now created as AutoMapper mappings, and are ready to be used to map between instances of types. You would typically want to do this in your MVC or API controllers.

Examples

If using AutoMapper's static API, then you can choose to use AutoMapper's Mapper type directly in your controllers:

[HttpGet]
[Route("api/users")]
public IHttpActionResult Users()
{
   var users = this.usersRepository.GetUsers();
   var userApiModels = Mapper.Map<IEnumerable<User>, IEnumerable<UserApiModel>>(users);
   return this.Ok(userApiModels);
}

However, you may wish to introduce an abstraction for mapping, for testing purposes:

public UsersController(IUsersRepository usersRepository, IMapper<User, UserApiModel> userMapper)
{
   this.usersRepository = usersRepository;
   this.userMapper = userMapper;
}

public IHttpActionResult Users()
{
   var users = this.usersRepository.GetUsers();
   var userApiModels = this.userMapper.MapCollection(users);
   return this.Ok(userApiModels);
}

Here mapper is an instance of a domain IMapper<TSource, TDestination> type, which has one AutoMapper implementation:

public interface IMapper<in TSource, out TDestination>
{
    TDestination Map(TSource source);

    IEnumerable<TDestination> MapCollection(IEnumerable<TSource> source);
}

internal class AutoMapperMapper<TSource, TDestination> : IMapper<TSource, TDestination>
{   
    private readonly IMapper mapper;
    
    public AutoMapperMapper(IMapper mapper)
    {
       if (mapper == null)
       {
           throw new ArgumentNullException(nameof(mapper));
       }
       
       this.mapper = mapper;
    }
    
    public TDestination Map(TSource source)
    {
        return this.mapper.Map<TSource, TDestination>(source);
        
        // could also use static API here:
        // return Mapper.Map<TSource, TDestination>(source);
    }

    public IEnumerable<TDestination> MapCollection(IEnumerable<TSource> source)
    {
        return this.mapper.Map<IEnumerable<TSource>, IEnumerable<TDestination>>(source);
        
        // could also use static API here:
        // return Mapper.Map<IEnumerable<TSource>, IEnumerable<TDestination>>(source);
    }
}

Alternatively, if you prefer to use the instance based API, then the IMapper instance returned from CreateMapper can be injected either into your controllers, or into your AutoMapperMapper implementation. You can register the IMapper instance with your IoC container, for example with Autofac:

// On bootstrap
var config = new MapperConfiguration(
   cfg =>
       {
           cfg.CreateMaps(
               new AssemblyTypeResolver(
                   new[]
                       {
                           typeof(MyApiModel).Assembly,
                           typeof(MyDto).Assembly
                       }));
       });

var mapper = config.CreateMapper();

// Register AutoMapper's IMapper
builder.RegisterInstance(mapper);

// Register your IMapper<,> implemenation
builder.RegisterGeneric(typeof(AutoMapperMapper<,>))
    .As(typeof(IMapper<,>))    
    .SingleInstance();

// Injecting AutoMapper's IMapper into your API controller directly:
public UsersController(IUsersRepository usersRepository, IMapper mapper)

// Or via your own IMapper<,> abstraction (allows better extensibility/testing options):
public class AutoMapperMapper<TSource, TDestination> : IMapper<TSource, TDestination>
{
    private readonly IMapper mapper;

    public AutoMapperMapper(IMapper mapper)
    {
        if (mapper == null)
        {
            throw new ArgumentNullException(nameof(mapper));
        }

        this.mapper = mapper;
    }

    public TDestination Map(TSource source)
    {
        return this.mapper.Map<TSource, TDestination>(source);
    }

    public IEnumerable<TDestination> Map(IEnumerable<TSource> source)
    {
        return this.mapper.Map<IEnumerable<TSource>, IEnumerable<TDestination>>(source);
    }
}

// Inject your mapper abstraction into your controller:
public UsersController(IUsersRepository usersRepository, IMapper<MyDomain, MyApiModel> mapper)

Configuring Mapping from Data Transfer Objects (DTOs) to Domain Models

Domain models use encapsulation to protect their invariants, therefore AutoMapper is not used to map from DTOs to domain models. However, MapStrap does provide a couple of convenience methods to standardise how domain models are instantiated from DTOs.

A typical example is when returning domain models from domain repositories where repository implementations sit within a data layer, and the data layer uses DTOs to model the data store - for example mapping Entity Framework DTOs used within DbSets to domain models.

Examples

In the majority of cases your data DTO will map to a single domain model type. You can use the provided IMapToDomain<TDomain> interface on your data DTO. It defines a single TDomain ToDomainModel() method:

public class UserDto : IMapToDomain<User>
{
    public int Id { get; set; }
    public string Name { get; set; }

    public User ToDomainModel()
    {
       return new User(this.Id, this.Name);
    }
}

In some cases your data DTO may map to more than one domain type. In this case, you can continue decorating your data DTO type with multiple instances of IMapToAlternativeDomain<TDomain> - one for each domain model type. This interface defines a single void ToDomainModel(out TDomain result) method:

public class UserDto : IMapToDomain<User>, IMapToAlternativeDomain<Admin>
{
    public int Id { get; set; }
    public string Name { get; set; }

    public User ToDomainModel()
    {
       return new User(this.Id, this.Name);
    }
    
    public void ToDomainModel(out Admin result)
    {
       result = new Admin(...);
    }
}

Examples

Here we have an Entity Framework DTO type (UserDto) which we have a collection of within a DbSet on our DbContext. In our domain repository Entity Framework implementation, we use projection to make the conversion from DTO to domain model:

public class EntityFrameworkUsersRepository : IUsersRepository
{
    ...
    
    public IEnumerable<User> GetUsers()
    {
       using (var context = this.contextFactory.Create())
       {
          return context.Users.Select(u => u.ToDomainModel()).ToList();
       }
    }
}

Downloads

MapStrap is available via NuGet: