MongoDB
 sql >> डेटाबेस >  >> NoSQL >> MongoDB

MongoDB में arrayFilters के साथ नेस्टेड उप-दस्तावेज़ अपडेट करें

तो arrayFilters स्थितीय फ़िल्टर के साथ विकल्प $[<identifier>] मोंगोडीबी 3.5.12 के बाद से विकास रिलीज श्रृंखला के साथ वास्तव में ठीक से काम करता है और मोंगोडीबी 3.6 श्रृंखला के लिए वर्तमान रिलीज उम्मीदवारों में भी, जहां यह वास्तव में आधिकारिक तौर पर जारी किया जाएगा। एकमात्र समस्या यह है कि उपयोग में आने वाले "ड्राइवर" वास्तव में अभी तक इसे नहीं पकड़ पाए हैं।

उसी सामग्री को फिर से दोहराना जो मैंने पहले से ही MongoDB के साथ नेस्टेड ऐरे को अपडेट करने पर रखा है:

<ब्लॉकक्वॉट>

नोट कुछ हद तक विडंबना यह है कि यह .update() के लिए "विकल्प" तर्क में निर्दिष्ट है। और विधियों की तरह, सिंटैक्स आम तौर पर सभी हाल के रिलीज़ ड्राइवर संस्करणों के साथ संगत है।

हालांकि यह mongo . के बारे में सच नहीं है खोल, जिस तरह से वहां विधि लागू की गई है ("विडंबना यह है कि पिछड़े संगतता के लिए") arrayFilters तर्क को एक आंतरिक विधि द्वारा पहचाना और हटाया नहीं जाता है जो पिछले MongoDB सर्वर संस्करणों और "विरासत" .update() के साथ "पिछड़ा संगतता" देने के लिए विकल्पों को पार्स करता है। एपीआई कॉल सिंटैक्स।

इसलिए यदि आप mongo . में कमांड का उपयोग करना चाहते हैं शेल या अन्य "खोल आधारित" उत्पाद (विशेषकर रोबो 3T) आपको विकास शाखा से नवीनतम संस्करण या 3.6 या उससे अधिक के उत्पादन रिलीज की आवश्यकता है।

इसका मतलब यह है कि .update() . का वर्तमान "ड्राइवर" कार्यान्वयन वास्तव में arrayFilters . की परिभाषा के साथ आवश्यक तर्कों को "हटा" देता है . NodeJS के लिए इसे ड्राइवर की 3.x रिलीज़ श्रृंखला में संबोधित किया जाएगा, और निश्चित रूप से "मोंगोज़" को उस रिलीज़ के बाद अपडेट किए गए ड्राइवर पर अपनी निर्भरता को लागू करने में कुछ समय लगेगा, जो अब "स्ट्रिप" नहीं होगा। ऐसी कार्रवाइयां।

हालांकि आप इसे अभी भी समर्थित . पर चला सकते हैं सर्वर इंस्टेंस, मूल "अपडेट कमांड" सिंटैक्स उपयोग को वापस छोड़कर, क्योंकि यह कार्यान्वित ड्राइवर विधि को छोड़ देता है:

const mongoose = require('mongoose'),
      Schema = mongoose.Schema,
      ObjectId = mongoose.Types.ObjectId;

mongoose.Promise = global.Promise;
mongoose.set('debug',true);

const uri = 'mongodb://localhost/test',
      options = { useMongoClient: true };

const contactSchema = new Schema({
  data: String,
  type: String,
  priority: String,
  retries: String
});

const personSchema = new Schema({
  name: String,
  level: String,
  priority: String,
  enabled: Boolean,
  contacts: [contactSchema]
});

const groupSchema = new Schema({
  name: String,
  people: [personSchema],
  workingHours: { start: String, end: String },
  workingDays: { type: [Number], default: undefined },
  contactTypes: {
    workingHours: { type: [String], default: undefined },
    contactTypes: { type: [String], default: undefined }
  }
});

const Group = mongoose.model('Group', groupSchema);

function log(data) {
  console.log(JSON.stringify(data, undefined, 2))
}

