sql >> Databáze >  >> NoSQL >> MongoDB

MongoDB:Je možné zachytit události TTL pomocí Change Stream pro emulaci plánovače (cronjob)?

Byl jsem schopen použít Change Streams a TTL k emulaci cronjob. Zveřejnil jsem příspěvek vysvětlující podrobně, co jsem udělal, a uvedl poděkování na adrese:https://www. patreon.com/posts/17697287

Ale v zásadě kdykoli potřebuji naplánovat „událost“ pro dokument, když vytvářím dokument, vytvořím paralelně také dokument události. Tento dokument události bude mít jako _id stejné id jako první dokument.

Také pro tento dokument události nastavím TTL.

Když vyprší TTL, zachytím jeho změnu "smazat" pomocí Change Streams. A pak použiji documentKey změny (protože je to stejné ID jako dokument, který chci spustit) k nalezení cílového dokumentu v první kolekci a udělám s dokumentem, co chci.

Pro přístup k MongoDB používám Node.js s Express a Mongoose. Zde je relevantní část, kterou je třeba přidat do App.js:

const { ReplSet } = require('mongodb-topology-manager');

run().catch(error => console.error(error));

async function run() {
    console.log(new Date(), 'start');
    const bind_ip = 'localhost';
    // Starts a 3-node replica set on ports 31000, 31001, 31002, replica set
    // name is "rs0".
    const replSet = new ReplSet('mongod', [
        { options: { port: 31000, dbpath: `${__dirname}/data/db/31000`, bind_ip } },
        { options: { port: 31001, dbpath: `${__dirname}/data/db/31001`, bind_ip } },
        { options: { port: 31002, dbpath: `${__dirname}/data/db/31002`, bind_ip } }
    ], { replSet: 'rs0' });

    // Initialize the replica set
    await replSet.purge();
    await replSet.start();
    console.log(new Date(), 'Replica set started...');

    // Connect to the replica set
    const uri = 'mongodb://localhost:31000,localhost:31001,localhost:31002/' + 'test?replicaSet=rs0';
    await mongoose.connect(uri);
    var db = mongoose.connection;
    db.on('error', console.error.bind(console, 'connection error:'));
    db.once('open', function () {
        console.log("Connected correctly to server");
    });

    // To work around "MongoError: cannot open $changeStream for non-existent database: test" for this example
    await mongoose.connection.createCollection('test');

    // *** we will add our scheduler here *** //

    var Item = require('./models/item');
    var ItemExpiredEvent = require('./models/scheduledWithin');

    let deleteOps = {
      $match: {
          operationType: "delete" 
      }
    };

    ItemExpiredEvent.watch([deleteOps]).
        on('change', data => {
            // *** treat the event here *** //
            console.log(new Date(), data.documentKey);
            Item.findById(data.documentKey, function(err, item) {
                console.log(item);
            });
        });

    // The TTL set in ItemExpiredEvent will trigger the change stream handler above
    console.log(new Date(), 'Inserting item');
    Item.create({foo:"foo", bar: "bar"}, function(err, cupom) {
        ItemExpiredEvent.create({_id : item._id}, function(err, event) {
            if (err) console.log("error: " + err);
            console.log('event inserted');
        });
    });

}

A zde je kód pro model/ScheduledWithin:

var mongoose = require('mongoose');
var Schema = mongoose.Schema;

var ScheduledWithin = new Schema({
    _id: mongoose.Schema.Types.ObjectId,
}, {timestamps: true}); 
// timestamps: true will automatically create a "createdAt" Date field

ScheduledWithin.index({createdAt: 1}, {expireAfterSeconds: 90});

module.exports = mongoose.model('ScheduledWithin', ScheduledWithin);


  1. MongoDB 3.6.2 2008R2 Plus se neinstaluje

  2. K čemu slouží dvojtečky v klíčích Redis

  3. jarní data mongodb mapování dynamického pole

  4. mongodb-java POJO nevrací id