From 2a673986d4ac411477eae5fbae39d21fd9cd1376 Mon Sep 17 00:00:00 2001 From: Stephen Sawchuk Date: Mon, 28 Nov 2016 12:58:34 -0500 Subject: [PATCH] pubsub: support generated subscription names (#1799) --- README.md | 6 +- packages/pubsub/README.md | 6 +- packages/pubsub/package.json | 2 +- packages/pubsub/src/index.js | 52 ++++++++++------ packages/pubsub/src/subscription.js | 14 ++++- packages/pubsub/src/topic.js | 17 ++++-- packages/pubsub/system-test/pubsub.js | 9 ++- packages/pubsub/test/index.js | 85 +++++++++++++++++---------- packages/pubsub/test/subscription.js | 59 +++++++++++++++++-- 9 files changed, 177 insertions(+), 73 deletions(-) diff --git a/README.md b/README.md index a7fb2f2b8a8..6fc81d93090 100644 --- a/README.md +++ b/README.md @@ -432,11 +432,7 @@ var topic = pubsubClient.topic('my-topic'); topic.publish('New message!', function(err) {}); // Subscribe to the topic. -var options = { - reuseExisting: true -}; - -topic.subscribe('subscription-name', options, function(err, subscription) { +topic.subscribe('subscription-name', function(err, subscription) { // Register listeners to start pulling for messages. function onError(err) {} function onMessage(message) {} diff --git a/packages/pubsub/README.md b/packages/pubsub/README.md index f61b991eb52..877bb0846ae 100644 --- a/packages/pubsub/README.md +++ b/packages/pubsub/README.md @@ -23,11 +23,7 @@ var topic = pubsub.topic('my-topic'); topic.publish('New message!', function(err) {}); // Subscribe to the topic. -var options = { - reuseExisting: true -}; - -topic.subscribe('subscription-name', options, function(err, subscription) { +topic.subscribe('subscription-name', function(err, subscription) { // Register listeners to start pulling for messages. function onError(err) {} function onMessage(message) {} diff --git a/packages/pubsub/package.json b/packages/pubsub/package.json index 46beec11475..ff44918d4ed 100644 --- a/packages/pubsub/package.json +++ b/packages/pubsub/package.json @@ -57,12 +57,12 @@ "google-proto-files": "^0.8.0", "is": "^3.0.1", "modelo": "^4.2.0", + "node-uuid": "^1.4.3", "propprop": "^0.3.0" }, "devDependencies": { "async": "^1.5.2", "mocha": "^3.0.1", - "node-uuid": "^1.4.3", "proxyquire": "^1.7.10" }, "scripts": { diff --git a/packages/pubsub/src/index.js b/packages/pubsub/src/index.js index c45ee1357df..ca4c916cca8 100644 --- a/packages/pubsub/src/index.js +++ b/packages/pubsub/src/index.js @@ -409,6 +409,8 @@ PubSub.prototype.getTopicsStream = common.paginator.streamify('getTopics'); /** * Create a subscription to a topic. * + * All generated subscription names share a common prefix, `autogenerated-`. + * * @resource [Subscriptions: create API Documentation]{@link https://cloud.google.com/pubsub/docs/reference/rest/v1/projects.subscriptions/create} * * @throws {Error} If a Topic instance or topic name is not provided. @@ -416,7 +418,8 @@ PubSub.prototype.getTopicsStream = common.paginator.streamify('getTopics'); * * @param {module:pubsub/topic|string} topic - The Topic to create a * subscription to. - * @param {string} subName - The name of the subscription. + * @param {string=} subName - The name of the subscription. If a name is not + * provided, a random subscription name will be generated and created. * @param {object=} options - See a * [Subscription resource](https://cloud.google.com/pubsub/docs/reference/rest/v1/projects.subscriptions) * @param {number} options.ackDeadlineSeconds - The maximum time after receiving @@ -431,10 +434,6 @@ PubSub.prototype.getTopicsStream = common.paginator.streamify('getTopics'); * simultaneously. * @param {string} options.pushEndpoint - A URL to a custom endpoint that * messages should be pushed to. - * @param {boolean} options.reuseExisting - If the subscription already exists, - * reuse it. The options of the existing subscription are not changed. If - * false, attempting to create a subscription that already exists will fail. - * (default: false) * @param {number} options.timeout - Set a maximum amount of time in * milliseconds on an HTTP request to pull new messages to wait for a * response before the connection is broken. @@ -453,6 +452,14 @@ PubSub.prototype.getTopicsStream = common.paginator.streamify('getTopics'); * pubsub.subscribe(topic, name, function(err, subscription, apiResponse) {}); * * //- + * // Omit the name to have one generated automatically. All generated names + * // share a common prefix, `autogenerated-`. + * //- + * pubsub.subscribe(topic, function(err, subscription, apiResponse) { + * // subscription.name = The generated name. + * }); + * + * //- * // Customize the subscription. * //- * pubsub.subscribe(topic, name, { @@ -474,21 +481,28 @@ PubSub.prototype.subscribe = function(topic, subName, options, callback) { throw new Error('A Topic is required for a new subscription.'); } - if (!is.string(subName)) { - throw new Error('A subscription name is required for a new subscription.'); + if (is.string(topic)) { + topic = this.topic(topic); } - if (!callback) { + if (is.object(subName)) { callback = options; - options = {}; + options = subName; + subName = ''; } - options = options || {}; + if (is.fn(subName)) { + callback = subName; + subName = ''; + } - if (is.string(topic)) { - topic = this.topic(topic); + if (is.fn(options)) { + callback = options; + options = {}; } + options = options || {}; + var subscription = this.subscription(subName, options); var protoOpts = { @@ -513,11 +527,10 @@ PubSub.prototype.subscribe = function(topic, subName, options, callback) { delete reqOpts.interval; delete reqOpts.maxInProgress; delete reqOpts.pushEndpoint; - delete reqOpts.reuseExisting; delete reqOpts.timeout; this.request(protoOpts, reqOpts, function(err, resp) { - if (err && !(err.code === 409 && options.reuseExisting)) { + if (err && err.code !== 409) { callback(err, null, resp); return; } @@ -531,9 +544,10 @@ PubSub.prototype.subscribe = function(topic, subName, options, callback) { * requests. You will receive a {module:pubsub/subscription} object, * which will allow you to interact with a subscription. * - * @throws {Error} If a name is not provided. + * All generated names share a common prefix, `autogenerated-`. * - * @param {string} name - The name of the subscription. + * @param {string=} name - The name of the subscription. If a name is not + * provided, a random subscription name will be generated. * @param {object=} options - Configuration object. * @param {boolean} options.autoAck - Automatically acknowledge the message once * it's pulled. (default: false) @@ -560,12 +574,14 @@ PubSub.prototype.subscribe = function(topic, subName, options, callback) { * }); */ PubSub.prototype.subscription = function(name, options) { - if (!name) { - throw new Error('The name of a subscription is required.'); + if (is.object(name)) { + options = name; + name = undefined; } options = options || {}; options.name = name; + return new Subscription(this, options); }; diff --git a/packages/pubsub/src/subscription.js b/packages/pubsub/src/subscription.js index a5fb904762c..101570e7dda 100644 --- a/packages/pubsub/src/subscription.js +++ b/packages/pubsub/src/subscription.js @@ -26,6 +26,7 @@ var events = require('events'); var is = require('is'); var modelo = require('modelo'); var prop = require('propprop'); +var uuid = require('node-uuid'); /** * @type {module:pubsub/iam} @@ -140,7 +141,9 @@ var PUBSUB_API_TIMEOUT = 90000; * subscription.removeListener('message', onMessage); */ function Subscription(pubsub, options) { - this.name = Subscription.formatName_(pubsub.projectId, options.name); + var name = options.name || Subscription.generateName_(); + + this.name = Subscription.formatName_(pubsub.projectId, name); var methods = { /** @@ -374,6 +377,15 @@ Subscription.formatName_ = function(projectId, name) { return 'projects/' + projectId + '/subscriptions/' + name; }; +/** + * Generate a random name to use for a name-less subscription. + * + * @private + */ +Subscription.generateName_ = function() { + return 'autogenerated-' + uuid.v4(); +}; + /** * Acknowledge to the backend that the message was retrieved. You must provide * either a single ackId or an array of ackIds. diff --git a/packages/pubsub/src/topic.js b/packages/pubsub/src/topic.js index 466aaedacea..8a3b3e0f688 100644 --- a/packages/pubsub/src/topic.js +++ b/packages/pubsub/src/topic.js @@ -477,9 +477,12 @@ Topic.prototype.publish = function(messages, options, callback) { /** * Create a subscription to this topic. * + * All generated subscription names share a common prefix, `autogenerated-`. + * * @resource [Subscriptions: create API Documentation]{@link https://cloud.google.com/pubsub/docs/reference/rest/v1/projects.subscriptions/create} * - * @param {string} subName - The name of the subscription. + * @param {string=} subName - The name of the subscription. If a name is not + * provided, a random subscription name will be generated and created. * @param {object=} options - See a * [Subscription resource](https://cloud.google.com/pubsub/docs/reference/rest/v1/projects.subscriptions) * @param {number} options.ackDeadlineSeconds - The maximum time after @@ -494,10 +497,6 @@ Topic.prototype.publish = function(messages, options, callback) { * simultaneously. * @param {string} options.pushEndpoint - A URL to a custom endpoint that * messages should be pushed to. - * @param {boolean=} options.reuseExisting - If the subscription already exists, - * reuse it. The options of the existing subscription are not changed. If - * false, attempting to create a subscription that already exists will fail. - * (default: false) * @param {number} options.timeout - Set a maximum amount of time in * milliseconds on an HTTP request to pull new messages to wait for a * response before the connection is broken. @@ -507,6 +506,14 @@ Topic.prototype.publish = function(messages, options, callback) { * // Without specifying any options. * topic.subscribe('newMessages', function(err, subscription, apiResponse) {}); * + * //- + * // Omit the name to have one generated automatically. All generated names + * // share a common prefix, `autogenerated-`. + * //- + * topic.subscribe(function(err, subscription, apiResponse) { + * // subscription.name = The generated name. + * }); + * * // With options. * topic.subscribe('newMessages', { * ackDeadlineSeconds: 90, diff --git a/packages/pubsub/system-test/pubsub.js b/packages/pubsub/system-test/pubsub.js index 2679591acff..56b28f129c9 100644 --- a/packages/pubsub/system-test/pubsub.js +++ b/packages/pubsub/system-test/pubsub.js @@ -302,8 +302,15 @@ describe('pubsub', function() { }); }); + it('should create a subscription with a generated name', function(done) { + topic.subscribe(function(err, sub) { + assert.ifError(err); + sub.delete(done); + }); + }); + it('should re-use an existing subscription', function(done) { - pubsub.subscribe(topic, SUB_NAMES[0], { reuseExisting: true }, done); + pubsub.subscribe(topic, SUB_NAMES[0], done); }); it('should error when using a non-existent subscription', function(done) { diff --git a/packages/pubsub/test/index.js b/packages/pubsub/test/index.js index 9d81a9cc152..df69e1d3cff 100644 --- a/packages/pubsub/test/index.js +++ b/packages/pubsub/test/index.js @@ -491,10 +491,22 @@ describe('PubSub', function() { }, /A Topic is required for a new subscription\./); }); - it('should throw if no sub name is provided', function() { - assert.throws(function() { - pubsub.subscribe('topic'); - }, /A subscription name is required for a new subscription\./); + it('should not require a subscription name', function(done) { + pubsub.request = function(protoOpts, reqOpts, callback) { + callback(null, apiResponse); + }; + + pubsub.subscribe(TOPIC_NAME, done); + }); + + it('should not require a sub name and accept options', function(done) { + var opts = {}; + + pubsub.request = function(protoOpts, reqOpts, callback) { + callback(null, apiResponse); + }; + + pubsub.subscribe(TOPIC_NAME, opts, done); }); it('should not require configuration options', function(done) { @@ -573,7 +585,6 @@ describe('PubSub', function() { interval: 3, maxInProgress: 5, pushEndpoint: 'https://domain/push', - reuseExisting: false, timeout: 30000 }; @@ -591,7 +602,6 @@ describe('PubSub', function() { delete expectedBody.interval; delete expectedBody.maxInProgress; delete expectedBody.pushEndpoint; - delete expectedBody.reuseExisting; delete expectedBody.timeout; pubsub.topic = function() { @@ -625,7 +635,7 @@ describe('PubSub', function() { }; }); - it('should re-use existing subscription if specified', function(done) { + it('should re-use existing subscription', function(done) { var apiResponse = { code: 409 }; pubsub.subscription = function() { @@ -636,18 +646,9 @@ describe('PubSub', function() { callback({ code: 409 }, apiResponse); }; - // Don't re-use an existing subscription (error if one exists). - pubsub.subscribe(TOPIC_NAME, SUB_NAME, function(err, sub, resp) { - assert.equal(err.code, 409); - assert.strictEqual(resp, apiResponse); - }); - - // Re-use an existing subscription (ignore error if one exists). - var opts = { reuseExisting: true }; - pubsub.subscribe(TOPIC_NAME, SUB_NAME, opts, function(err, sub) { + pubsub.subscribe(TOPIC_NAME, SUB_NAME, function(err, subscription) { assert.ifError(err); - assert.deepEqual(sub, SUBSCRIPTION); - + assert.strictEqual(subscription, SUBSCRIPTION); done(); }); }); @@ -700,18 +701,20 @@ describe('PubSub', function() { var SUB_NAME = 'new-sub-name'; var CONFIG = { autoAck: true, interval: 90 }; - it('should throw if no name is provided', function() { - assert.throws(function() { - pubsub.subscription(); - }, /The name of a subscription is required\./); - }); - it('should return a Subscription object', function() { SubscriptionOverride = function() {}; var subscription = pubsub.subscription(SUB_NAME, {}); assert(subscription instanceof SubscriptionOverride); }); + it('should pass specified name to the Subscription', function(done) { + SubscriptionOverride = function(pubsub, options) { + assert.equal(options.name, SUB_NAME); + done(); + }; + pubsub.subscription(SUB_NAME, {}); + }); + it('should honor settings', function(done) { SubscriptionOverride = function(pubsub, options) { assert.deepEqual(options, CONFIG); @@ -720,18 +723,38 @@ describe('PubSub', function() { pubsub.subscription(SUB_NAME, CONFIG); }); - it('should pass specified name to the Subscription', function(done) { + it('should not require a name', function(done) { SubscriptionOverride = function(pubsub, options) { - assert.equal(options.name, SUB_NAME); + assert.deepEqual(options, { + name: undefined + }); done(); }; - pubsub.subscription(SUB_NAME, {}); + + pubsub.subscription(); }); - it('should not require options', function() { - assert.doesNotThrow(function() { - pubsub.subscription(SUB_NAME); - }); + it('should not require a name and accept options', function(done) { + SubscriptionOverride = function(pubsub, options) { + var expectedOptions = extend({}, CONFIG); + expectedOptions.name = undefined; + + assert.deepEqual(options, expectedOptions); + done(); + }; + + pubsub.subscription(CONFIG); + }); + + it('should not require options', function(done) { + SubscriptionOverride = function(pubsub, options) { + assert.deepEqual(options, { + name: SUB_NAME + }); + done(); + }; + + pubsub.subscription(SUB_NAME); }); }); diff --git a/packages/pubsub/test/subscription.js b/packages/pubsub/test/subscription.js index 81fcc475dcc..8498099f47a 100644 --- a/packages/pubsub/test/subscription.js +++ b/packages/pubsub/test/subscription.js @@ -106,13 +106,60 @@ describe('Subscription', function() { assert(promisified); }); - it('should format name', function(done) { - var formatName_ = Subscription.formatName_; - Subscription.formatName_ = function() { + describe('name', function() { + var FORMATTED_NAME = 'formatted-name'; + var GENERATED_NAME = 'generated-name'; + + var formatName_; + var generateName_; + + before(function() { + formatName_ = Subscription.formatName_; + generateName_ = Subscription.generateName_; + + Subscription.formatName_ = function() { + return FORMATTED_NAME; + }; + + Subscription.generateName_ = function() { + return GENERATED_NAME; + }; + }); + + afterEach(function() { + Subscription.formatName_ = function() { + return FORMATTED_NAME; + }; + + Subscription.generateName_ = function() { + return GENERATED_NAME; + }; + }); + + after(function() { Subscription.formatName_ = formatName_; - done(); - }; - new Subscription(PUBSUB, { name: SUB_NAME }); + Subscription.generateName_ = generateName_; + }); + + it('should generate name', function(done) { + Subscription.formatName_ = function(projectId, name) { + assert.strictEqual(name, GENERATED_NAME); + done(); + }; + + new Subscription(PUBSUB, {}); + }); + + it('should format name', function() { + Subscription.formatName_ = function(projectId, name) { + assert.strictEqual(projectId, PROJECT_ID); + assert.strictEqual(name, SUB_NAME); + return FORMATTED_NAME; + }; + + var subscription = new Subscription(PUBSUB, { name: SUB_NAME }); + assert.strictEqual(subscription.name, FORMATTED_NAME); + }); }); it('should honor configuration settings', function() {