У меня сложная ситуация: база данных стран/регионов/штатов/городов, первичный ключ которой состоит из кода (nvarchar(3)) в столбце с именем «Id» плюс все ключевые столбцы «предков» (регионы/штаты/ города).
Таким образом, в таблице country есть только один ключевой столбец (Id), а в city — 4 ключевых столбца (Id, StateId, regionId, CountryId). Очевидно, что все они связаны, поэтому каждый столбец-предок является внешним ключом для связанной таблицы.
У меня есть объекты в моей модели, которые отображают эти отношения. Но все они происходят от одного типа, называемого Entity‹T›, где T может быть простым типом (строка, in и т. д.) или сложным (компонент, реализующий ключ). Entity‹T› реализует одно свойство Id типа T.
Для каждой таблицы db, если у нее есть ключ comlex, я реализую его в отдельном компоненте, который переопределяет также методы Equals и GetHashCode() (в будущем я реализую их в базовом классе Entity).
Итак, у меня есть компонент RegionKey, который имеет 2 свойства (Id и CountryId). У меня есть соглашения для именования и типа внешнего ключа и первичного ключа, и это нормально. У меня также есть Mapping ovverrides для каждой сложной сущности.
Для простоты давайте сосредоточимся только на таблице «Страны и регионы». Они здесь:
public class Country: Entity<string>
{
public virtual string Name { get; set; }
public virtual IList<Region> Regions { get; set; }
}
public class Region: Entity<RegionKey>
{
public virtual string Name { get; set; }
public virtual Country Country { get; set; }
}
и компонент RegionKey:
namespace Hell.RealHellState.Api.Entities.Keys
{
[Serializable]
public class RegionKey
{
public virtual string Id { get; set; }
public virtual string CountryId { get; set; }
public override bool Equals(object obj)
{
if (obj == null)
return false;
var t = obj as RegionKey;
if (t == null)
return false;
return Id == t.Id && CountryId == t.CountryId;
}
public override int GetHashCode()
{
return (Id + "|" + CountryId).GetHashCode();
}
}
}
Вот конфигурация AutoPersistenceModel:
public ISessionFactory CreateSessionFactory()
{
return Fluently.Configure()
.Database(
MsSqlCeConfiguration.Standard
.ConnectionString(x=>x.Is(_connectionString))
)
.Mappings(m => m.AutoMappings.Add(AutoMappings))
.ExposeConfiguration(BuildSchema)
.BuildSessionFactory();
}
private AutoPersistenceModel AutoMappings()
{
return AutoMap.Assembly(typeof (Country).Assembly)
.IgnoreBase(typeof(Entity<>))
.Conventions.AddFromAssemblyOf<DataFacility>()
.UseOverridesFromAssembly(GetType().Assembly)
.Where(type => type.Namespace.EndsWith("Entities"));
}
private static void BuildSchema(Configuration config)
{
//Creates database structure
new SchemaExport(config).Create(false, true);
//new SchemaUpdate(config).Execute(false, true);
}
Вот переопределение объекта «Регионы»
public class RegionMappingOverride : IAutoMappingOverride<Region>
{
public void Override(AutoMapping<Region> mapping)
{
mapping.CompositeId(x=>x.Id)
.KeyProperty(x => x.Id, x=> x.ColumnName("Id").Length(3).Type(typeof(string)))
.KeyProperty(x => x.CountryId, x => x.ColumnName("CountryId").Length(3).Type(typeof(string)));
}
}
Хорошо, теперь, когда я тестирую это сопоставление, я получаю сообщение об ошибке: Типы данных столбцов в отношениях не совпадают.
Я также пробовал это переопределение:
public void Override(AutoMapping<Region> mapping)
{
mapping.CompositeId()
.ComponentCompositeIdentifier(x=>x.Id)
.KeyProperty(x => x.Id.Id, x=> x.ColumnName("Id").Length(3).Type(typeof(string)))
.KeyProperty(x => x.Id.CountryId, x => x.ColumnName("CountryId").Length(3).Type(typeof(string)));
}
И это почти работает, но создает таблицу регионов с одним ключом столбца varbinary(8000), а это не то, что мне нужно:
CREATE TABLE [hell_Regions] (
[Id] varbinary(8000) NOT NULL
, [Name] nvarchar(50) NULL
, [CountryId] nvarchar(3) NULL
);
GO
ALTER TABLE [hell_Regions] ADD CONSTRAINT [PK__hell_Regions__0000000000000153] PRIMARY KEY ([Id]);
GO
ALTER TABLE [hell_Regions] ADD CONSTRAINT [FK_Regions_Country] FOREIGN KEY ([CountryId]) REFERENCES [hell_Countries]([Id]) ON DELETE NO ACTION ON UPDATE NO ACTION;
GO
Я понятия не имею, как с этим бороться, потому что мне кажется, что все в порядке.
Заранее спасибо за ваши ответы