(async function() {

  try {

    const conn = await mongoose.connect(uri,options);

    // Clean data
    await Promise.all(
      Object.entries(conn.models).map(([k,m]) => m.remove() )
    );

    // Create sample

    await Group.create({
      name: "support",
      people: [
        {
          "_id": ObjectId("5a05a8c3e0ce3444f8ec5bd8"),
          "enabled": true,
          "level": "1",
          "name": "Someone",
          "contacts": [
            {
              "type": "email",
              "data": "[email protected]"
            },
            {
              "_id": ObjectId("5a05a8dee0ce3444f8ec5bda"),
              "retries": "1",
              "priority": "1",
              "type": "email",
              "data": "[email protected]"
            }
          ]
        }
      ]
    });

    let result = await conn.db.command({
      "update": Group.collection.name,
      "updates": [
        {
          "q": {},
          "u": { "$set": { "people.$[i].contacts.$[j].data": "new data" } },
          "multi": true,
          "arrayFilters": [
            { "i._id": ObjectId("5a05a8c3e0ce3444f8ec5bd8") },
            { "j._id": ObjectId("5a05a8dee0ce3444f8ec5bda") }
          ]
        }
      ]
    });

    log(result);

    let group = await Group.findOne();
    log(group);

  } catch(e) {
    console.error(e);
  } finally {
    mongoose.disconnect();
  }

})()

चूंकि यह "कमांड" को सीधे सर्वर के माध्यम से भेजता है, हम देखते हैं कि अपेक्षित अपडेट वास्तव में होता है:

Mongoose: groups.remove({}, {})
Mongoose: groups.insert({ name: 'support', _id: ObjectId("5a06557fb568aa0ad793c5e4"), people: [ { _id: ObjectId("5a05a8c3e0ce3444f8ec5bd8"), enabled: true, level: '1', name: 'Someone', contacts: [ { type: 'email', data: '[email protected]', _id: ObjectId("5a06557fb568aa0ad793c5e5") }, { _id: ObjectId("5a05a8dee0ce3444f8ec5bda"), retries: '1', priority: '1', type: 'email', data: '[email protected]' } ] } ], __v: 0 })
{ n: 1,
  nModified: 1,
  opTime:
   { ts: Timestamp { _bsontype: 'Timestamp', low_: 3, high_: 1510364543 },
     t: 24 },
  electionId: 7fffffff0000000000000018,
  ok: 1,
  operationTime: Timestamp { _bsontype: 'Timestamp', low_: 3, high_: 1510364543 },
  '$clusterTime':
   { clusterTime: Timestamp { _bsontype: 'Timestamp', low_: 3, high_: 1510364543 },
     signature: { hash: [Object], keyId: 0 } } }
Mongoose: groups.findOne({}, { fields: {} })
{
  "_id": "5a06557fb568aa0ad793c5e4",
  "name": "support",
  "__v": 0,
  "people": [
    {
      "_id": "5a05a8c3e0ce3444f8ec5bd8",
      "enabled": true,
      "level": "1",
      "name": "Someone",
      "contacts": [
        {
          "type": "email",
          "data": "[email protected]",
          "_id": "5a06557fb568aa0ad793c5e5"
        },
        {
          "_id": "5a05a8dee0ce3444f8ec5bda",
          "retries": "1",
          "priority": "1",
          "type": "email",
          "data": "new data"            // <-- updated here
        }
      ]
    }
  ]
}

तो ठीक है "अभी" "ऑफ़ द शेल्फ़" उपलब्ध ड्राइवर वास्तव में .update() . को लागू नहीं करते हैं या यह अन्य कार्यान्वयन समकक्ष इस तरह से है जो वास्तव में आवश्यक arrayFilters से गुजरने के अनुकूल है बहस। इसलिए यदि आप एक विकास श्रृंखला या रिलीज कैंडिडेट सर्वर के साथ "खेल" रहे हैं, तो आपको वास्तव में "ब्लीडिंग एज" और अप्रकाशित ड्राइवरों के साथ भी काम करने के लिए तैयार रहना चाहिए।

लेकिन आप वास्तव में ऐसा कर सकते हैं जैसा कि किसी भी ड्राइवर में दिखाया गया है, सही रूप में जहां जारी किया जा रहा आदेश बदला नहीं जा रहा है।

<ब्लॉकक्वॉट>

11 नवंबर 2017 को लिखे जाने तक कोई "आधिकारिक" . नहीं है MongoDB या समर्थित ड्राइवरों की रिहाई जो वास्तव में इसे लागू करते हैं। उत्पादन का उपयोग केवल सर्वर और समर्थित ड्राइवरों के आधिकारिक रिलीज पर आधारित होना चाहिए।




  1. Redis
  2.   
  3. MongoDB
  4.   
  5. Memcached
  6.   
  7. HBase
  8.   
  9. CouchDB
  1. नेवला स्कीमा निर्माण

  2. MongoDB संग्रह के अंदर एक विशिष्ट एम्बेडेड दस्तावेज़ कैसे प्राप्त करें?

  3. MongoDB इंडेक्स बनाता है - उपयोगकर्ताओं को नए बिल्ड को ट्रिगर करने से रोकता है

  4. नेवला किसी मौजूदा क्षेत्र के लिए अपरिभाषित लौटाता है

  5. मोंगोडीबी गिनती कमांड