Clé unique avec code EF en premier

J’ai un modèle suivant dans mon projet

public class Category { public Guid ID { get; set; } [Required(ErrorMessage = "Title cannot be empty")] public ssortingng Title { get; set; } } 

et j’essaie de faire de Title une clé unique, j’ai cherché Google sur la solution, mais je n’ai rien trouvé. Peut-on me suggérer comment le faire, s’il vous plaît?

Malheureusement, vous ne pouvez pas le définir en tant que clé unique dans le code, car EF ne prend pas du tout en charge les clés uniques (il est à espérer que cela sera prévu pour la prochaine version majeure). Vous pouvez créer un intializer de firebase database personnalisé et append manuellement un index unique en appelant la commande SQL:

 public class MyInitializer : CreateDatabaseIfNotExists { protected override void Seed(MyContext context) { context.Database.ExecuteSqlCommand("CREATE UNIQUE INDEX IX_Category_Title ON Categories (Title)"); } } 

Et vous devez définir cet initialiseur dans le bootstrap de votre application.

 Database.SetInitializer(new MyInitializer()); 

modifier

Maintenant (EF 6.1 et suivantes), vous pouvez facilement avoir des contraintes uniques,

 [Index("TitleIndex", IsUnique = true)] public ssortingng Title { get; set; } 

Commencez par créer la classe d’atsortingbuts personnalisée:

 [AtsortingbuteUsage(AtsortingbuteTargets.Property, AllowMultiple = false, Inherited = true)] public class UniqueAtsortingbute : ValidationAtsortingbute { public override Boolean IsValid(Object value) { // constraint implemented on database return true; } } 

Ajoutez ensuite à vos cours:

 public class Email { [Key] public int EmailID { get; set; } public int PersonId { get; set; } [Unique] [Required] [MaxLength(100)] public ssortingng EmailAddress { get; set; } public virtual bool IsDefault { get; set; } public virtual Boolean IsApprovedForLogin { get; set; } public virtual Ssortingng ConfirmationToken { get; set; } [ForeignKey("PersonId")] public virtual Person Person { get; set; } } 

Ajoutez ensuite un initialiseur sur votre DbContext:

 public class Initializer : IDatabaseInitializer { public void InitializeDatabase(myEntities context) { if (System.Diagnostics.Debugger.IsAttached && context.Database.Exists() && !context.Database.CompatibleWithModel(false)) { context.Database.Delete(); } if (!context.Database.Exists()) { context.Database.Create(); var contextObject = context as System.Object; var contextType = contextObject.GetType(); var properties = contextType.GetProperties(); System.Type t = null; ssortingng tableName = null; ssortingng fieldName = null; foreach (var pi in properties) { if (pi.PropertyType.IsGenericType && pi.PropertyType.Name.Contains("DbSet")) { t = pi.PropertyType.GetGenericArguments()[0]; var mytableName = t.GetCustomAtsortingbutes(typeof(TableAtsortingbute), true); if (mytableName.Length > 0) { TableAtsortingbute mytable = mytableName[0] as TableAtsortingbute; tableName = mytable.Name; } else { tableName = pi.Name; } foreach (var piEntity in t.GetProperties()) { if (piEntity.GetCustomAtsortingbutes(typeof(UniqueAtsortingbute), true).Length > 0) { fieldName = piEntity.Name; context.Database.ExecuteSqlCommand("ALTER TABLE " + tableName + " ADD CONSTRAINT con_Unique_" + tableName + "_" + fieldName + " UNIQUE (" + fieldName + ")"); } } } } } } } 

Et pour la dernière fois, ajoutez l’initialiseur à Application_Start dans Global.asax.cs

 System.Data.Entity.Database.SetInitializer(new MyApp.Models.DomainModels.myEntities.Initializer()); 

C’est tout. basé sur le code vb sur https://stackoverflow.com/a/7426773

Voici la version de VB.Net – notez l’implémentation de génériques un peu différents au niveau de la classe.

 Public Class MyInitializer(Of T As DbContext) Inherits CreateDatabaseIfNotExists(Of T) Protected Overrides Sub Seed(context As T) context.Database.ExecuteSqlCommand("CREATE UNIQUE INDEX IX_Category_Title ON Categories (Title)") End Sub End Class 

Je crée cette classe (qui a été améliorée à partir d’une autre réponse Stackoverflow – Exécute un script SQL volumineux (avec les commandes GO) ), ce qui me permet de déposer les scripts SQL dans un répertoire et de les exécuter chaque fois qu’ils sont requirejs (Seed ou migration). Je ne vais pas laisser cela ouvert après mon déploiement en production, mais pendant le développement, il est facile d’appliquer des scripts chaque fois que la firebase database est recréée.

 using System; using System.Collections.Generic; using System.Data.SqlClient; using System.IO; using System.Linq; using System.Text; using System.Threading.Tasks; //dll Microsoft.SqlServer.Smo //dll Microsoft.SqlServer.Management.Sdk.Sfc //dll Microsoft.SqlServer.ConnectionInfo using Microsoft.SqlServer.Management.Common; using Microsoft.SqlServer.Management.Smo; using Monitor.Common; namespace MonitorDB.DataLayer.Migrations { public class ExecuteSQLScripts :Monitor.Common.ExceptionHandling { public ExecuteSQLScripts() { } public bool ExecuteScriptsInDirectory(DBContext.SolArcMsgMonitorContext context, ssortingng scriptDirectory) { bool Result = false; try { SqlConnection connection = new SqlConnection(context.Database.Connection.ConnectionSsortingng); Server server = new Server(new ServerConnection(connection)); DirectoryInfo di = new DirectoryInfo(scriptDirectory); FileInfo[] rgFiles = di.GetFiles("*.sql"); foreach (FileInfo fi in rgFiles) { FileInfo fileInfo = new FileInfo(fi.FullName); ssortingng script = fileInfo.OpenText().ReadToEnd(); server.ConnectionContext.ExecuteNonQuery(script); } Result = true; } catch (Exception ex) { CatchException("ExecuteScriptsInDirectory", ex); } return Result; } 

}}

Voici à quoi ressemble la solution VS:

J’ai trouvé cette solution qui, bien que ne créant pas de clé unique au niveau SQL, utilise la validation DataAnnotations, consultez-la:

http://blogs.microsoft.co.il/blogs/shimmy/archive/2012/01/23/validationatsortingbute-that-valide-a-unique-field-against-its-fellow-rows-in-the-database. aspx