如何通过文档快照updateTime查询Firestore?

时间:2019-02-03 12:05:37

标签: node.js google-cloud-firestore

是否可以通过updateTime查询Firestore文档。可从文档快照中以doc.updateTime格式访问的字段,该字段可在where查询中使用?

我正在使用node.js SDK。

2 个答案:

答案 0 :(得分:1)

据我所知,无法查询Firestore自动维护的元数据。如果您需要查询最近的更新日期,则需要在文档数据中添加一个具有该值的字段。

答案 1 :(得分:1)

我确实需要在文档_updateTime上查询Firebase,因此编写了一个函数,将隐藏的内部时间戳记复制到可查询的字段。这需要一些工作来解决,所以我要发布完整的解决方案。 (从技术上讲,这是“ Cloud Firestore”,而不是“实时数据库”。)

这是使用Firebase Functions完成的,Firebase Functions本身需要进行一些尝试才能开始工作。本教程很有帮助:

https://firebase.google.com/docs/functions/get-started

但是,在Windows 10上,唯一起作用的命令行是新的Bash shell,自2017年以来可用。这是安装的一种解决方法,但有必要。 GIT Bash Shell否则很有用,无法在Firebase项目设置期间跟踪屏幕位置。

在我的示例代码中,我保留了所有的'console.log'语句,以显示详细信息。起初并不明显是这些日志的去向。他们不转到命令行,而是转到Firebase控制台:

https://console.firebase.google.com/u/0/

在(您的项目)>功能>日志中

对于测试,我发现起初仅部署一个功能(在CLI中)很有用:

firebase deploy --only functions:testFn

下面是我的工作功能,经过大量评论,并留有一些多余的插图。将“ PlantSpp”替换为文档集合的名称:

// The Cloud Functions for Firebase SDK to create Cloud Functions and setup triggers.
const functions = require('firebase-functions');

// The Firebase Admin SDK to access the Firebase Realtime Database.
const admin = require('firebase-admin');
admin.initializeApp();

// Firestore maintains an interal _updateTime for every document, but this is
// not queryable. This function copies that to a visible field 'Updated'
exports.makeUpdateTimeVisible = functions.firestore
      .document('PlantSpp/{sppId}')
      .onWrite((sppDoc, context) => {
  console.log("Event type: ", context.eventType);
  // context.eventType = 'google.firestore.document.write', so cannot use
  // to distinguish e.g. create from update
  const docName = context.params.sppId // this is how to get the document name
  console.log("Before: ", sppDoc.before); // if a create, a 'DocumentSnapshot',
  // otherwise a 'QueryDocumentSnapshot'
  // if a create, everything about sppDoc.before is undefined
  if (typeof sppDoc.before._fieldsProto === "undefined"){
    console.log('document "', docName, '" has been created');
    // set flags here if desired
  }
  console.log("After: ", sppDoc.after); // if a delete, a 'DocumentSnapshot',
  // otherwise a 'QueryDocumentSnapshot'
  // if a delete, everything about sppDoc.after is undefined
  if (typeof sppDoc.after._fieldsProto === "undefined"){
    console.log('document "', docName, '" has been deleted');
    // other fields could be fetched from sppDoc.before
    return null; // no need to proceed
  }
  console.log(sppDoc.after.data()); // the user defined fields:values
  // inside curly braces
  console.log(sppDoc.after._fieldsProto); // similar to previous except with
  // data types, e.g.
  // data() has { Code: 'OLDO',...
  // _fieldsProto has { Code: { stringValue: 'OLDO' },...
  const timeJustUpdated = sppDoc.after._updateTime; // this is how to get the
  // internal nonqueryable timestamp
  console.log(timeJustUpdated);
  //  e.g.      Timestamp { _seconds: 1581615533, _nanoseconds: 496655000 }
  //  later:    Timestamp { _seconds: 1581617552, _nanoseconds: 566223000 }
  // shows this is correctly updating
  // see if the doc has the 'Updated' field yet
  if (sppDoc.after._fieldsProto.hasOwnProperty('Updated')) {
    console.log("doc has the field 'Updated' with the value",
                  sppDoc.after._fieldsProto.Updated);
    console.log("sppDoc:", sppDoc);
    const secondsInternal = timeJustUpdated._seconds;
    console.log(secondsInternal, "seconds, internal timestamp");
    const secondsExternal = sppDoc.after.data().Updated._seconds;
    console.log(secondsExternal, "seconds, external timestamp");
    // Careful here. If we just update the externally visible time to the
    // internal time, we will go into an infinite loop because that update
    // will call this function again, and by then the internal time will have
    // advanced
    // the following exit will not work:
    if (secondsInternal === secondsExternal) return null; // will never exit
    // instead, allow the external time to lag the internal by a little
    const secondsLate = secondsInternal - secondsExternal;
    if (secondsLate < 120) { // two minutes sufficient for this purpose
      console.log("the field 'Updated' is", secondsLate,
                  "seconds late, good enough");
       return null;
    }
    console.log("the field 'Updated' is", secondsLate,
                  "seconds late, updating");
    // return a promise of a set operation to update the timestamp
    return sppDoc.after.ref.set({
      Updated: timeJustUpdated
    }, {merge: true}); // 'merge' prevents overwriting whole doc
    // this change will call this same function again
  } else { // field 'Updated' does not exist in the document yet
    // this illustrates how to add a field
    console.log("doc does not have the field 'Updated', adding it now.");
    // return a promise of a set operation to create the timestamp
    return sppDoc.after.ref.set({
      Updated: timeJustUpdated
    }, {merge: true}); // 'merge' prevents overwriting the whole doc
    // this change will call this same function again
  }
});