From d914137ad492023254bfa10aa427088345a622a6 Mon Sep 17 00:00:00 2001 From: Jason Dobry Date: Mon, 7 Nov 2016 10:06:00 -0800 Subject: [PATCH] Fixes #48 Fixes #234 --- datastore/concepts.js | 2052 +++++++++++------------- datastore/error.js | 67 +- datastore/package.json | 12 +- datastore/system-test/concepts.test.js | 401 ++--- datastore/system-test/error.test.js | 42 +- datastore/system-test/tasks.test.js | 110 +- datastore/system-test/util.js | 41 +- datastore/tasks.js | 195 ++- datastore/test/concepts.test.js | 28 +- datastore/test/error.test.js | 28 +- datastore/test/tasks.test.js | 28 +- 11 files changed, 1380 insertions(+), 1624 deletions(-) diff --git a/datastore/concepts.js b/datastore/concepts.js index 658f8cb578..73790bcc6c 100644 --- a/datastore/concepts.js +++ b/datastore/concepts.js @@ -1,1075 +1,1034 @@ -// Copyright 2015, Google, Inc. -// Licensed under the Apache License, Version 2.0 (the "License"); -// you may not use this file except in compliance with the License. -// You may obtain a copy of the License at -// -// http://www.apache.org/licenses/LICENSE-2.0 -// -// Unless required by applicable law or agreed to in writing, software -// distributed under the License is distributed on an "AS IS" BASIS, -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -// See the License for the specific language governing permissions and -// limitations under the License. +/** + * Copyright 2016, Google, Inc. + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ 'use strict'; -var asyncUtil = require('async'); +const assert = require('power-assert'); // By default, the client will authenticate using the service account file // specified by the GOOGLE_APPLICATION_CREDENTIALS environment variable and use // the project specified by the GCLOUD_PROJECT environment variable. See // https://googlecloudplatform.github.io/gcloud-node/#/docs/google-cloud/latest/guides/authentication -var Datastore = require('@google-cloud/datastore'); - -module.exports = { - Entity: Entity, - Index: Index, - Metadata: Metadata, - Query: Query, - Transaction: Transaction -}; +const Datastore = require('@google-cloud/datastore'); // This mock is used in the documentation snippets. -var datastore = { - delete: function () {}, - get: function () {}, - insert: function () {}, - key: function () {}, - update: function () {}, - upsert: function () {}, - runQuery: function () {}, - save: function () {} +let datastore = { + delete: sinon.stub().returns(Promise.resolve([])), + get: sinon.stub().returns(Promise.resolve([])), + insert: sinon.stub().returns(Promise.resolve([])), + key: sinon.stub().returns(Promise.resolve([])), + update: sinon.stub().returns(Promise.resolve([])), + upsert: sinon.stub().returns(Promise.resolve([])), + runQuery: sinon.stub().returns(Promise.resolve([[]])), + save: sinon.stub().returns(Promise.resolve([])) }; -function Entity (projectId) { - var options = { - projectId: projectId - }; - - this.datastore = Datastore(options); +class TestHelper { + constructor (projectId) { + const options = { + projectId: projectId + }; + this.datastore = Datastore(options); + } +} - // To create the keys, we have to use this instance of Datastore. - datastore.key = this.datastore.key; +class Entity extends TestHelper { + constructor (projectId) { + super(projectId); + // To create the keys, we have to use this instance of Datastore. + datastore.key = this.datastore.key; - this.incompleteKey = this.getIncompleteKey(); - this.namedKey = this.getNamedKey(); - this.keyWithParent = this.getKeyWithParent(); - this.keyWithMultiLevelParent = this.getKeyWithMultiLevelParent(); -} + this.incompleteKey = this.getIncompleteKey(); + this.namedKey = this.getNamedKey(); + this.keyWithParent = this.getKeyWithParent(); + this.keyWithMultiLevelParent = this.getKeyWithMultiLevelParent(); + } -Entity.prototype.getIncompleteKey = function () { - // [START incomplete_key] - var taskKey = datastore.key('Task'); - // [END incomplete_key] + getIncompleteKey () { + // [START incomplete_key] + const taskKey = datastore.key('Task'); + // [END incomplete_key] - return taskKey; -}; + return taskKey; + } -Entity.prototype.getNamedKey = function () { - // [START named_key] - var taskKey = datastore.key([ - 'Task', - 'sampleTask' - ]); - // [END named_key] + getNamedKey () { + // [START named_key] + const taskKey = datastore.key([ + 'Task', + 'sampleTask' + ]); + // [END named_key] - return taskKey; -}; + return taskKey; + } -Entity.prototype.getKeyWithParent = function () { - // [START key_with_parent] - var taskKey = datastore.key([ - 'TaskList', - 'default', - 'Task', - 'sampleTask' - ]); - // [END key_with_parent] - - return taskKey; -}; + getKeyWithParent () { + // [START key_with_parent] + const taskKey = datastore.key([ + 'TaskList', + 'default', + 'Task', + 'sampleTask' + ]); + // [END key_with_parent] + + return taskKey; + } -Entity.prototype.getKeyWithMultiLevelParent = function () { - // [START key_with_multilevel_parent] - var taskKey = datastore.key([ - 'User', - 'alice', - 'TaskList', - 'default', - 'Task', - 'sampleTask' - ]); - // [END key_with_multilevel_parent] - - return taskKey; -}; + getKeyWithMultiLevelParent () { + // [START key_with_multilevel_parent] + const taskKey = datastore.key([ + 'User', + 'alice', + 'TaskList', + 'default', + 'Task', + 'sampleTask' + ]); + // [END key_with_multilevel_parent] + + return taskKey; + } -Entity.prototype.getTask = function () { - // [START basic_entity] - var task = { - category: 'Personal', - done: false, - priority: 4, - description: 'Learn Cloud Datastore' - }; - // [END basic_entity] + getTask () { + // [START basic_entity] + const task = { + category: 'Personal', + done: false, + priority: 4, + description: 'Learn Cloud Datastore' + }; + // [END basic_entity] - return task; -}; + return task; + } -Entity.prototype.testIncompleteKey = function (callback) { - this.datastore.save({ - key: this.incompleteKey, - data: {} - }, callback); -}; + testIncompleteKey () { + return this.datastore.save({ + key: this.incompleteKey, + data: {} + }); + } -Entity.prototype.testNamedKey = function (callback) { - this.datastore.save({ - key: this.namedKey, - data: {} - }, callback); -}; + testNamedKey () { + return this.datastore.save({ + key: this.namedKey, + data: {} + }); + } -Entity.prototype.testKeyWithParent = function (callback) { - this.datastore.save({ - key: this.keyWithParent, - data: {} - }, callback); -}; + testKeyWithParent () { + return this.datastore.save({ + key: this.keyWithParent, + data: {} + }); + } -Entity.prototype.testKeyWithMultiLevelParent = function (callback) { - this.datastore.save({ - key: this.keyWithMultiLevelParent, - data: {} - }, callback); -}; + testKeyWithMultiLevelParent () { + return this.datastore.save({ + key: this.keyWithMultiLevelParent, + data: {} + }); + } -Entity.prototype.testEntityWithParent = function (callback) { - var taskKey = this.keyWithParent; + testEntityWithParent () { + const taskKey = this.keyWithParent; - // [START entity_with_parent] - var task = { - key: taskKey, - data: { - category: 'Personal', - done: false, - priority: 4, - description: 'Learn Cloud Datastore' - } - }; - // [END entity_with_parent] + // [START entity_with_parent] + const task = { + key: taskKey, + data: { + category: 'Personal', + done: false, + priority: 4, + description: 'Learn Cloud Datastore' + } + }; + // [END entity_with_parent] - this.datastore.save(task, callback); -}; + return this.datastore.save(task); + } -Entity.prototype.testProperties = function (callback) { - // jshint camelcase:false - // [START properties] - var task = [ - { - name: 'category', - value: 'Personal' - }, - { - name: 'created', - value: new Date() - }, - { - name: 'done', - value: false - }, - { - name: 'priority', - value: 4 - }, - { - name: 'percent_complete', - value: 10.0 - }, - { - name: 'description', - value: 'Learn Cloud Datastore', - excludeFromIndexes: true - } - ]; - // [END properties] + testProperties () { + // [START properties] + const task = [ + { + name: 'category', + value: 'Personal' + }, + { + name: 'created', + value: new Date() + }, + { + name: 'done', + value: false + }, + { + name: 'priority', + value: 4 + }, + { + name: 'percent_complete', + value: 10.0 + }, + { + name: 'description', + value: 'Learn Cloud Datastore', + excludeFromIndexes: true + } + ]; + // [END properties] - this.datastore.save({ - key: this.incompleteKey, - data: task - }, callback); -}; + return this.datastore.save({ + key: this.incompleteKey, + data: task + }); + } -Entity.prototype.testArrayValue = function (callback) { - // [START array_value] - var task = { - tags: [ - 'fun', - 'programming' - ], - collaborators: [ - 'alice', - 'bob' - ] - }; - // [END array_value] + testArrayValue () { + // [START array_value] + const task = { + tags: [ + 'fun', + 'programming' + ], + collaborators: [ + 'alice', + 'bob' + ] + }; + // [END array_value] - this.datastore.save({ - key: this.incompleteKey, - data: task - }, callback); -}; + return this.datastore.save({ + key: this.incompleteKey, + data: task + }); + } -Entity.prototype.testBasicEntity = function (callback) { - this.datastore.save({ - key: this.getIncompleteKey(), - data: this.getTask() - }, callback); -}; + testBasicEntity () { + return this.datastore.save({ + key: this.getIncompleteKey(), + data: this.getTask() + }); + } -Entity.prototype.testUpsert = function (callback) { - var taskKey = this.getIncompleteKey(); - var task = this.getTask(); - - // [START upsert] - datastore.upsert({ - key: taskKey, - data: task - }, function (err) { - if (!err) { - // Task inserted successfully. - } - }); - // [END upsert] + testUpsert () { + const taskKey = this.getIncompleteKey(); + const task = this.getTask(); - this.datastore.upsert({ - key: this.datastore.key(['Task', 1]), - data: task - }, callback); -}; + // [START upsert] + const entity = { + key: taskKey, + data: task + }; -Entity.prototype.testInsert = function (callback) { - var taskKey = this.getIncompleteKey(); - var task = this.getTask(); - - // [START insert] - datastore.insert({ - key: taskKey, - data: task - }, function (err) { - if (!err) { - // Task inserted successfully. - } - }); - // [END insert] + datastore.upsert(entity) + .then(() => { + // Task inserted successfully. + }); + // [END upsert] - this.datastore.save({ - method: 'insert', - key: taskKey, - data: task - }, callback); -}; + return this.datastore.upsert({ + key: this.datastore.key(['Task', 1]), + data: task + }); + } -Entity.prototype.testLookup = function (callback) { - var self = this; - var taskKey = this.getIncompleteKey(); - - // jshint unused:false - // [START lookup] - datastore.get(taskKey, function (err, entity) { - if (!err) { - // Task found. - - // entity.data = { - // category: 'Personal', - // done: false, - // priority: 4, - // description: 'Learn Cloud Datastore' - // }; - } - }); - // [END lookup] - - this.datastore.save({ - method: 'insert', - key: taskKey, - data: {} - }, function (err) { - if (err) { - callback(err); - return; - } + testInsert () { + const taskKey = this.getIncompleteKey(); + const task = this.getTask(); - self.datastore.get(taskKey, callback); - }); -}; + // [START insert] + const entity = { + key: taskKey, + data: task + }; -Entity.prototype.testUpdate = function (callback) { - var self = this; - var taskKey = this.getIncompleteKey(); - var task = this.getTask(); - - // [START update] - datastore.update({ - key: taskKey, - data: task - }, function (err) { - if (!err) { - // Task updated successfully. - } - }); - // [END update] - - this.datastore.save({ - method: 'insert', - key: taskKey, - data: {} - }, function (err) { - if (err) { - callback(err); - return; - } + datastore.insert(entity) + .then(() => { + // Task inserted successfully. + }); + // [END insert] - self.datastore.update({ + return this.datastore.save({ + method: 'insert', key: taskKey, data: task - }, callback); - }); -}; + }); + } -Entity.prototype.testDelete = function (callback) { - var self = this; - var taskKey = this.getIncompleteKey(); + testLookup () { + const taskKey = this.getIncompleteKey(); + + // [START lookup] + datastore.get(taskKey) + .then((results) => { + // Task found. + const entity = results[0]; + + // entity.data = { + // category: 'Personal', + // done: false, + // priority: 4, + // description: 'Learn Cloud Datastore' + // }; + console.log(entity); + }); + // [END lookup] - // [START delete] - datastore.delete(taskKey, function (err) { - if (!err) { - // Task deleted successfully. - } - }); - // [END delete] - - this.datastore.save({ - method: 'insert', - key: taskKey, - data: {} - }, function (err) { - if (err) { - callback(err); - return; - } + return this.datastore.save({ + method: 'insert', + key: taskKey, + data: {} + }).then(() => this.datastore.get(taskKey)); + } - self.datastore.delete(taskKey, callback); - }); -}; + testUpdate () { + const taskKey = this.getIncompleteKey(); + const task = this.getTask(); -Entity.prototype.testBatchUpsert = function (callback) { - var taskKey1 = this.datastore.key(['Task', 1]); - var taskKey2 = this.datastore.key(['Task', 2]); + // [START update] + const entity = { + key: taskKey, + data: task + }; - var task1 = { - category: 'Personal', - done: false, - priority: 4, - description: 'Learn Cloud Datastore' - }; + datastore.update(entity) + .then(() => { + // Task updated successfully. + }); + // [END update] - var task2 = { - category: 'Work', - done: false, - priority: 8, - description: 'Integrate Cloud Datastore' - }; + return this.datastore.save({ + method: 'insert', + key: taskKey, + data: {} + }).then(() => this.datastore.update({ key: taskKey, data: task })); + } - // [START batch_upsert] - datastore.upsert([ - { - key: taskKey1, - data: task1 - }, - { - key: taskKey2, - data: task2 - } - ], function (err) { - if (!err) { - // Tasks inserted successfully. - } - }); - // [END batch_upsert] - - this.datastore.upsert([ - { - key: taskKey1, - data: task1 - }, - { - key: taskKey2, - data: task2 - } - ], callback); -}; + testDelete () { + const taskKey = this.getIncompleteKey(); -Entity.prototype.testBatchLookup = function (callback) { - var taskKey1 = this.datastore.key(['Task', 1]); - var taskKey2 = this.datastore.key(['Task', 2]); - - // jshint unused:false - // [START batch_lookup] - datastore.get([ - taskKey1, - taskKey2 - ], function (err, tasks) { - if (!err) { - // Tasks retrieved successfully. - } - }); - // [END batch_lookup] + // [START delete] + datastore.delete(taskKey) + .then(() => { + // Task deleted successfully. + }); + // [END delete] - this.datastore.get([ - taskKey1, - taskKey2 - ], callback); -}; + return this.datastore.save({ + method: 'insert', + key: taskKey, + data: {} + }).then(() => this.datastore.delete(taskKey)); + } -Entity.prototype.testBatchDelete = function (callback) { - var taskKey1 = this.datastore.key(['Task', 1]); - var taskKey2 = this.datastore.key(['Task', 2]); - - // [START batch_delete] - datastore.delete([ - taskKey1, - taskKey2 - ], function (err) { - if (!err) { - // Tasks deleted successfully. - } - }); - // [END batch_delete] + testBatchUpsert () { + const taskKey1 = this.datastore.key(['Task', 1]); + const taskKey2 = this.datastore.key(['Task', 2]); - this.datastore.delete([ - taskKey1, - taskKey2 - ], callback); -}; + const task1 = { + category: 'Personal', + done: false, + priority: 4, + description: 'Learn Cloud Datastore' + }; -function Index (projectId) { - var options = { - projectId: projectId - }; + const task2 = { + category: 'Work', + done: false, + priority: 8, + description: 'Integrate Cloud Datastore' + }; - this.datastore = Datastore(options); -} + // [START batch_upsert] + const entities = [ + { + key: taskKey1, + data: task1 + }, + { + key: taskKey2, + data: task2 + } + ]; -Index.prototype.testUnindexedPropertyQuery = function (callback) { - var datastore = this.datastore; + datastore.upsert(entities) + .then(() => { + // Tasks inserted successfully. + }); + // [END batch_upsert] + + return this.datastore.upsert([ + { + key: taskKey1, + data: task1 + }, + { + key: taskKey2, + data: task2 + } + ]); + } - // [START unindexed_property_query] - var query = datastore.createQuery('Task') - .filter('description', '=', 'A task description.'); - // [END unindexed_property_query] + testBatchLookup () { + const taskKey1 = this.datastore.key(['Task', 1]); + const taskKey2 = this.datastore.key(['Task', 2]); - this.datastore.runQuery(query, callback); -}; + // [START batch_lookup] + const keys = [taskKey1, taskKey2]; -Index.prototype.testExplodingProperties = function (callback) { - var original = datastore.key; - datastore.key = this.datastore.key; + datastore.get(keys) + .then((results) => { + // Tasks retrieved successfully. + const tasks = results[0]; - // [START exploding_properties] - var task = { - method: 'insert', - key: datastore.key('Task'), - data: { - tags: [ - 'fun', - 'programming', - 'learn' - ], - collaborators: [ - 'alice', - 'bob', - 'charlie' - ], - created: new Date() - } - }; - // [END exploding_properties] + console.log(tasks); + }); + // [END batch_lookup] - datastore.key = original; + return this.datastore.get([taskKey1, taskKey2]); + } - this.datastore.save(task, callback); -}; + testBatchDelete () { + const taskKey1 = this.datastore.key(['Task', 1]); + const taskKey2 = this.datastore.key(['Task', 2]); -function Metadata (projectId) { - var options = { - projectId: projectId - }; + // [START batch_delete] + const keys = [taskKey1, taskKey2]; - this.datastore = Datastore(options); -} + datastore.delete(keys) + .then(() => { + // Tasks deleted successfully. + }); + // [END batch_delete] -Metadata.prototype.testNamespaceRunQuery = function (callback) { - var self = this; + return this.datastore.delete([taskKey1, taskKey2]); + } +} - datastore.createQuery = this.datastore.createQuery; - datastore.key = this.datastore.key; +class Index extends TestHelper { + testUnindexedPropertyQuery () { + const datastore = this.datastore; - var startNamespace = 'Animals'; - var endNamespace = 'Zoos'; + // [START unindexed_property_query] + const query = datastore.createQuery('Task') + .filter('description', '=', 'A task description.'); + // [END unindexed_property_query] - this.datastore.save([ - { - key: datastore.key({ - namespace: 'Animals', - path: ['Ant', 1] - }), - data: {} - } - ], function (err) { - if (err) { - callback(err); - return; - } + return this.datastore.runQuery(query); + } - // jshint unused:false - // [START namespace_run_query] - var query = datastore.createQuery('__namespace__') - .select('__key__') - .filter('__key__', '>=', datastore.key(['__namespace__', startNamespace])) - .filter('__key__', '<', datastore.key(['__namespace__', endNamespace])); + testExplodingProperties () { + const original = datastore.key; + datastore.key = this.datastore.key; - datastore.runQuery(query, function (err, entities) { - if (err) { - // An error occurred while running the query. - return; + // [START exploding_properties] + const task = { + method: 'insert', + key: datastore.key('Task'), + data: { + tags: [ + 'fun', + 'programming', + 'learn' + ], + collaborators: [ + 'alice', + 'bob', + 'charlie' + ], + created: new Date() } + }; + // [END exploding_properties] - var namespaces = entities.map(function (entity) { - return entity.key.path.pop(); + datastore.key = original; + + return this.datastore.save(task) + .then(() => { + assert(task.key); + assert(task.key.id); }); - console.log('namespaces', namespaces); - }); - // [END namespace_run_query] + } +} - self.datastore.runQuery(query, callback); - }); -}; +class Metadata extends TestHelper { + testNamespaceRunQuery () { + const datastore = this.datastore; -Metadata.prototype.testKindRunQuery = function (callback) { - datastore.createQuery = this.datastore.createQuery; + const startNamespace = 'Animals'; + const endNamespace = 'Zoos'; - // jshint unused:false - // [START kind_run_query] - var query = datastore.createQuery('__kind__') - .select('__key__'); + return datastore.save({ + key: datastore.key({ + namespace: 'Animals', + path: ['Ant', 1] + }), + data: {} + }) + .then(() => { + // [START namespace_run_query] + function runNamespaceQuery (startNamespace, endNamespace) { + const startKey = datastore.key(['__namespace__', startNamespace]); + const endKey = datastore.key(['__namespace__', endNamespace]); + + const query = datastore.createQuery('__namespace__') + .select('__key__') + .filter('__key__', '>=', startKey) + .filter('__key__', '<', endKey); + + return datastore.runQuery(query) + .then((results) => { + const entities = results[0]; + const namespaces = entities.map((entity) => entity[datastore.KEY].name); + + console.log('Namespaces:'); + namespaces.forEach((namespace) => console.log(namespace)); + + return namespaces; + }); + } + // [END namespace_run_query] - datastore.runQuery(query, function (err, entities) { - if (err) { - // An error occurred while running the query. - return; - } + return runNamespaceQuery(startNamespace, endNamespace); + }) + .then((namespaces) => { + assert.deepEqual(namespaces, ['Animals']); + }); + } - var kinds = entities.map(function (entity) { - return entity.key.path.pop(); - }); - console.log('kinds', kinds); - }); - // [END kind_run_query] + testKindRunQuery () { + const datastore = this.datastore; - this.datastore.runQuery(query, callback); -}; + // [START kind_run_query] + function runKindQuery () { + const query = datastore.createQuery('__kind__') + .select('__key__'); -Metadata.prototype.testPropertyRunQuery = function (callback) { - datastore.createQuery = this.datastore.createQuery; + return datastore.runQuery(query) + .then((results) => { + const entities = results[0]; + const kinds = entities.map((entity) => entity[datastore.KEY].name); - // [START property_run_query] - var query = datastore.createQuery('__property__') - .select('__key__'); + console.log('Kinds:'); + kinds.forEach((kind) => console.log(kind)); - datastore.runQuery(query, function (err, entities) { - if (err) { - // An error occurred while running the query. - return; + return kinds; + }); } + // [END kind_run_query] - var propertiesByKind = {}; + return runKindQuery() + .then((kinds) => { + assert.equal(kinds.includes('Account'), true); + }); + } - entities.forEach(function (entity) { - var kind = entity.key.path[1]; - var propertyName = entity.key.path[3]; + testPropertyRunQuery () { + const datastore = this.datastore; - propertiesByKind[kind] = propertiesByKind[kind] || []; - propertiesByKind[kind].push(propertyName); - }); - }); - // [END property_run_query] + // [START property_run_query] + function runPropertyQuery () { + const query = datastore.createQuery('__property__') + .select('__key__'); - this.datastore.runQuery(query, callback); -}; + return datastore.runQuery(query) + .then((results) => { + const entities = results[0]; + const propertiesByKind = {}; -Metadata.prototype.testPropertyByKindRunQuery = function (callback) { - var datastore = this.datastore; + entities.forEach((entity) => { + const key = entity[datastore.KEY]; + const kind = key.path[1]; + const property = key.path[3]; - // jshint camelcase:false - // [START property_by_kind_run_query] - var ancestorKey = datastore.key(['__kind__', 'Task']); + propertiesByKind[kind] = propertiesByKind[kind] || []; + propertiesByKind[kind].push(property); + }); - var query = datastore.createQuery('__property__') - .hasAncestor(ancestorKey); + console.log('Properties by Kind:'); + for (let key in propertiesByKind) { + console.log(key, propertiesByKind[key]); + } - datastore.runQuery(query, function (err, entities) { - if (err) { - // An error occurred while running the query. - return; + return propertiesByKind; + }); } + // [END property_run_query] - var representationsByProperty = {}; + return runPropertyQuery() + .then((propertiesByKind) => { + assert.deepEqual(propertiesByKind.Account, ['balance']); + }); + } - entities.forEach(function (entity) { - var propertyName = entity.key.path.pop(); - var propertyType = entity.data.property_representation; + testPropertyByKindRunQuery () { + const datastore = this.datastore; - representationsByProperty[propertyName] = propertyType; - }); - }); - // [END property_by_kind_run_query] + // [START property_by_kind_run_query] + function runPropertyByKindQuery () { + const ancestorKey = datastore.key(['__kind__', 'Account']); - this.datastore.runQuery(query, callback); -}; + const query = datastore.createQuery('__property__') + .hasAncestor(ancestorKey); -function Query (projectId) { - var options = { - projectId: projectId - }; - - this.datastore = Datastore(options); + return datastore.runQuery(query) + .then((results) => { + const entities = results[0]; - this.basicQuery = this.getBasicQuery(); - this.projectionQuery = this.getProjectionQuery(); - this.ancestorQuery = this.getAncestorQuery(); -} + const representationsByProperty = {}; -Query.prototype.getBasicQuery = function () { - var datastore = this.datastore; + entities.forEach((entity) => { + const key = entity[datastore.KEY]; + const propertyName = key.name; + const propertyType = entity.property_representation; - // [START basic_query] - var query = datastore.createQuery('Task') - .filter('done', '=', false) - .filter('priority', '>=', 4) - .order('priority', { - descending: true - }); - // [END basic_query] - - return query; -}; + representationsByProperty[propertyName] = propertyType; + }); -Query.prototype.getProjectionQuery = function () { - var datastore = this.datastore; + console.log('Task property representations:'); + for (let key in representationsByProperty) { + console.log(key, representationsByProperty[key]); + } - // [START projection_query] - var query = datastore.createQuery('Task') - .select(['priority', 'percent_complete']); - // [END projection_query] + return representationsByProperty; + }); + } + // [END property_by_kind_run_query] - return query; -}; + return runPropertyByKindQuery() + .then((propertiesByKind) => { + assert.deepEqual(propertiesByKind, { + balance: ['INT64'] + }); + }); + } +} -Query.prototype.getAncestorQuery = function () { - var datastore = this.datastore; +class Query extends TestHelper { + constructor (projectId) { + super(projectId); - // [START ancestor_query] - var ancestorKey = datastore.key(['TaskList', 'default']); + this.basicQuery = this.getBasicQuery(); + this.projectionQuery = this.getProjectionQuery(); + this.ancestorQuery = this.getAncestorQuery(); + } - var query = datastore.createQuery('Task') - .hasAncestor(ancestorKey); - // [END ancestor_query] + getBasicQuery () { + const datastore = this.datastore; - return query; -}; + // [START basic_query] + const query = datastore.createQuery('Task') + .filter('done', '=', false) + .filter('priority', '>=', 4) + .order('priority', { + descending: true + }); + // [END basic_query] -Query.prototype.testRunQuery = function (callback) { - var query = this.basicQuery; + return query; + } - // jshint unused:false - // [START run_query] - datastore.runQuery(query, function (err, tasks) { - if (!err) { - // Task entities found. - } - }); - // [END run_query] + getProjectionQuery () { + const datastore = this.datastore; - this.datastore.runQuery(query, callback); -}; + // [START projection_query] + const query = datastore.createQuery('Task') + .select(['priority', 'percent_complete']); + // [END projection_query] -Query.prototype.testPropertyFilter = function (callback) { - var datastore = this.datastore; + return query; + } - // [START property_filter] - var query = datastore.createQuery('Task') - .filter('done', '=', false); - // [END property_filter] + getAncestorQuery () { + const datastore = this.datastore; - this.datastore.runQuery(query, callback); -}; + // [START ancestor_query] + const ancestorKey = datastore.key(['TaskList', 'default']); -Query.prototype.testCompositeFilter = function (callback) { - var datastore = this.datastore; + const query = datastore.createQuery('Task') + .hasAncestor(ancestorKey); + // [END ancestor_query] - // [START composite_filter] - var query = datastore.createQuery('Task') - .filter('done', '=', false) - .filter('priority', '=', 4); - // [END composite_filter] + return query; + } - this.datastore.runQuery(query, callback); -}; + testRunQuery () { + const query = this.basicQuery; -Query.prototype.testKeyFilter = function (callback) { - var datastore = this.datastore; + // [START run_query] + datastore.runQuery(query) + .then((results) => { + // Task entities found. + const tasks = results[0]; - // [START key_filter] - var query = datastore.createQuery('Task') - .filter('__key__', '>', datastore.key(['Task', 'someTask'])); - // [END key_filter] + console.log('Tasks:'); + tasks.forEach((task) => console.log(task)); + }); + // [END run_query] - this.datastore.runQuery(query, callback); -}; + return this.datastore.runQuery(query); + } -Query.prototype.testAscendingSort = function (callback) { - var datastore = this.datastore; + testPropertyFilter () { + const datastore = this.datastore; - // [START ascending_sort] - var query = datastore.createQuery('Task') - .order('created'); - // [END ascending_sort] + // [START property_filter] + const query = datastore.createQuery('Task') + .filter('done', '=', false); + // [END property_filter] - this.datastore.runQuery(query, callback); -}; + return this.datastore.runQuery(query); + } -Query.prototype.testDescendingSort = function (callback) { - var datastore = this.datastore; + testCompositeFilter () { + const datastore = this.datastore; - // [START descending_sort] - var query = datastore.createQuery('Task') - .order('created', { - descending: true - }); - // [END descending_sort] + // [START composite_filter] + const query = datastore.createQuery('Task') + .filter('done', '=', false) + .filter('priority', '=', 4); + // [END composite_filter] - this.datastore.runQuery(query, callback); -}; + return this.datastore.runQuery(query); + } -Query.prototype.testMultiSort = function (callback) { - var datastore = this.datastore; + testKeyFilter () { + const datastore = this.datastore; - // [START multi_sort] - var query = datastore.createQuery('Task') - .order('priority', { - descending: true - }) - .order('created'); - // [END multi_sort] + // [START key_filter] + const query = datastore.createQuery('Task') + .filter('__key__', '>', datastore.key(['Task', 'someTask'])); + // [END key_filter] - this.datastore.runQuery(query, callback); -}; + return this.datastore.runQuery(query); + } -Query.prototype.testKindlessQuery = function (callback) { - var datastore = this.datastore; - var lastSeenKey = this.datastore.key(['Task', Date.now()]); + testAscendingSort () { + const datastore = this.datastore; - // [START kindless_query] - var query = datastore.createQuery() - .filter('__key__', '>', lastSeenKey) - .limit(1); - // [END kindless_query] + // [START ascending_sort] + const query = datastore.createQuery('Task') + .order('created'); + // [END ascending_sort] - this.datastore.runQuery(query, callback); -}; + return this.datastore.runQuery(query); + } -Query.prototype.testRunQueryProjection = function (callback) { - var self = this; - var query = this.projectionQuery; + testDescendingSort () { + const datastore = this.datastore; - // Overwrite the mock to actually run the query. - datastore.runQuery = function (query, queryCallback) { - // Restore the mock. - datastore.runQuery = function () {}; + // [START descending_sort] + const query = datastore.createQuery('Task') + .order('created', { + descending: true + }); + // [END descending_sort] - self.datastore.runQuery(query, function (err) { - if (err) { - return callback(err); - } + return this.datastore.runQuery(query); + } - queryCallback.apply(null, arguments); + testMultiSort () { + const datastore = this.datastore; - if (priorities.length === 0 || percentCompletes.length === 0) { - callback(new Error('Projection lists did not build up.')); - } else { - callback(); - } - }); - }; + // [START multi_sort] + const query = datastore.createQuery('Task') + .order('priority', { + descending: true + }) + .order('created'); + // [END multi_sort] - // jshint unused:false, camelcase:false - // [START run_query_projection] - var priorities = []; - var percentCompletes = []; + return this.datastore.runQuery(query); + } - datastore.runQuery(query, function (err, tasks) { - if (err) { - // An error occurred while running the query. - return; - } + testKindlessQuery () { + const datastore = this.datastore; + const lastSeenKey = this.datastore.key(['Task', Date.now()]); - tasks.forEach(function (task) { - priorities.push(task.data.priority); - percentCompletes.push(task.data.percent_complete); - }); - }); - // [END run_query_projection] -}; + // [START kindless_query] + const query = datastore.createQuery() + .filter('__key__', '>', lastSeenKey) + .limit(1); + // [END kindless_query] -Query.prototype.testKeysOnlyQuery = function (callback) { - var datastore = this.datastore; + return this.datastore.runQuery(query); + } - // [START keys_only_query] - var query = datastore.createQuery() - .select('__key__') - .limit(1); - // [END keys_only_query] + testRunQueryProjection () { + const datastore = this.datastore; + const query = this.projectionQuery; - this.datastore.runQuery(query, callback); -}; + // [START run_query_projection] + function runProjectionQuery () { + const priorities = []; + const percentCompletes = []; -Query.prototype.testDistinctQuery = function (callback) { - var datastore = this.datastore; + return datastore.runQuery(query) + .then((results) => { + const tasks = results[0]; - // [START distinct_query] - var query = datastore.createQuery('Task') - .groupBy(['category', 'priority']) - .order('category') - .order('priority'); - // [END distinct_query] + tasks.forEach((task) => { + priorities.push(task.priority); + percentCompletes.push(task.percent_complete); + }); - this.datastore.runQuery(query, callback); -}; + return { + priorities: priorities, + percentCompletes: percentCompletes + }; + }); + } + // [END run_query_projection] -Query.prototype.testDistinctOnQuery = function (callback) { - var datastore = this.datastore; + return runProjectionQuery(); + } - // [START distinct_on_query] - var query = datastore.createQuery('Task') - .groupBy('category') - .order('category') - .order('priority'); - // [END distinct_on_query] + testKeysOnlyQuery () { + const datastore = this.datastore; - this.datastore.runQuery(query, callback); -}; + // [START keys_only_query] + const query = datastore.createQuery() + .select('__key__') + .limit(1); + // [END keys_only_query] -Query.prototype.testArrayValueInequalityRange = function (callback) { - var datastore = this.datastore; + return this.datastore.runQuery(query); + } - // [START array_value_inequality_range] - var query = datastore.createQuery('Task') - .filter('tag', '>', 'learn') - .filter('tag', '<', 'math'); - // [END array_value_inequality_range] + testDistinctQuery () { + const datastore = this.datastore; - this.datastore.runQuery(query, callback); -}; + // [START distinct_query] + const query = datastore.createQuery('Task') + .groupBy(['category', 'priority']) + .order('category') + .order('priority'); + // [END distinct_query] -Query.prototype.testArrayValueEquality = function (callback) { - var datastore = this.datastore; + return this.datastore.runQuery(query); + } - // [START array_value_equality] - var query = datastore.createQuery('Task') - .filter('tag', '=', 'fun') - .filter('tag', '=', 'programming'); - // [END array_value_equality] + testDistinctOnQuery () { + const datastore = this.datastore; - this.datastore.runQuery(query, callback); -}; + // [START distinct_on_query] + const query = datastore.createQuery('Task') + .groupBy('category') + .order('category') + .order('priority'); + // [END distinct_on_query] -Query.prototype.testInequalityRange = function (callback) { - var datastore = this.datastore; + return this.datastore.runQuery(query); + } - // [START inequality_range] - var query = datastore.createQuery('Task') - .filter('created', '>', new Date('1990-01-01T00:00:00z')) - .filter('created', '<', new Date('2000-12-31T23:59:59z')); - // [END inequality_range] + testArrayValueInequalityRange () { + const datastore = this.datastore; - this.datastore.runQuery(query, callback); -}; + // [START array_value_inequality_range] + const query = datastore.createQuery('Task') + .filter('tag', '>', 'learn') + .filter('tag', '<', 'math'); + // [END array_value_inequality_range] -Query.prototype.testInequalityInvalid = function (callback) { - var datastore = this.datastore; + return this.datastore.runQuery(query); + } - // [START inequality_invalid] - var query = datastore.createQuery('Task') - .filter('priority', '>', 3) - .filter('created', '>', new Date('1990-01-01T00:00:00z')); - // [END inequality_invalid] + testArrayValueEquality () { + const datastore = this.datastore; - this.datastore.runQuery(query, callback); -}; + // [START array_value_equality] + const query = datastore.createQuery('Task') + .filter('tag', '=', 'fun') + .filter('tag', '=', 'programming'); + // [END array_value_equality] -Query.prototype.testEqualAndInequalityRange = function (callback) { - var datastore = this.datastore; + return this.datastore.runQuery(query); + } - // [START equal_and_inequality_range] - var query = datastore.createQuery('Task') - .filter('priority', '=', 4) - .filter('done', '=', false) - .filter('created', '>', new Date('1990-01-01T00:00:00z')) - .filter('created', '<', new Date('2000-12-31T23:59:59z')); - // [END equal_and_inequality_range] + testInequalityRange () { + const datastore = this.datastore; - this.datastore.runQuery(query, callback); -}; + // [START inequality_range] + const query = datastore.createQuery('Task') + .filter('created', '>', new Date('1990-01-01T00:00:00z')) + .filter('created', '<', new Date('2000-12-31T23:59:59z')); + // [END inequality_range] -Query.prototype.testInequalitySort = function (callback) { - var datastore = this.datastore; + return this.datastore.runQuery(query); + } - // [START inequality_sort] - var query = datastore.createQuery('Task') - .filter('priority', '>', 3) - .order('priority') - .order('created'); - // [END inequality_sort] + testInequalityInvalid () { + const datastore = this.datastore; - this.datastore.runQuery(query, callback); -}; + // [START inequality_invalid] + const query = datastore.createQuery('Task') + .filter('priority', '>', 3) + .filter('created', '>', new Date('1990-01-01T00:00:00z')); + // [END inequality_invalid] -Query.prototype.testInequalitySortInvalidNotSame = function (callback) { - var datastore = this.datastore; + return this.datastore.runQuery(query); + } - // [START inequality_sort_invalid_not_same] - var query = datastore.createQuery('Task') - .filter('priority', '>', 3) - .order('created'); - // [END inequality_sort_invalid_not_same] + testEqualAndInequalityRange () { + const datastore = this.datastore; - this.datastore.runQuery(query, callback); -}; + // [START equal_and_inequality_range] + const query = datastore.createQuery('Task') + .filter('priority', '=', 4) + .filter('done', '=', false) + .filter('created', '>', new Date('1990-01-01T00:00:00z')) + .filter('created', '<', new Date('2000-12-31T23:59:59z')); + // [END equal_and_inequality_range] -Query.prototype.testInequalitySortInvalidNotFirst = function (callback) { - var datastore = this.datastore; + return this.datastore.runQuery(query); + } - // [START inequality_sort_invalid_not_first] - var query = datastore.createQuery('Task') - .filter('priority', '>', 3) - .order('created') - .order('priority'); - // [END inequality_sort_invalid_not_first] + testInequalitySort () { + const datastore = this.datastore; - this.datastore.runQuery(query, callback); -}; + // [START inequality_sort] + const query = datastore.createQuery('Task') + .filter('priority', '>', 3) + .order('priority') + .order('created'); + // [END inequality_sort] -Query.prototype.testLimit = function (callback) { - var datastore = this.datastore; + return this.datastore.runQuery(query); + } - // [START limit] - var query = datastore.createQuery('Task') - .limit(5); - // [END limit] + testInequalitySortInvalidNotSame () { + const datastore = this.datastore; - this.datastore.runQuery(query, callback); -}; + // [START inequality_sort_invalid_not_same] + const query = datastore.createQuery('Task') + .filter('priority', '>', 3) + .order('created'); + // [END inequality_sort_invalid_not_same] -Query.prototype.testCursorPaging = function (callback) { - var pageSize = 1; - var pageCursor = ''; + return this.datastore.runQuery(query); + } - datastore.createQuery = this.datastore.createQuery; + testInequalitySortInvalidNotFirst () { + const datastore = this.datastore; - // [START cursor_paging] - // By default, gcloud-node will automatically paginate through all of the - // results that match a query. However, this sample implements manual - // pagination using limits and cursor tokens. - var query = datastore.createQuery('Task') - .limit(pageSize) - .start(pageCursor); + // [START inequality_sort_invalid_not_first] + const query = datastore.createQuery('Task') + .filter('priority', '>', 3) + .order('created') + .order('priority'); + // [END inequality_sort_invalid_not_first] - this.datastore.runQuery(query, function (err, results, info) { - if (err) { - // An error occurred while running the query. - return; - } + return this.datastore.runQuery(query); + } - var nextPageCursor; + testLimit () { + const datastore = this.datastore; - if (info.moreResults !== Datastore.NO_MORE_RESULTS) { - // If there are more results to retrieve, the end cursor is - // automatically set on `info`. To get this value directly, access - // the `endCursor` property. - nextPageCursor = info.endCursor; - } else { - // No more results exist. - } - console.log('nextPageCursor', nextPageCursor); - }); - // [END cursor_paging] + // [START limit] + const query = datastore.createQuery('Task') + .limit(5); + // [END limit] - delete datastore.createQuery; - this.datastore.runQuery(query, function (err, results, info) { - if (err) { - callback(err); - return; - } + return this.datastore.runQuery(query); + } - if (!info || !info.endCursor) { - callback(new Error('An `info` with an `endCursor` is not present.')); - } else { - callback(); - } - }); -}; + testCursorPaging () { + const datastore = this.datastore; + const pageSize = 1; -Query.prototype.testEventualConsistentQuery = function () { - // [START eventual_consistent_query] - // Read consistency cannot be specified in gcloud-node. - // [END eventual_consistent_query] -}; + // [START cursor_paging] + // By default, google-cloud-node will automatically paginate through all of + // the results that match a query. However, this sample implements manual + // pagination using limits and cursor tokens. + function runPageQuery (pageCursor) { + let query = datastore.createQuery('Task') + .limit(pageSize); -// [START transactional_update] -function transferFunds (fromKey, toKey, amount, callback) { - var transaction = datastore.transaction(); + if (pageCursor) { + query = query.start(pageCursor); + } - transaction.run(function (err) { - if (err) { - return callback(err); - } + return datastore.runQuery(query) + .then((results) => { + const entities = results[0]; + const info = results[1]; + + if (info.moreResults !== Datastore.NO_MORE_RESULTS) { + // If there are more results to retrieve, the end cursor is + // automatically set on `info`. To get this value directly, access + // the `endCursor` property. + return runPageQuery(info.endCursor) + .then((results) => { + results[0] = entities.concat(results[0]); + return results; + }); + } - transaction.get([ - fromKey, - toKey - ], function (err, accounts) { - if (err) { - return transaction.rollback(function (_err) { - return callback(_err || err); + return [entities, info]; }); - } - - accounts[0].data.balance -= amount; - accounts[1].data.balance += amount; + } + // [END cursor_paging] + + return runPageQuery() + .then((results) => { + const entities = results[0]; + assert.equal(Array.isArray(entities), true); + const info = results[1]; + if (!info || !info.endCursor) { + throw new Error('An `info` with an `endCursor` is not present.'); + } + }); + } - transaction.save(accounts); + testEventualConsistentQuery () { + // [START eventual_consistent_query] + // Read consistency cannot be specified in google-cloud-node. + // [END eventual_consistent_query] + } +} - transaction.commit(function (err) { - if (err) { - return callback(err); +// [START transactional_update] +function transferFunds (fromKey, toKey, amount) { + const transaction = datastore.transaction(); + + return transaction.run() + .then(() => Promise.all([transaction.get(fromKey), transaction.get(toKey)])) + .then((results) => { + const accounts = results + .map((result) => result[0]); + + accounts[0].balance -= amount; + accounts[1].balance += amount; + + transaction.save([ + { + key: fromKey, + data: accounts[0] + }, + { + key: toKey, + data: accounts[1] } + ]); - // The transaction completed successfully. - callback(); - }); - }); - }); + return transaction.commit(); + }) + .catch(() => transaction.rollback()); } // [END transactional_update] @@ -1088,7 +1047,7 @@ function Transaction (projectId) { } Transaction.prototype.restoreBankAccountBalances = function (config, callback) { - var saveArray = config.keys.map(function (key) { + const entities = config.keys.map((key) => { return { key: key, data: { @@ -1097,256 +1056,191 @@ Transaction.prototype.restoreBankAccountBalances = function (config, callback) { }; }); - this.datastore.save(saveArray, callback); + if (callback) { + this.datastore.save(entities, callback); + } else { + return this.datastore.save(entities); + } }; -Transaction.prototype.testTransactionalUpdate = function (callback) { - var self = this; +Transaction.prototype.testTransactionalUpdate = function () { + const fromKey = this.fromKey; + const toKey = this.toKey; + const originalBalance = this.originalBalance; + const amountToTransfer = this.amountToTransfer; + const datastoreMock = datastore; - var fromKey = this.fromKey; - var toKey = this.toKey; - var originalBalance = this.originalBalance; - var amountToTransfer = this.amountToTransfer; + // Overwrite so the real Datastore instance is used in `transferFunds`. + datastore = this.datastore; - this.restoreBankAccountBalances({ + return this.restoreBankAccountBalances({ keys: [fromKey, toKey], balance: originalBalance - }, function (err) { - if (err) { - callback(err); - return; - } - - // Overwrite so the real Datastore instance is used in `transferFunds`. - var datastoreMock = datastore; - datastore = self.datastore; - - transferFunds(fromKey, toKey, amountToTransfer, function (err) { + }) + .then(() => transferFunds(fromKey, toKey, amountToTransfer)) + .then(() => Promise.all([this.datastore.get(fromKey), this.datastore.get(toKey)])) + .then((results) => { + const accounts = results.map((result) => result[0]); // Restore `datastore` to the mock API. datastore = datastoreMock; - - if (err) { - callback(err); - return; - } - - self.datastore.get([ - fromKey, - toKey - ], function (err, accounts) { - if (err) { - callback(err); - return; - } - - var transactionWasSuccessful = - accounts[0].data.balance === originalBalance - amountToTransfer && - accounts[1].data.balance === originalBalance + amountToTransfer; - - if (!transactionWasSuccessful) { - callback(new Error('Accounts were not updated successfully.')); - } else { - callback(); - } - }); + assert.equal(accounts[0].balance, originalBalance - amountToTransfer); + assert.equal(accounts[1].balance, originalBalance + amountToTransfer); + }) + .catch((err) => { + // Restore `datastore` to the mock API. + datastore = datastoreMock; + return Promise.reject(err); }); - }); }; -Transaction.prototype.testTransactionalRetry = function (callback) { +Transaction.prototype.testTransactionalRetry = function () { // Overwrite so the real Datastore instance is used in `transferFunds`. - var datastoreMock = datastore; + const datastoreMock = datastore; datastore = this.datastore; - var originalCallback = callback; - callback = function () { - // Restore `datastore` to the mock API. - datastore = datastoreMock; - originalCallback.apply(null, arguments); - }; - - var fromKey = this.fromKey; - var toKey = this.toKey; + const fromKey = this.fromKey; + const toKey = this.toKey; - this.restoreBankAccountBalances({ + return this.restoreBankAccountBalances({ keys: [fromKey, toKey], balance: this.originalBalance - }, function (err) { - if (err) { - callback(err); - return; - } - - // [START transactional_retry] - var async = require('async'); - - function attemptTransfer (callback) { - transferFunds(fromKey, toKey, 10, callback); - } + }) + .then(() => { + // [START transactional_retry] + function transferFundsWithRetry () { + const maxTries = 5; + let currentAttempt = 1; + let delay = 100; + + function tryRequest () { + return transferFunds(fromKey, toKey, 10) + .catch((err) => { + if (currentAttempt <= maxTries) { + // Use exponential backoff + return new Promise((resolve, reject) => { + setTimeout(() => { + currentAttempt++; + delay *= 2; + tryRequest().then(resolve, reject); + }, delay); + }); + } + return Promise.reject(err); + }); + } - async.retry(5, attemptTransfer, callback); - // [END transactional_retry] - }); + return tryRequest(1, 5); + } + // [END transactional_retry] + return transferFundsWithRetry(); + }) + .then(() => { + // Restore `datastore` to the mock API. + datastore = datastoreMock; + }) + .catch(() => { + // Restore `datastore` to the mock API. + datastore = datastoreMock; + }); }; -Transaction.prototype.testTransactionalGetOrCreate = function (callback) { - var taskKey = this.datastore.key(['Task', Date.now()]); +Transaction.prototype.testTransactionalGetOrCreate = function () { + const taskKey = this.datastore.key(['Task', Date.now()]); // Overwrite so the real Datastore instance is used in `transferFunds`. - var datastoreMock = datastore; + const datastoreMock = datastore; datastore = this.datastore; - var originalCallback = callback; - callback = function () { - // Restore `datastore` to the mock API. - datastore = datastoreMock; - originalCallback.apply(null, arguments); - }; - // [START transactional_get_or_create] - function getOrCreate (taskKey, taskData, callback) { - var taskEntity = { + function getOrCreate (taskKey, taskData) { + const taskEntity = { key: taskKey, data: taskData }; - var transaction = datastore.transaction(); - - transaction.run(function (err) { - if (err) { - return callback(err); - } - - transaction.get(taskKey, function (err, task) { - if (err) { - // An error occurred while getting the values. - return transaction.rollback(function (_err) { - return callback(_err || err); - }); - } + const transaction = datastore.transaction(); + return transaction.run() + .then(() => transaction.get(taskKey)) + .then((results) => { + const task = results[0]; if (task) { // The task entity already exists. - transaction.rollback(callback); + return transaction.rollback(); } else { // Create the task entity. transaction.save(taskEntity); - transaction.commit(function (err) { - if (err) { - return callback(err); - } - // The transaction completed successfully. - callback(null, taskEntity); - }); + return transaction.commit(); } - }); - }); + }) + .then(() => taskEntity) + .catch(() => transaction.rollback()); } // [END transactional_get_or_create] - asyncUtil.series([ - // Create: - testWithCreateBehavior, - // Then try to get it: - testWithGetBehavior - ], callback); - - function testWithCreateBehavior (callback) { - getOrCreate(taskKey, {}, function (err, task) { - if (err) { - callback(err); - return; - } - - if (!task) { - return callback(new Error('Entity was not created successfully.')); - } - callback(); - }); - } - - function testWithGetBehavior (callback) { - getOrCreate(taskKey, {}, function (err, task) { - if (err) { - callback(err); - return; - } - - if (!task) { - return callback(new Error('Entity was not retrieved successfully.')); - } - callback(); + return getOrCreate(taskKey, {}) + .then((task) => { + assert(task, 'Should have a task.'); + return getOrCreate(taskKey, {}); + }) + .then((task) => { + assert(task, 'Should have a task.'); + // Restore `datastore` to the mock API. + datastore = datastoreMock; + }) + .catch((err) => { + // Restore `datastore` to the mock API. + datastore = datastoreMock; + return Promise.reject(err); }); - } }; -Transaction.prototype.testSingleEntityGroupReadOnly = function (callback) { +Transaction.prototype.testSingleEntityGroupReadOnly = function () { // Overwrite so the real Datastore instance is used in `transferFunds`. - var datastoreMock = datastore; + const datastoreMock = datastore; datastore = this.datastore; - var originalCallback = callback; - callback = function () { - // Restore `datastore` to the mock API. - datastore = datastoreMock; - originalCallback.apply(null, arguments); - }; - // [START transactional_single_entity_group_read_only] - function getTaskListEntities (callback) { - var taskListEntities; - - var transaction = datastore.transaction(); + function getTaskListEntities () { + let taskList, taskListEntities; - transaction.run(function (err) { - if (err) { - return callback(err); - } - - var taskListKey = datastore.key(['TaskList', 'default']); + const transaction = datastore.transaction(); + const taskListKey = datastore.key(['TaskList', 'default']); - datastore.get(taskListKey, function (err) { - if (err) { - return transaction.rollback(function (_err) { - return callback(_err || err); - }); - } - - var query = datastore.createQuery('Task') + return transaction.run() + .then(() => datastore.get(taskListKey)) + .then((results) => { + taskList = results[0]; + const query = datastore.createQuery('Task') .hasAncestor(taskListKey); - - datastore.runQuery(query, function (err, entities) { - if (err) { - // An error occurred while running the query. - return transaction.rollback(function (_err) { - return callback(_err || err); - }); - } - - taskListEntities = entities; - transaction.commit(function (err) { - if (err) { - return callback(err); - } - - // The transaction completed successfully. - callback(null, taskListEntities); - }); - }); - }); - }); + return datastore.runQuery(query); + }) + .then((results) => { + taskListEntities = results[0]; + return transaction.commit(); + }) + .then(() => [taskList, taskListEntities]) + .catch(() => transaction.rollback()); } // [END transactional_single_entity_group_read_only] - getTaskListEntities(function (err, entities) { - if (err) { - callback(err); - return; - } + return getTaskListEntities() + .then((results) => { + // Restore `datastore` to the mock API. + datastore = datastoreMock; + assert.equal(results.length, 2); + assert.equal(Array.isArray(results[1]), true); + }, (err) => { + // Restore `datastore` to the mock API. + datastore = datastoreMock; + return Promise.reject(err); + }); +}; - if (!entities) { - return callback(new Error('Entities were not retrieved successfully.')); - } - callback(); - }); +module.exports = { + Entity: Entity, + Index: Index, + Metadata: Metadata, + Query: Query, + Transaction: Transaction }; diff --git a/datastore/error.js b/datastore/error.js index 4ed1c93c94..a9434e4845 100644 --- a/datastore/error.js +++ b/datastore/error.js @@ -1,34 +1,37 @@ -// Copyright 2016, Google, Inc. -// Licensed under the Apache License, Version 2.0 (the "License"); -// you may not use this file except in compliance with the License. -// You may obtain a copy of the License at -// -// http://www.apache.org/licenses/LICENSE-2.0 -// -// Unless required by applicable law or agreed to in writing, software -// distributed under the License is distributed on an "AS IS" BASIS, -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -// See the License for the specific language governing permissions and -// limitations under the License. +/** + * Copyright 2016, Google, Inc. + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ 'use strict'; -// By default, the client will authenticate using the service account file -// specified by the GOOGLE_APPLICATION_CREDENTIALS environment variable and use -// the project specified by the GCLOUD_PROJECT environment variable. See -// https://googlecloudplatform.github.io/gcloud-node/#/docs/google-cloud/latest/guides/authentication -var Datastore = require('@google-cloud/datastore'); - -// Instantiate a datastore client -var datastore = Datastore(); +const Datastore = require('@google-cloud/datastore'); // [START error] -function runQuery (cb) { - var query = datastore.createQuery(['Company']).start('badrequest'); - - datastore.runQuery(query, function (err, entities) { - // Check for an error - if (err) { +function runQuery () { + // Instantiates a client + const datastore = Datastore(); + + const query = datastore.createQuery(['Company']).start('badrequest'); + + return datastore.runQuery(query) + .then((results) => { + const entities = results[0]; + console.log('Entities:'); + entities.forEach((entity) => console.log(entity)); + return entities; + }) + .catch((err) => { console.log(err.errors); // [...] console.log(err.code); // 400 console.log(err.message); // "Bad Request" @@ -38,21 +41,17 @@ function runQuery (cb) { // For example, treat permission error like no entities were found if (err.code === 403) { - return cb(null, []); + return []; } // Forward the error to the caller - return cb(err); - } - - // We're good - return cb(null, entities); - }); + return Promise.reject(err); + }); } // [END error] exports.runQuery = runQuery; if (module === require.main) { - runQuery(console.log); + exports.runQuery(); } diff --git a/datastore/package.json b/datastore/package.json index 009de7d153..0ea0ab6297 100644 --- a/datastore/package.json +++ b/datastore/package.json @@ -5,16 +5,16 @@ "license": "Apache Version 2.0", "author": "Google Inc.", "scripts": { - "test": "mocha -R spec -t 120000 --require intelli-espower-loader ../test/_setup.js test/*.test.js", - "system-test": "mocha -R spec -t 120000 --require intelli-espower-loader ../system-test/_setup.js system-test/*.test.js" + "test": "mocha -R spec -t 1000 --require intelli-espower-loader ../test/_setup.js test/*.test.js", + "system-test": "mocha -R spec -t 10000 --require intelli-espower-loader ../system-test/_setup.js system-test/*.test.js" }, "dependencies": { - "@google-cloud/datastore": "^0.1.1", - "async": "^2.0.1", - "yargs": "^5.0.0" + "@google-cloud/datastore": "^0.5.0", + "async": "^2.1.2", + "yargs": "^6.3.0" }, "devDependencies": { - "mocha": "^3.0.2" + "mocha": "^3.1.2" }, "engines": { "node": ">=4.3.2" diff --git a/datastore/system-test/concepts.test.js b/datastore/system-test/concepts.test.js index 74bb3b143b..7c0e38dd0b 100644 --- a/datastore/system-test/concepts.test.js +++ b/datastore/system-test/concepts.test.js @@ -1,36 +1,41 @@ -// Copyright 2015, Google, Inc. -// Licensed under the Apache License, Version 2.0 (the "License"); -// you may not use this file except in compliance with the License. -// You may obtain a copy of the License at -// -// http://www.apache.org/licenses/LICENSE-2.0 -// -// Unless required by applicable law or agreed to in writing, software -// distributed under the License is distributed on an "AS IS" BASIS, -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -// See the License for the specific language governing permissions and -// limitations under the License. +/** + * Copyright 2016, Google, Inc. + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ -'use strict'; - -var testUtil = require('./util'); -var concepts = require('../concepts'); -var transaction; -var metadata; -var index; -var query; -var entity; - -var Transaction = concepts.Transaction; -var Metadata = concepts.Metadata; -var Index = concepts.Index; -var Entity = concepts.Entity; -var Query = concepts.Query; +'use strict'; -describe('datastore:concepts', function () { - before(function () { - var projectId = process.env.GCLOUD_PROJECT || 'nodejs-docs-samples'; +const assert = require(`power-assert`); +const concepts = require(`../concepts`); +const testUtil = require(`./util`); + +let transaction; +let metadata; +let index; +let query; +let entity; + +const Transaction = concepts.Transaction; +const Metadata = concepts.Metadata; +const Index = concepts.Index; +const Entity = concepts.Entity; +const Query = concepts.Query; + +describe(`datastore:concepts`, () => { + before(() => { + const projectId = process.env.GCLOUD_PROJECT; + assert(projectId, `You must set the GCLOUD_PROJECT env var!`); transaction = new Transaction(projectId); metadata = new Metadata(projectId); index = new Index(projectId); @@ -38,245 +43,107 @@ describe('datastore:concepts', function () { query = new Query(projectId); }); - after(function (done) { - var datastore = transaction.datastore; - var query = datastore.createQuery('Task'); - - testUtil.deleteEntities(datastore, query, done); - }); - - // Transactions - - it('performs a transactional update', function (done) { - transaction.testTransactionalUpdate(done); - }); - - it('performs retries if necessary', function (done) { - transaction.testTransactionalRetry(done); - }); - - it('performs a get or create', function (done) { - transaction.testTransactionalGetOrCreate(done); - }); - - it('gets a snapshot of task list entities', function (done) { - transaction.testSingleEntityGroupReadOnly(done); - }); - - // Metadata - - it('performs a namespace query', function (done) { - metadata.testNamespaceRunQuery(done); - }); - - it('performs a kind query', function (done) { - metadata.testKindRunQuery(done); - }); - - it('performs a property query', function (done) { - metadata.testPropertyRunQuery(done); - }); - - it('performs a property by kind query', function (done) { - metadata.testPropertyByKindRunQuery(done); - }); - - // Indexes - - it( - 'performs a query with a filter on an unindexed property', - function (done) { - index.testUnindexedPropertyQuery(done); - } - ); - - it('inserts arrays of data', function (done) { - index.testExplodingProperties(done); - }); - - // Queries - - it('performs a basic query', function (done) { - query.testRunQuery(done); - }); - - it('performs a query with a property filter', function (done) { - query.testPropertyFilter(done); - }); - - it('performs a query with a composite filter', function (done) { - query.testCompositeFilter(done); - }); - - it('performs a query with a key filter', function (done) { - query.testKeyFilter(done); - }); - - it('performs a query with ascending sort', function (done) { - query.testAscendingSort(done); - }); - - it('performs a query with descending sort', function (done) { - query.testDescendingSort(done); - }); - - it('performs a query with multi sort', function (done) { - query.testMultiSort(done); - }); - - it('performs a kindless query', function (done) { - query.testKindlessQuery(done); - }); - - it('performs a projection query', function (done) { - entity.testProperties(function (err, tasks) { - console.log(err, tasks); - assert.ifError(err); - setTimeout(function () { - query.testRunQueryProjection(done); - }, 1000); + after(() => { + const datastore = transaction.datastore; + const query = datastore.createQuery(`Task`).select(`__key__`); + return testUtil.deleteEntities(datastore, query); + }); + + describe(`Transactions`, () => { + it(`performs a transactional update`, () => transaction.testTransactionalUpdate()); + it(`performs retries if necessary`, () => transaction.testTransactionalRetry()); + it(`performs a get or create`, () => transaction.testTransactionalGetOrCreate()); + it(`gets a snapshot of task list entities`, () => transaction.testSingleEntityGroupReadOnly()); + }); + + describe(`Metadata`, () => { + it(`performs a namespace query`, () => metadata.testNamespaceRunQuery()); + it(`performs a kind query`, () => metadata.testKindRunQuery()); + it(`performs a property query`, () => metadata.testPropertyRunQuery()); + it(`performs a property by kind query`, () => metadata.testPropertyByKindRunQuery()); + }); + + describe(`Indexes`, () => { + it(`performs a query with a filter on an unindexed property`, () => index.testUnindexedPropertyQuery()); + it(`inserts arrays of data`, () => index.testExplodingProperties()); + }); + + describe(`Queries`, () => { + it(`performs a basic query`, () => query.testRunQuery()); + it(`performs a query with a property filter`, () => query.testPropertyFilter()); + it(`performs a query with a composite filter`, () => query.testCompositeFilter()); + it(`performs a query with a key filter`, () => query.testKeyFilter()); + it(`performs a query with ascending sort`, () => query.testAscendingSort()); + it(`performs a query with descending sort`, () => query.testDescendingSort()); + it(`performs a query with multi sort`, () => query.testMultiSort()); + it(`performs a kindless query`, () => query.testKindlessQuery()); + it('performs a projection query', () => { + return entity.testProperties() + .then(() => { + return new Promise((resolve, reject) => { + setTimeout(function () { + query.testRunQueryProjection().then(resolve, reject); + }, 1000); + }); + }) + .then((results) => { + assert.deepEqual(results, { + priorities: [4], + percentCompletes: [10] + }); + }); }); - }); - - it('performs a keys only query', function (done) { - query.testKeysOnlyQuery(done); - }); - - it('performs a distinct query', function (done) { - query.testDistinctQuery(done); - }); - - it('performs a distinct on query', function (done) { - query.testDistinctOnQuery(done); - }); - - it('performs an array value inequality query', function (done) { - query.testArrayValueInequalityRange(done); - }); - - it('performs an array value equality query', function (done) { - query.testArrayValueEquality(done); - }); - - it('performs an inequality range query', function (done) { - query.testInequalityRange(done); - }); - - it('returns an error from an invalid query', function (done) { - query.testInequalityInvalid(function (err) { - assert(err); - done(); + it(`performs a keys only query`, () => query.testKeysOnlyQuery()); + it(`performs a distinct query`, () => query.testDistinctQuery()); + it(`performs a distinct on query`, () => query.testDistinctOnQuery()); + it(`performs an array value inequality query`, () => query.testArrayValueInequalityRange()); + it(`performs an array value equality query`, () => query.testArrayValueEquality()); + it(`performs an inequality range query`, () => query.testInequalityRange()); + it(`returns an error from an invalid query`, () => { + return query.testInequalityInvalid() + .then(() => assert.fail(), (err) => assert(err)); }); - }); - - it('performs an equal and inequality range query', function (done) { - query.testEqualAndInequalityRange(done); - }); - - it('performs an equality sort query', function (done) { - query.testInequalitySort(done); - }); - - it( - 'returns an error when not sorted on filtered property', - function (done) { - query.testInequalitySortInvalidNotSame(function (err) { - assert(err); - done(); - }); - } - ); - - it( - 'returns an error when not sorted on first filter prop', - function (done) { - query.testInequalitySortInvalidNotFirst(function (err) { - assert(err); - done(); - }); - } - ); - - it('performs a query with a limit', function (done) { - query.testLimit(done); - }); - - it('allows manual pagination through results', function (done) { - entity.testBatchUpsert(function (err) { - assert.ifError(err); - setTimeout(function () { - query.testCursorPaging(done); - }, 1000); + it(`performs an equal and inequality range query`, () => query.testEqualAndInequalityRange()); + it(`performs an equality sort query`, () => query.testInequalitySort()); + it(`returns an error when not sorted on filtered property`, () => { + return query.testInequalitySortInvalidNotSame() + .then(() => assert.fail(), (err) => assert(err)); }); - }); - - it.skip('performs an ancestor query', function (done) { - query.testEventualConsistentQuery(done); - }); - - // Entities - - it('saves with an incomplete key', function (done) { - entity.testIncompleteKey(done); - }); - - it('saves with a named key', function (done) { - entity.testNamedKey(done); - }); - - it('saves a key with a parent', function (done) { - entity.testKeyWithParent(done); - }); - - it('saves a key with multiple parents', function (done) { - entity.testKeyWithMultiLevelParent(done); - }); - - it('saves an entity with a parent', function (done) { - entity.testEntityWithParent(done); - }); - - it('saves an entity with properties', function (done) { - entity.testProperties(done); - }); - - it('saves an entity with arrays', function (done) { - entity.testArrayValue(done); - }); - - it('saves a basic entity', function (done) { - entity.testBasicEntity(done); - }); - - it('saves with an upsert', function (done) { - entity.testUpsert(done); - }); - - it('saves with an insert', function (done) { - entity.testInsert(done); - }); - - it('performs a lookup', function (done) { - entity.testLookup(done); - }); - - it('saves with an update', function (done) { - entity.testUpdate(done); - }); - - it('deletes an entity', function (done) { - entity.testDelete(done); - }); - - it('performs a batch upsert', function (done) { - entity.testBatchUpsert(done); - }); - - it('performs a batch lookup', function (done) { - entity.testBatchLookup(done); - }); - - it('performs a batch delete', function (done) { - entity.testBatchDelete(done); + it(`returns an error when not sorted on first filter prop`, () => { + return query.testInequalitySortInvalidNotFirst() + .then(() => assert.fail(), (err) => assert(err)); + }); + it(`performs a query with a limit`, () => query.testLimit()); + it(`allows manual pagination through results`, () => { + return entity.testBatchUpsert() + .then(() => { + return new Promise((resolve, reject) => { + setTimeout(() => { + query.testCursorPaging() + .then(resolve, reject); + }, 1000); + }); + }); + }); + it(`performs an ancestor query`, () => query.testEventualConsistentQuery()); + }); + + describe(`Entities`, () => { + it(`saves with an incomplete key`, () => entity.testIncompleteKey()); + it(`saves with a named key`, () => entity.testNamedKey()); + it(`saves a key with a parent`, () => entity.testKeyWithParent()); + it(`saves a key with multiple parents`, () => entity.testKeyWithMultiLevelParent()); + it(`saves an entity with a parent`, () => entity.testEntityWithParent()); + it(`saves an entity with properties`, () => entity.testProperties()); + it(`saves an entity with arrays`, () => entity.testArrayValue()); + it(`saves a basic entity`, () => entity.testBasicEntity()); + it(`saves with an upsert`, () => entity.testUpsert()); + it(`saves with an insert`, () => entity.testInsert()); + it(`performs a lookup`, () => entity.testLookup()); + it(`saves with an update`, () => entity.testUpdate()); + it(`deletes an entity`, () => entity.testDelete()); + it(`performs a batch upsert`, () => entity.testBatchUpsert()); + it(`performs a batch lookup`, () => entity.testBatchLookup()); + it(`performs a batch delete`, () => entity.testBatchDelete()); }); }); diff --git a/datastore/system-test/error.test.js b/datastore/system-test/error.test.js index ef304b6086..007ef8bbcd 100644 --- a/datastore/system-test/error.test.js +++ b/datastore/system-test/error.test.js @@ -1,26 +1,28 @@ -// Copyright 2015-2016, Google, Inc. -// Licensed under the Apache License, Version 2.0 (the "License"); -// you may not use this file except in compliance with the License. -// You may obtain a copy of the License at -// -// http://www.apache.org/licenses/LICENSE-2.0 -// -// Unless required by applicable law or agreed to in writing, software -// distributed under the License is distributed on an "AS IS" BASIS, -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -// See the License for the specific language governing permissions and -// limitations under the License. +/** + * Copyright 2016, Google, Inc. + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ 'use strict'; -var error = require('../error'); +const error = require('../error'); -describe('datastore:error', function () { - it('should have an error', function (done) { - error.runQuery(function (err) { - assert(err); - assert.equal(err.code, 400); - done(); - }); +describe(`datastore:error`, () => { + it(`should have an error`, () => { + return error.runQuery() + .catch((err) => { + assert(err); + assert.equal(err.code, 400); + }); }); }); diff --git a/datastore/system-test/tasks.test.js b/datastore/system-test/tasks.test.js index 3d7e94e2a4..f3069d64dd 100644 --- a/datastore/system-test/tasks.test.js +++ b/datastore/system-test/tasks.test.js @@ -1,62 +1,72 @@ -// Copyright 2015-2016, Google, Inc. -// Licensed under the Apache License, Version 2.0 (the "License"); -// you may not use this file except in compliance with the License. -// You may obtain a copy of the License at -// -// http://www.apache.org/licenses/LICENSE-2.0 -// -// Unless required by applicable law or agreed to in writing, software -// distributed under the License is distributed on an "AS IS" BASIS, -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -// See the License for the specific language governing permissions and -// limitations under the License. +/** + * Copyright 2016, Google, Inc. + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ 'use strict'; -var async = require('async'); -var tasks = require('../tasks'); -var taskIds = []; - -describe('datastore:tasks', function () { - after(function (done) { - async.parallel(taskIds.map(function (taskId) { - return function (cb) { - tasks.deleteEntity(taskId, cb); - }; - }), done); - }); +const datastore = require(`@google-cloud/datastore`)(); +const tasks = require(`../tasks`); - it('should add a task', function (done) { - getTaskId(done); - }); +describe(`datastore:tasks`, () => { + const description = `description`; + let key; - it('should mark a task as done', function (done) { - getTaskId(function (err, taskId) { - assert.ifError(err); - tasks.updateEntity(taskId, done); - }); + after(() => datastore.delete(key)); + + it(`should add a task`, () => { + return tasks.addTask(description) + .then((taskKey) => { + key = taskKey; + return datastore.get(key); + }) + .then((results) => { + const task = results[0]; + const taskKey = task[datastore.KEY]; + assert.equal(taskKey.id, key.id); + assert.equal(task.description, description); + }); }); - it('should list tasks', function (done) { - tasks.retrieveEntities(done); + it(`should mark a task as done`, () => { + return tasks.markDone(key.id) + .then(() => datastore.get(key)) + .then((results) => { + const task = results[0]; + const taskKey = task[datastore.KEY]; + assert.equal(taskKey.id, key.id); + assert.equal(task.description, description); + assert.equal(task.done, true); + }); }); - it('should delete a task', function (done) { - getTaskId(function (err, taskId) { - assert.ifError(err); - tasks.deleteEntity(taskId, done); - }); + it(`should list tasks`, () => { + return new Promise((resolve, reject) => { + setTimeout(() => { + tasks.listTasks().then(resolve, reject); + }, 5000); + }) + .then((tasks) => { + tasks = tasks.filter((task) => task[datastore.KEY].id === key.id); + assert.equal(tasks.length, 1); + }); }); - function getTaskId (callback) { - tasks.addEntity('description', function (err, taskKey) { - if (err) { - return callback(err); - } - - var taskId = taskKey.path.pop(); - taskIds.push(taskId); - callback(null, taskId); - }); - } + it(`should delete a task`, () => { + return tasks.deleteTask(key.id) + .then(() => datastore.get(key)) + .then((results) => { + assert.equal(results[0], undefined); + }); + }); }); diff --git a/datastore/system-test/util.js b/datastore/system-test/util.js index 1c47393b60..f6df7b600d 100644 --- a/datastore/system-test/util.js +++ b/datastore/system-test/util.js @@ -1,30 +1,23 @@ -// Copyright 2015, Google, Inc. -// Licensed under the Apache License, Version 2.0 (the "License"); -// you may not use this file except in compliance with the License. -// You may obtain a copy of the License at -// -// http://www.apache.org/licenses/LICENSE-2.0 -// -// Unless required by applicable law or agreed to in writing, software -// distributed under the License is distributed on an "AS IS" BASIS, -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -// See the License for the specific language governing permissions and -// limitations under the License. +/** + * Copyright 2016, Google, Inc. + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ 'use strict'; module.exports = { - deleteEntities: function (datastore, query, callback) { - datastore.runQuery(query, function (err, entities) { - if (err) { - return callback(err); - } - - var keys = entities.map(function (entity) { - return entity.key; - }); - - datastore.delete(keys, callback); - }); + deleteEntities: (datastore, query) => { + return datastore.runQuery(query) + .then((results) => datastore.delete(results[0].map((entity) => entity[datastore.KEY]))); } }; diff --git a/datastore/tasks.js b/datastore/tasks.js index 7d8230778f..d386b7e00e 100755 --- a/datastore/tasks.js +++ b/datastore/tasks.js @@ -1,15 +1,17 @@ -// Copyright 2015-2016, Google, Inc. -// Licensed under the Apache License, Version 2.0 (the "License"); -// you may not use this file except in compliance with the License. -// You may obtain a copy of the License at -// -// http://www.apache.org/licenses/LICENSE-2.0 -// -// Unless required by applicable law or agreed to in writing, software -// distributed under the License is distributed on an "AS IS" BASIS, -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -// See the License for the specific language governing permissions and -// limitations under the License. +/** + * Copyright 2016, Google, Inc. + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ 'use strict'; @@ -17,11 +19,11 @@ // By default, the client will authenticate using the service account file // specified by the GOOGLE_APPLICATION_CREDENTIALS environment variable and use // the project specified by the GCLOUD_PROJECT environment variable. See -// https://googlecloudplatform.github.io/gcloud-node/#/docs/google-cloud/latest/guides/authentication -var Datastore = require('@google-cloud/datastore'); +// https://googlecloudplatform.github.io/google-cloud-node/#/docs/datastore/latest/guides/authentication +const Datastore = require('@google-cloud/datastore'); -// Instantiate a datastore client -var datastore = Datastore(); +// Instantiates a client +const datastore = Datastore(); // [END build_service] /* @@ -46,7 +48,7 @@ npm install 5. With the gcloud SDK, be sure you are authenticated: ```sh -gcloud auth login +gcloud beta auth application-default login ``` 6. At a command prompt, run the following, where `` is the ID of @@ -62,10 +64,9 @@ node tasks */ // [START add_entity] -function addTask (description, callback) { - var taskKey = datastore.key('Task'); - - datastore.save({ +function addTask (description) { + const taskKey = datastore.key('Task'); + const entity = { key: taskKey, data: [ { @@ -82,102 +83,86 @@ function addTask (description, callback) { value: false } ] - }, function (err) { - if (err) { - return callback(err); - } - - var taskId = taskKey.path.pop(); - console.log('Task %d created successfully.', taskId); - return callback(null, taskKey); - }); + }; + + return datastore.save(entity) + .then(() => { + console.log(`Task ${taskKey.id} created successfully.`); + return taskKey; + }); } // [END add_entity] // [START update_entity] -function markDone (taskId, callback) { - var transaction = datastore.transaction(); - - transaction.run(function (err) { - if (err) { - return callback(err); - } - - var taskKey = datastore.key([ - 'Task', - taskId - ]); - - transaction.get(taskKey, function (err, task) { - if (err) { - // An error occurred while getting the task - return transaction.rollback(function (_err) { - return callback(err || _err); - }); - } - - task.data.done = true; - - transaction.save(task); - - // Commit the transaction - transaction.commit(function (err) { - if (err) { - return callback(err); - } +function markDone (taskId) { + const transaction = datastore.transaction(); + const taskKey = datastore.key([ + 'Task', + taskId + ]); - // The transaction completed successfully. - console.log('Task %d updated successfully.', taskId); - return callback(null); + return transaction.run() + .then(() => transaction.get(taskKey)) + .then((results) => { + const task = results[0]; + task.done = true; + transaction.save({ + key: taskKey, + data: task }); - }); - }); + return transaction.commit(); + }) + .then(() => { + // The transaction completed successfully. + console.log(`Task ${taskId} updated successfully.`); + }) + .catch(() => transaction.rollback()); } // [END update_entity] // [START retrieve_entities] -function listTasks (callback) { - var query = datastore.createQuery('Task') +function listTasks () { + const query = datastore.createQuery('Task') .order('created'); - datastore.runQuery(query, function (err, tasks) { - if (err) { - return callback(err); - } + return datastore.runQuery(query) + .then((results) => { + const tasks = results[0]; + + console.log('Tasks:'); + tasks.forEach((task) => { + const taskKey = task[datastore.KEY]; + console.log(taskKey.id, task); + }); - console.log('Found %d task(s)!', tasks.length); - return callback(null, tasks); - }); + return tasks; + }); } // [END retrieve_entities] // [START delete_entity] -function deleteTask (taskId, callback) { - var taskKey = datastore.key([ +function deleteTask (taskId) { + const taskKey = datastore.key([ 'Task', taskId ]); - datastore.delete(taskKey, function (err) { - if (err) { - return callback(err); - } - - console.log('Task %d deleted successfully.', taskId); - return callback(null); - }); + return datastore.delete(taskKey) + .then(() => { + console.log(`Task ${taskId} deleted successfully.`); + }); } // [END delete_entity] -var cli = require('yargs'); -var makeHandler = require('../utils').makeHandler; +const cli = require(`yargs`); +const noop = require(`../utils`).noop; -var program = module.exports = { - addEntity: addTask, - updateEntity: markDone, - retrieveEntities: listTasks, - deleteEntity: deleteTask, - main: function (args) { +const program = module.exports = { + addTask: addTask, + markDone: markDone, + listTasks: listTasks, + deleteTask: deleteTask, + main: (args) => { // Run the command-line program cli.help().strict().parse(args).argv; } @@ -185,25 +170,25 @@ var program = module.exports = { cli .demand(1) - .command('new ', 'Adds a task with a description .', {}, function (options) { - addTask(options.description, makeHandler()); + .command(`new `, `Adds a task with a description .`, {}, (opts) => { + addTask(opts.description, noop); }) - .command('done ', 'Marks the specified task as done.', {}, function (options) { - markDone(options.taskId, makeHandler()); + .command(`done `, `Marks the specified task as done.`, {}, (opts) => { + markDone(opts.taskId, noop); }) - .command('list', 'Lists all tasks ordered by creation time.', {}, function (options) { - listTasks(makeHandler()); + .command(`list`, `Lists all tasks ordered by creation time.`, {}, (opts) => { + listTasks(noop); }) - .command('delete ', 'Deletes a task.', {}, function (options) { - deleteTask(options.taskId, makeHandler()); + .command(`delete `, `Deletes a task.`, {}, (opts) => { + deleteTask(opts.taskId, noop); }) - .example('node $0 new "Buy milk"', 'Adds a task with description "Buy milk".') - .example('node $0 done 12345', 'Marks task 12345 as Done.') - .example('node $0 list', 'Lists all tasks ordered by creation time') - .example('node $0 delete 12345', 'Deletes task 12345.') + .example(`node $0 new "Buy milk"`, `Adds a task with description "Buy milk".`) + .example(`node $0 done 12345`, `Marks task 12345 as Done.`) + .example(`node $0 list`, `Lists all tasks ordered by creation time`) + .example(`node $0 delete 12345`, `Deletes task 12345.`) .wrap(120) .recommendCommands() - .epilogue('For more information, see https://cloud.google.com/datastore/docs'); + .epilogue(`For more information, see https://cloud.google.com/datastore/docs`); if (module === require.main) { program.main(process.argv.slice(2)); diff --git a/datastore/test/concepts.test.js b/datastore/test/concepts.test.js index 8deb451a29..61bf592685 100644 --- a/datastore/test/concepts.test.js +++ b/datastore/test/concepts.test.js @@ -1,18 +1,20 @@ -// Copyright 2016, Google, Inc. -// Licensed under the Apache License, Version 2.0 (the "License"); -// you may not use this file except in compliance with the License. -// You may obtain a copy of the License at -// -// http://www.apache.org/licenses/LICENSE-2.0 -// -// Unless required by applicable law or agreed to in writing, software -// distributed under the License is distributed on an "AS IS" BASIS, -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -// See the License for the specific language governing permissions and -// limitations under the License. +/** + * Copyright 2016, Google, Inc. + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ 'use strict'; -describe('datastore:concepts', function () { +describe('datastore:concepts', () => { it('should be tested'); }); diff --git a/datastore/test/error.test.js b/datastore/test/error.test.js index 6396d45d5f..8b5fcd95aa 100644 --- a/datastore/test/error.test.js +++ b/datastore/test/error.test.js @@ -1,18 +1,20 @@ -// Copyright 2016, Google, Inc. -// Licensed under the Apache License, Version 2.0 (the "License"); -// you may not use this file except in compliance with the License. -// You may obtain a copy of the License at -// -// http://www.apache.org/licenses/LICENSE-2.0 -// -// Unless required by applicable law or agreed to in writing, software -// distributed under the License is distributed on an "AS IS" BASIS, -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -// See the License for the specific language governing permissions and -// limitations under the License. +/** + * Copyright 2016, Google, Inc. + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ 'use strict'; -describe('datastore:error', function () { +describe('datastore:error', () => { it('should be tested'); }); diff --git a/datastore/test/tasks.test.js b/datastore/test/tasks.test.js index 2353d821f9..ec43d10105 100644 --- a/datastore/test/tasks.test.js +++ b/datastore/test/tasks.test.js @@ -1,18 +1,20 @@ -// Copyright 2016, Google, Inc. -// Licensed under the Apache License, Version 2.0 (the "License"); -// you may not use this file except in compliance with the License. -// You may obtain a copy of the License at -// -// http://www.apache.org/licenses/LICENSE-2.0 -// -// Unless required by applicable law or agreed to in writing, software -// distributed under the License is distributed on an "AS IS" BASIS, -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -// See the License for the specific language governing permissions and -// limitations under the License. +/** + * Copyright 2016, Google, Inc. + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ 'use strict'; -describe('datastore:tasks', function () { +describe('datastore:tasks', () => { it('should be tested'); });