DocumentDB从另一个存储过程或其自身调用存储过程

时间:2015-03-17 19:03:14

标签: stored-procedures recursion azure-cosmosdb

有没有办法以递归方式调用存储过程(甚至UDF,如果它可以工作)对DocumentDB文档?

我们的文档看起来像这样:

{
  "docID" : "my_id",
  "owner" : "fred",
  "items" : [
    {
      "itemID" : "1",
      "type" : "item",
      "value" : 3
    },
    {
      "itemID" : "2",
      "type" : "group",
      "items" : [
        {
          "itemID" : "2.1",
          "type" : "group",
          "items" : [
            {
              "itemID" : "2.1.1",
              "type" : "item",
              "value" : 2
            },
            {
              "itemID" : "2.1.2",
              "type" : "item",
              "value" : 4
            }
          ]
        },
        {
          "itemID" : "2.2",
          "type" : "item",
          "value" : 1
        }
      ]
    }
  ]
}

只要我们"items""items"数组就可以包含"type" : "item""type" : "group"混合的条目。 "type" : "item"的条目具有需要求和的简单"value"字段。 "type" : "group"的条目具有"items"数组......等等。从理论上讲,递归水平没有限制,我承认这是一个问题,但在实践中,水平很少会低于4或5深。

我正在尝试编写的伪代码看起来像这样:

function sumValues(items) {
    int total = 0;
    forEach(item in items) {
        if (item.type == "item") {
            total += item.value;
        } else {
            total += sumValues(item.items);
        }
    }
    return total;
}

function sumAllValues() {
  var ctx = getContext();
  var coll = ctx.getCollection();
  var response = ctx.getResponse();

  // query for docs by owner
  var filterQuery = 'SELECT * FROM Docs d where d.owner = \\\"fred\\\"';
  var done = coll.queryDocuments(coll.getSelfLink(), filterQuery, {},
    function (err, docs, options) {
      if (err) throw new Error ('Error' + err.message);

      var total = 0;
      docs.forEach(function(doc) {
        total += sumTotals(doc.items);
      });

      response.setBody('Total: ' + total);
    });
}

这甚至可能吗? DocumentDB是否支持从另一个sproc调用sproc?一个sproc可以自己调用吗?

我在网上找到了一些DocumentDB存储过程参考,包括thisthis以及thisthis以及许多其他网页。

如果有可能,我想我可能不得不以某种方式查询集合以获取我想要调用的sproc,然后以某种方式引用sproc,而不是直接调用sumTotals(),就像单独调用它一样语言。

我们刚刚开始研究使用DocumentDB编程,所以我们还不能完全确定我们可以用它做什么。感谢您的任何帮助或建议。

1 个答案:

答案 0 :(得分:6)

我认为你现在正走在正确的轨道上。

无法从存储过程中执行存储过程。

但是,您可以在存储过程中定义JS函数,可以在该存储过程中引用,调用和重用它们。

在这种情况下,只需在父sumValues()存储过程中定义sumAllValues()函数(就像您提到的swapItems()示例一样)。

function sumAllValues() {
    var ctx = getContext();
    var coll = ctx.getCollection();
    var response = ctx.getResponse();

    // query for docs by owner
    var filterQuery = 'SELECT * FROM Docs d where d.owner = \\\"fred\\\"';
    var done = coll.queryDocuments(coll.getSelfLink(), filterQuery, {},
        function (err, docs, options) {
            if (err) throw new Error ('Error' + err.message);

            var total = 0;
            docs.forEach(function(doc) {
                total += sumValues(doc.items);
            });

            response.setBody('Total: ' + total);
        });

    function sumValues(items) {
        int total = 0;
        items.forEach(function(item) {
            if (item.type == "item") {
                total += item.value;
            } else {
                total += sumValues(item.items);
            }
        });
        return total;
    }
}

您还可以为要在多个存储过程和查询中共享和重用的逻辑定义UDF。