Усилие (EF Unit Testing) дает ошибки

Я пытаюсь выполнить модульное тестирование некоторых классов, которые отвечают на контекст БД Entity Framework. В качестве помощи мне удалось найти библиотеку под названием Effort, которая, кажется, немного устарела и не очень хорошо документирована, но, похоже, работает и довольно популярна.

Я пытаюсь использовать загрузчик данных CSV.

При выполнении ToArray() я получаю исключения, говорящие Sequence contains no matching element.

Любые идеи о том, что я могу делать неправильно? Или, если не другая библиотека, я мог бы дать шанс?

Несколько фрагментов:

[Table("SEC_USER")]
public class SecUser {
    [Key][Column("USERID")]
    public int UserId { get; set; }

    [Column("USERNAME")]
    public string UserName { get; set; }
}

Контекст БД:

public class MusketeerDbContext : DbContext
{
    public virtual IDbSet<IbsCommunity> Communities { get; set; }
    public virtual IDbSet<IbsFunctionLinkLocation> Functionlinklocations { get; set; }
    public virtual IDbSet<IbsInstance> Instances { get; set; }
    public virtual IDbSet<SecUser> Users { get; set; }
    public virtual IDbSet<IbsFieldType> FieldTypes { get; set; }
    public virtual IDbSet<IbsLink> Links { get; set; }
    public virtual IDbSet<IbsFieldFll> FieldFlls { get; set; }
    public virtual IDbSet<IbsFieldValue> FieldValues { get; set; }

    public MusketeerDbContext() : base("name=EGS.My.MySettings.Conn") { }
    public MusketeerDbContext(DbConnection connection) : base(connection, true) { }

    protected override void OnModelCreating(DbModelBuilder modelBuilder)
    {
        modelBuilder.HasDefaultSchema("SA");
        modelBuilder.Conventions.Add(new FunctionConvention(typeof(OracleFunctions)));

        modelBuilder.Entity<IbsFieldValue>()
            .HasMany(fv => fv.InstancesFieldValues)
            .WithMany(i => i.InstancesFieldValues)
            .Map(mm =>
            {
                mm.MapLeftKey("FIELDVALUEID");
                mm.MapRightKey("INSTANCEID");
                mm.ToTable("IBS_INSTANCEFIELDVALUE");
            });
    }
}

public static class OracleFunctions
{
    [Function(FunctionType.BuiltInFunction, "TO_CHAR")]
    public static string ToChar(this int value) => Function.CallNotSupported<string>();

    [Function(FunctionType.BuiltInFunction, "TO_NCHAR")]
    public static string ToChar(this string value) => Function.CallNotSupported<string>();
}

SEC_USER.csv:

USERID,USERNAME
"1","Jonathan"

Тест:

var path = @"C:\...\CSVs";
var dataLoader = new Effort.DataLoaders.CsvDataLoader(path);
var context = Effort.DbConnectionFactory.CreateTransient(dataLoader);
db = new MusketeerDbContext(context);
var users = db.Users.ToArray();

System.InvalidOperationException:

Message: "Sequence contains no matching element"
InnerException: null
StackTrace:
    at System.Linq.Enumerable.Single[TSource](IEnumerable`1 source, Func`2 predicate)
    at System.Data.Entity.Utilities.DbProviderManifestExtensions.GetStoreTypeFromName(DbProviderManifest providerManifest, String name)
    at System.Data.Entity.ModelConfiguration.Configuration.Properties.Primitive.PrimitivePropertyConfiguration.ConfigureColumn(EdmProperty column, EntityType table, DbProviderManifest providerManifest)
    at System.Data.Entity.ModelConfiguration.Configuration.Properties.Primitive.PrimitivePropertyConfiguration.Configure(EdmProperty column, EntityType table, DbProviderManifest providerManifest, Boolean allowOverride, Boolean fillFromExistingConfiguration)
    at System.Data.Entity.ModelConfiguration.Configuration.Properties.Primitive.PrimitivePropertyConfiguration.<>c__DisplayClass4.<Configure>b__3(Tuple`2 pm)
    at System.Data.Entity.Utilities.IEnumerableExtensions.Each[T](IEnumerable`1 ts, Action`1 action)
    at System.Data.Entity.ModelConfiguration.Configuration.Properties.Primitive.PrimitivePropertyConfiguration.Configure(IEnumerable`1 propertyMappings, DbProviderManifest providerManifest, Boolean allowOverride, Boolean fillFromExistingConfiguration)
    at System.Data.Entity.ModelConfiguration.Configuration.Types.StructuralTypeConfiguration.ConfigurePropertyMappings(IList`1 propertyMappings, DbProviderManifest providerManifest, Boolean allowOverride)
    at System.Data.Entity.ModelConfiguration.Configuration.Types.EntityTypeConfiguration.ConfigurePropertyMappings(DbDatabaseMapping databaseMapping, EntityType entityType, DbProviderManifest providerManifest, Boolean allowOverride)
    at System.Data.Entity.ModelConfiguration.Configuration.Types.EntityTypeConfiguration.Configure(EntityType entityType, DbDatabaseMapping databaseMapping, DbProviderManifest providerManifest)
    at System.Data.Entity.ModelConfiguration.Configuration.ModelConfiguration.ConfigureEntityTypes(DbDatabaseMapping databaseMapping, ICollection`1 entitySets, DbProviderManifest providerManifest)
    at System.Data.Entity.ModelConfiguration.Configuration.ModelConfiguration.Configure(DbDatabaseMapping databaseMapping, DbProviderManifest providerManifest)
    at System.Data.Entity.DbModelBuilder.Build(DbProviderManifest providerManifest, DbProviderInfo providerInfo)
    at System.Data.Entity.DbModelBuilder.Build(DbConnection providerConnection)
    at System.Data.Entity.Internal.LazyInternalContext.CreateModel(LazyInternalContext internalContext)
    at System.Data.Entity.Internal.RetryLazy`2.GetValue(TInput input)
    at System.Data.Entity.Internal.LazyInternalContext.InitializeContext()
    at System.Data.Entity.Internal.InternalContext.GetEntitySetAndBaseTypeForType(Type entityType)
    at System.Data.Entity.Internal.Linq.InternalSet`1.Initialize()
    at System.Data.Entity.Internal.Linq.InternalSet`1.GetEnumerator()
    at System.Data.Entity.Infrastructure.DbQuery`1.System.Collections.Generic.IEnumerable<TResult>.GetEnumerator()
    at System.Linq.Buffer`1..ctor(IEnumerable`1 source)
    at System.Linq.Enumerable.ToArray[TSource](IEnumerable`1 source)
    at Igs.Musketeer.DbRepository.Tests.SecurityRepositoryTest..ctor() in C:\Users\me\Source\Repos\Musketeer\Igs.Musketeer.DbRepositoryTests\SecurityRepositoryTest.cs:line 21

person JonathanPeel    schedule 10.05.2017    source источник
comment
Не могли бы вы опубликовать здесь трассировку стека исключений? Я подозреваю, что исключение выбрасывается в методе фреймворка.   -  person Jean Hominal    schedule 10.05.2017
comment
Добавлена ​​трассировка стека.   -  person JonathanPeel    schedule 10.05.2017
comment
Проблема, похоже, в том, как настроен MusketeerDbContext, не могли бы вы опубликовать определение этого класса?   -  person Jean Hominal    schedule 10.05.2017
comment
Для корректности это не модульный тест, а интеграционный тест. Модульный тест использует только данные в памяти, без файлов, без сети, без базы данных.   -  person Ignacio Soler Garcia    schedule 10.05.2017
comment
Я добавил это. Если я закомментирую OnModelCreating, у меня все равно будут те же проблемы. Кроме того, это работает, единственная проблема, с которой я столкнулся, - это модульное тестирование.   -  person JonathanPeel    schedule 10.05.2017
comment
@IgnacioSolerGarcia, что, если я скомпилирую CSV как встроенный ресурс, а затем загрузлю их?   -  person JonathanPeel    schedule 10.05.2017
comment
Выглядит в том же томе, чтобы быть модульным тестом, вам нужно внедрить макет с предопределенным поведением (он всегда возвращает одни и те же фиксированные постоянные данные) вместо доступа к файлу. Unit Test должен быть быстрым, безопасным и воспроизводимым. Просто чтобы вы знали, это не связано напрямую с вашей проблемой.   -  person Ignacio Soler Garcia    schedule 10.05.2017
comment
Спасибо. И я знаю, что могу наследоваться от dbcontext, что, если я не сделаю это в ближайшее время, я мог бы. Я искал что-то легкое.   -  person JonathanPeel    schedule 10.05.2017


Ответы (1)


У меня только что возникла эта проблема, и я обнаружил, что проблема в том, что я использовал аннотацию данных ColumnType. Даже когда я перешел на использование соглашения построителя моделей HasColumnType, я все равно получил точно такую ​​же ошибку. Я предполагаю, что у одного из ваших объектов есть эта аннотация данных (или соглашение)?

В этом случае самым простым решением будет удаление аннотации данных. Если это невозможно, я создал виртуальное свойство (например, public virtual bool IsInMemoryContext { get; } = false;) в DbContext, которое я переопределил в своих тестах — public override bool IsInMemoryContext { get; } = true;, а в методе OnModelCreating я проверил, установлено ли это свойство, прежде чем добавлять типы столбцов.

if (!IsInMemoryContext)
{
    modelBuilder.Entity<AuditLog>()
        .Property(e => e.EventType)
        .HasColumnType("char");
}

Дополнительную информацию об ошибке можно найти здесь: ошибка GitHub с Effort

person Balah    schedule 17.10.2017