Создание уникального ключа MongoDB с С#
Я борюсь за создание уникального поля EmailAddress
. Я уже видел на форумах, что мне нужно создать индекс, но пока у меня это не получалось. У кого-нибудь есть пример кода? Нужно ли создавать индекс при каждом сохранении/вызове или достаточно создать его только один раз?
Я попробовал этот код:
DB.GetCollection<User>(Dbname)
.EnsureIndex(new IndexKeysBuilder()
.Ascending("EmailAddress"), IndexOptions.SetUnique(true));
DB.GetCollection<User>(Dbname).Save(user, SafeMode.True);
Моя модель User
выглядит следующим образом:
public class User
{
[Required(ErrorMessage = "Email Required")]
public string EmailAddress { get; set; }
public ObjectId Id { get; set; }
public string FirstName { get; set; }
public string LastName { get; set; }
}
Ответы
Ответ 1
Уникальный индекс нужно создать только один раз, после чего все вставки документа, содержащие дублированный адрес электронной почты, будут сбой. Вот пример:
var server = MongoServer.Create("mongodb://localhost");
var db = server.GetDatabase("myapp");
var users = db.GetCollection<User>("users");
users.EnsureIndex(new IndexKeysBuilder()
.Ascending("EmailAddress"), IndexOptions.SetUnique(true));
var user1 = new User { EmailAddress = "[email protected]" };
var user2 = new User { EmailAddress = "[email protected]" };
try
{
users.Save(user1, WriteConcern.Acknowledged);
users.Save(user2, WriteConcern.Acknowledged); // <-- throws MongoSafeModeException
}
catch (MongoSafeModeException ex)
{
Console.WriteLine(ex.Message);
}
Ответ 2
EnsureIndex() устарел/устарел согласно спецификации драйверов С# mongo версии 2.0: http://api.mongodb.org/csharp/current/html/M_MongoDB_Driver_MongoCollection_EnsureIndex_2.htm
Вот как это сделать асинхронно и с помощью кода 2.0:
var mongoClient = new MongoClient("connection");
var db = mongoClient.GetDatabase("database");
var options = new CreateIndexOptions() { Unique = true };
var field = new StringFieldDefinition<User>("EmailAddress");
var indexDefinition = new IndexKeysDefinitionBuilder<User>().Ascending(field);
await db.GetCollection<Users>("users").Indexes.CreateOneAsync(indexDefinition, options);
В случае нестрокового индекса:
var options = new CreateIndexOptions() { Unique = true };
IndexKeysDefinition<Foo> keyCode = "{ Code: 1 }";
var codeIndexModel = new CreateIndexModel<Foo>(keyCode, options);
Ответ 3
Ваш код выглядит правильно. Вот вам полная программа для сравнения:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using MongoDB.Bson;
using MongoDB.Driver;
using MongoDB.Driver.Builders;
namespace TestEnsureIndexOnEmailAddress {
public class User {
public ObjectId Id;
public string FirstName;
public string LastName;
public string EmailAddress;
}
public static class Program {
public static void Main(string[] args) {
var server = MongoServer.Create("mongodb://localhost/?safe=true");
var database = server["test"];
var users = database.GetCollection<User>("users");
if (users.Exists()) { users.Drop(); }
users.EnsureIndex(IndexKeys.Ascending("EmailAddress"), IndexOptions.SetUnique(true));
var john = new User { FirstName = "John", LastName = "Smith", EmailAddress = "[email protected]" };
users.Insert(john);
var joe = new User { FirstName = "Joe", LastName = "Smith", EmailAddress = "[email protected]" };
users.Insert(joe); // should throw exception
}
}
}
Вы также можете использовать оболочку mongo для подтверждения создания индекса:
> db.users.getIndexes()
[
{
"name" : "_id_",
"ns" : "test.users",
"key" : {
"_id" : 1
},
"v" : 0
},
{
"_id" : ObjectId("4de8152ee447ad2550e3b5fd"),
"name" : "EmailAddress_1",
"ns" : "test.users",
"key" : {
"EmailAddress" : 1
},
"unique" : true,
"v" : 0
}
]
>
Ответ 4
Начиная с версии 2.8 ниже представлен способ создания индекса. Обратите внимание на последние две строки. CreateOneAsync (indexDefinition, параметры) устарел.
var mongoClient = new MongoClient("connection");
var db = mongoClient.GetDatabase("database");
var options = new CreateIndexOptions() { Unique = true };
var field = new StringFieldDefinition<User>("EmailAddress");
var indexDefinition = new IndexKeysDefinitionBuilder<User>().Ascending(field);
var indexModel = new CreateIndexModel<User>(indexDefinition,options);
await db.GetCollection<Users>("users").Indexes.CreateOneAsync(indexModel);
Ответ 5
здесь рабочая программа с использованием MongoDB.Entities (отказ от ответственности: я автор)
using System;
using MongoDB.Driver;
using MongoDB.Entities;
namespace StackOverflow
{
public class User : Entity
{
public string Name { get; set; }
public string EmailAddress { get; set; }
}
class Program
{
static void Main(string[] args)
{
new DB("test");
DB.Index<User>()
.Options(o => o.Unique = true, o => o.Background = false)
.Key(u => u.EmailAddress, Type.Ascending)
.Create();
var user1 = new User { Name = "First User", EmailAddress = "[email protected]" };
user1.Save();
try
{
var user2 = new User { Name = "Second User", EmailAddress = "[email protected]" };
user2.Save();
}
catch (MongoWriteException x)
{
Console.WriteLine(x.Message);
Console.ReadKey();
}
}
}
}
попытка создать второго пользователя с тем же адресом электронной почты приводит к следующему исключению:
A write operation resulted in an error. E11000 duplicate key error collection: test.User index: EmailAddress(Asc) dup key: {: "[email protected]" }