Function: Advanced Document Controlled Expiry

  • Capella Operational
      +

      Purge a document automatically based on the document’s self-contained start and duration fields.

      The advancedDocControlledSelfExpiry function:

      • Demonstrates the self-expiry of a document (for example, a user trial)

      • Requires Eventing Storage (or a metadata collection) and a source collection

      • Requires a binding of type bucket alias

      • Processes the initial mutation to calculate and set the TTL of a newly-created document

      • Uses a JavaScript data object to set document expiration

      • Operates on any document where type == "trial_custoimers"

      • Ignores any document with a TTL that is not zero

      When you use a simple integer instead of a proper date or time object for your document’s expiration value, the expiration value is specified in one of the following ways:

      • As an offset from the current time if the value is less than 30 days (60 * 60 * 24 * 30 seconds).

      • As an absolute Unix time stamp if the value is greater than 30 days (60 * 60 * 24 * 30 seconds).

      If a Bucket Max Time-to-Live is set and specified in seconds, it’s enforced as a hard upper limit. Any subsequent document mutation, whether by SQL++, Eventing, or a Couchbase SDK, results in the document having its expiration adjusted and set to the bucket’s maximum TTL if the operation has:

      • No TTL

      • A TTL of zero

      • A TTL greater than the bucket’s TTL

      • advancedDocControlledSelfExpiry

      • Input data

      • Output data

      There are two variants of this function available: a Couchbase Server version 6.6 that relies on SQL++, and a Couchbase Server version 6.6.1+/7.0.0+ that directly sets the expiration.

      You can improve your function’s performance by avoiding N1QL() and using couchbase.replace(bucket_binding, meta, doc) instead.

      The following example directly sets the expiration.

      // Configure the settings for the advancedDocControlledSelfExpiry function as follows:
      //
      // Version 7.1+
      //   "Function Scope"
      //     *.* (or try bulk.data if non-privileged)
      // Version 7.0+
      //   "Listen to Location"
      //     bulk.data.source
      //   "Eventing Storage"
      //     rr100.eventing.metadata
      //   Binding(s)
      //    1. "binding type", "alias name...", "bucket.scope.collection", "Access"
      //       "bucket alias", "src_col",       "bulk.data.source",        "read and write"
      //
      // Version 6.6.1
      //   "Source Bucket"
      //     source
      //   "MetaData Bucket"
      //     metadata
      //   Binding(s)
      //    1. "binding type", "alias name...", "bucket", "Access"
      //       "bucket alias", "src_col",       "source", "read and write"
      
      function OnUpdate(doc, meta) {
          // Filter items that have not been updated
          if (meta.expiration !== 0) {
              log(meta.id, "IGNORE expiration "+meta.expiration+" !== 0 or "+
                  new Date(meta.expiration).toString());
              return;
          }
      
          // Optional filter to a specific field like 'type'
          if (doc.type !== 'trial_customers') return;
      
          // The expiry is based on a JavaScript date parsable field
          if (!doc.trialStartDate || !doc.trialDurationDays) return;
      
          // Convert the doc field timeStamp to Unix epoch time in milliseconds
          var docTimeStampMs = Date.parse(doc.trialStartDate);
      
          var keepDocForMs = doc.trialDurationDays * 1000 * 60 * 60 * 24 ;
          var nowMs = Date.now();  // Get current Unix time in milliseconds
      
          // Archive if it has been kept for too long; you do not need to set an expiration
          if( nowMs >= (docTimeStampMs + keepDocForMs) ) {
      
              // Delete the document from the source collection through the map alias
              delete src_col[meta.id];
      
              log(meta.id, "DELETE from src_col to dst_bkt alias as our expiration " +
                  new Date(docTimeStampMs + keepDocForMs).toString()) + " is already past";
          } else {
              var key = meta.id;
              // Set the meta.expiration=ttlMs
              var	ttlMs = docTimeStampMs + keepDocForMs;
      
              if (ttlMs !== 0) {
                  log(meta.id, "UPDATE expiration "+meta.expiration+" === 0 set to "+
                      ttlMs+" or " + new Date(ttlMs).toString());
                  // Advanced Bucket Accessors use JavaScript Date objects
                  var expiryDate = new Date(ttlMs);
                  // This is 4X to 5X faster than using N1QL(...) and you do not need to worry about recursion
                  var res = couchbase.replace(src_col,{"id":meta.id,"expiry_date":expiryDate},doc);
                  if (!res.success) {
                      log(meta.id,'Setting TTL to',expiryDate,'failed',res);
                  }
              }
          }
      }

      Create a test set of 4 documents using the Query Editor to insert the data items. You do not need an Index.

      If today’s date is past 08-25-2021 (MM-DD-YYYY), you can change the trialStartDate for the last two records to at least 90 days from today.

        UPSERT INTO `bulk`.`data`.`source` (KEY,VALUE)
        VALUES ( "trial_customers::0", {
          "type": "trial_customers",
          "id": 0,
          "trialStartDate": "08-25-2019",
          "trialDurationDays": 30,
          "note": "this is old will get immeadiately deleted"
        } ),
        VALUES ( "trial_customers::1",
        {
          "type": "trial_customers",
          "id": 1,
          "trialStartDate": "01-27-2020",
          "trialDurationDays": 30,
          "note": "this is old will get immeadiately deleted"
        } ),
        VALUES ( "trial_customers::2",
        {
          "type": "trial_customers",
          "id": 2,
          "trialStartDate": "08-25-2021",
          "trialDurationDays": 30,
          "note": "this will get an exiration set"
        } ),
        VALUES ( "trial_customers::3",
        {
          "type": "trial_customers",
          "id": 3,
          "trialStartDate": "08-26-2021",
          "trialDurationDays": 60,
          "note": "this will get an exiration set"
        } );
      NEW/OUTPUT: KEY trial_customers::2
      
      {
        "id": 2,
        "note": "this will get an exiration set",
        "trialDurationDays": 30,
        "trialStartDate": "08-25-2021",
        "type": "trial_customers"
      }
      
      NEW/OUTPUT: KEY trial_customers::3
      
      {
        "id": 3,
        "note": "this will get an exiration set",
        "trialDurationDays": 60,
        "trialStartDate": "08-26-2021",
        "type": "trial_customers"
      }
      
      Returns 2 of the 4 documnents.
      
      * "trial_customers::0" was deleted
      * "trial_customers::1" was deleted
      * "trial_customers::2" has an meta.expiration set for 1632466800 (or 2021-09-24 07:00:00 UTC) in it's metadata
      * "trial_customers::3" has an meta.expiration set for 1635145200 (or 2021-10-25 07:00:00 UTC) in it's metadata