Elemento de matriz de actualización Mongo (controlador .NET 2.0)

EDITAR: no busca la forma javascript de hacer esto. Estoy buscando la forma de hacer esto del controlador MongoDB C # 2.0 (sé que podría no ser posible, pero espero que alguien sepa una solución).

Estoy tratando de actualizar el valor de un elemento incrustado en una matriz en el documento principal en mi mongodb.

Estoy buscando una forma fuertemente tipada para hacer esto. Estoy usando el controlador Mongodb c # 2.0

Puedo hacerlo mostrando el elemento, actualizando el valor y luego reinsertándolo. Esto simplemente no se siente bien; ya que estoy sobrescribiendo lo que podría haberse escrito mientras tanto.

Esto es lo que he intentado hasta ahora pero sin suerte:

private readonly IMongoCollection _collection; void Main() { var collectionName = "Agenda"; var client = new MongoClient("mongodb://localhost:27017"); var db = client.GetDatabase("Test"); _collection = db.GetCollection(collectionName); UpdateItemTitle(1, 1, "hello"); } public void UpdateItemTitle(string agendaId, string itemId, string title){ var filter = Builders.Filter.Eq(x => x.AgendaId, agendaId); var update = Builders.Update.Set(x => x.Items.Single(p => p.Id.Equals(itemId)).Title, title); var result = _collection.UpdateOneAsync(filter, update).Result; } 

Me tomó un tiempo resolver esto ya que no parece ser mencionado en ninguna de la documentación oficial (ni en ningún otro lado). Sin embargo, encontré esto en su rastreador de problemas, que explica cómo usar el operador posicional $ con el controlador C # 2.0.

Esto debería hacer lo que quieras:

 public void UpdateItemTitle(string agendaId, string itemId, string title){ var filter = Builders.Filter.Where(x => x.AgendaId == agendaId && x.Items.Any(i => i.Id == itemId)); var update = Builders.Update.Set(x => x.Items[-1].Title, title); var result = _collection.UpdateOneAsync(filter, update).Result; } 

Observe que su cláusula Item.Single() se ha cambiado a Item.Any() y se ha movido a la definición del filtro.

[-1] o .ElementAt(-1) aparentemente se trata de manera especial (en realidad todo <0) y se reemplazará con el operador posicional $ .

Lo anterior se traducirá a esta consulta:

 db.Agenda.update({ AgendaId: 1, Items.Id: 1 }, { $set: { Items.$.Title: "hello" } }) 

Gracias, esto fue útil. Aunque tengo una adición, he usado lo anterior para las matrices, empujando a una matriz anidada y tirando de una. El problema que he encontrado es que si tuviera una matriz int (Así que no es un objeto, simplemente una matriz int simple) que el PullFilter realmente no funcionaba: “No se puede determinar la información de serialización”, lo cual es extraño ya que solo es una matriz de los ints. Lo que terminé haciendo fue convertirlo en una matriz de objetos con solo un parámetro int, y todo comenzó a funcionar. Posiblemente un error, o tal vez mi falta de comprensión. De todos modos, como he luchado para encontrar información acerca de tirar y empujar a matrices de objetos nesteds con el controlador C # 2.0, pensé que debería publicar mis hallazgos aquí, ya que utilizan la syntax anterior.

 var filter = Builders.Filter.Where(x => x._id == entity.ParentID && x.NestedArray.Any(i => i._id == entity._id)); var update = Builders.Update.PullFilter(x => x.NestedArray.ElementAt(-1).User, Builders.Filter.Eq(f => f.UserID, userID)); Collection(currentUser).UpdateOneAsync(filter, update); 

Y también:

 var filter = Builders.Filter.Where(x => x._id == entity.ParentID && x.NestedArray.Any(i => i._id == entity._id)); var update = Builders.Update.Push(x => x.NestedArray.ElementAt(-1).Users, new User { UserID = userID }); Collection(currentUser).UpdateOneAsync(filter, update); 

La forma correcta de actualizar un documento o matriz secundaria es la siguiente:

 var filter = Builders.Filter.Where(x => x.Id == di && x.RemuneracionMensualActual.RemuneracionActIndustrial.Any(s => s.Id == oid)); var update = Builders.Update.Set(x => x.RemuneracionMensualActual.RemuneracionActIndustrial.ElementAt(-1).Ingreso, datos.ActividadIndustrial.Ingreso) .Set(x => x.RemuneracionMensualActual.RemuneracionActIndustrial.ElementAt(-1).RazonSocial, datos.ActividadIndustrial.RazonSocial) .Set(x => x.RemuneracionMensualActual.RemuneracionActIndustrial.ElementAt(-1).TipoNegocio, datos.ActividadIndustrial.TipoNegocio);