簡體   English   中英

AutoMapper ProjectTo <>()找不到地圖

[英]AutoMapper ProjectTo<>() not finding map

我有一個ASP.NET 5(運行在4.6.2,而不是Core)應用程序。 我想使用AutoMapper的ProjectTo <>()方法將數據庫的結果投影到我的viewmodels。

我已經嘗試了很多測試,但似乎在使用ProjectTo <>()時無法找到地圖。 在具有相同模型和viewmodel的不同位置上使用mapper.Map <>()非常有效。

我想AutoMapper如何與我的DI(Autofac)一起工作有問題,但我無法弄清楚是什么。

無論如何,代碼:

Startup.Cs

 public IServiceProvider ConfigureServices(IServiceCollection services)
        {
            (...)

            // Autofac DI
            AutofacContainer = AutofacLoader.Configure(services).Build();

            return AutofacContainer.Resolve<IServiceProvider>();
        }

AutofacLoader.cs

public static ContainerBuilder Configure(IServiceCollection services)
        {
            var builder = new ContainerBuilder();

(...)


            // AutoMapper
            builder.RegisterModule<AutoMapperModule>();

            if (services != null)
            { 
                builder.Populate(services);

            }
            return builder;
        }

AutoMapperModule.cs

public class AutoMapperModule : Module
{
    protected override void Load(ContainerBuilder builder)
    {
        var mapping = new MapperConfiguration(cfg =>
        {
            cfg.AddProfile(new Core.Mappings.AutoMapperProfileConfiguration());
            cfg.AddProfile(new Dieet.Core.Mappings.AutoMapperProfileConfiguration());
        });
        builder.RegisterInstance(mapping.CreateMapper()).As<IMapper>().AutoActivate();
    }
}

測試因“從患者到患者視圖缺失地圖”而失敗。 使用Mapper.CreateMap創建'。

   [Fact]
    public async void InfohosServiceReturnsPatientViewModels()
    {
        var db = _container.Resolve<IInfohosDb>();

        var search = new PaginatedSearchBase();
        search.OrderBy = "Naam";

        var mapper = _container.Resolve<IMapper>();

        var result = await search.PagedResultAsAsync<Patient,PatientViewModel >(null,db.Patienten,mapper);
    }

PaginatedSearchBase

public class PaginatedSearchBase
{
    public string OrderBy { get; set; }
    public bool OrderDescending { get; set; }
    public int Page { get; set; } = 1;
    public int PageSize { get; set; } = 10;
}

最后是調用ProjectTo的擴展

public static class PagedResultExtensions
{
    public static async Task<PagedResult<T>> PagedResultAsync<T>(this PaginatedSearchBase vm, ICollection<Expression<Func<T, bool>>> whereCollection, IEnumerable<T> context) where T : class
    {
        int totalCount;
        var query = PrepareQuery(vm, whereCollection, context, out totalCount);

        return new PagedResult<T>
        {
            Results = await query.ToListAsync(),
            Page = vm.Page,
            PageSize = vm.PageSize,
            Total = totalCount
        };
    }
    public static async Task<PagedResult<TAs>> PagedResultAsAsync<T, TAs>(this PaginatedSearchBase vm, ICollection<Expression<Func<T, bool>>> whereCollection, IEnumerable<T> context, IMapper mapper) where T : class
    {
        int totalCount;
        var query = PrepareQuery(vm, whereCollection, context, out totalCount);

        return new PagedResult<TAs>
        {
----------> Results = await query.ProjectTo<TAs>(mapper).ToListAsync(),
            Page = vm.Page,
            PageSize = vm.PageSize,
            Total = totalCount
        };
    }

    private static IQueryable<T> PrepareQuery<T>(PaginatedSearchBase vm, ICollection<Expression<Func<T, bool>>> whereCollection, IEnumerable<T> context,
        out int totalCount) where T : class
    {
        var query = context.AsQueryable();
        if (whereCollection != null)
        {
            foreach (var w in whereCollection)
            {
                if (w != null)
                {
                    query = query.Where(w);
                }
            }
        }
        // Order by
        query = query.OrderBy($"{vm.OrderBy} {(vm.OrderDescending ? "DESC" : "ASC")}");

        // Total rows
        totalCount = query.Count();

        // Paging
        query = query.Skip((vm.Page - 1)*vm.PageSize).Take(vm.PageSize);
        return query;
    }
}

有關信息,我正在使用版本:

  • “Autofac”:“4.0.0-rc1-177”
  • “Autofac.Extensions.DependencyInjection”:“4.0.0-rc1-177”
  • “AutoMapper”:“4.2.1”

編輯:

我做的一個新測試,用於檢查映射是否真的有效:

var mapper = _container.Resolve<IMapper>();
        var p = new Patient();
        p.Naam = "Test";
        var vm = mapper.Map<PatientViewModel>(p);

        vm.Naam.ShouldBeEquivalentTo("Test");

這個測試通過

編輯2:

當我在Select()中使用Map <>時,它也可以工作,所以它實際上是ProjectTo <>()失敗:

var results = await query.ToListAsync();
        return new PagedResult<TAs>
        {
            Results = results.Select(mapper.Map<TAs>).ToList(),
            Page = vm.Page,
            PageSize = vm.PageSize,
            Total = totalCount
        };

這是有效的,但它需要包含映射器而不是注入,並且它不使用Automapper用於數據庫訪問的ProjectTo ...

我遇到了同樣的問題,但設法讓它工作。 這是由於Automapper最近采取的措施,而不是讓整個API使用靜態方法。 現在一切都是基於實例的,靜態擴展方法不再知道映射配置,現在必須將它們傳遞給方法。 我最終注冊了一個MapperConfiguration實例作為IConfigurationProvider(我正在使用Unity)

container.RegisterInstance(typeof (IConfigurationProvider), config);

這被注入我的查詢處理程序:

[Dependency]
public IConfigurationProvider MapperConfigurationProvider { get; set; }

最后,將MapperConfigurationProvider傳遞給對ProjectTo的調用:

.ProjectTo<Payment>(MapperConfigurationProvider);

希望這可以幫助。

您無需專門將ConfigurationProvider添加到DI。 如果您已經將IMapper添加到DI,那么您可以從Mapper本身讀取ConfigurationProvider。 示例:我遇到了同樣的問題並創建了一個包含IMapper的基類:

public abstract class ServiceBase { public IMapper Mapper { get; set; } }

這個類在我使用AutoMapper的所有服務中繼承。 現在,每當我的任何服務需要映射某些內容時,他們都會這樣做:

    return context.SomeEntity
        .Where(e => e.Id == filter.Id)
        .ProjectTo<EntityDto>(Mapper.ConfigurationProvider).ToList();

隨着Mapper被注入。 只要你把完全配置的Mapper放在DI中,你就可以了。

container.Register(Component.For<IMapper>().UsingFactoryMethod(x =>
    {
        return new AutoMapperConfig().ConfigureMapper();
    })
    );

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM