UNPKG

@google-cloud/compute

Version:

Google Compute Engine Client Library for Node.js

298 lines (273 loc) 9.11 kB
/*! * Copyright 2015 Google Inc. All Rights Reserved. * * 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 common = require('@google-cloud/common'); var util = require('util'); /** * An Operation object allows you to interact with a Google Compute Engine * operation. * * An operation can be a * [GlobalOperation](https://cloud.google.com/compute/docs/reference/v1/globalOperations), * [RegionOperation](https://cloud.google.com/compute/docs/reference/v1/regionOperations), * or * [ZoneOperation](https://cloud.google.com/compute/docs/reference/v1/zoneOperations). * * @class * @param {Compute|Zone|Region} scope The scope of the operation: a `Compute`, * `Zone`, or `Region` object. * @param {string} name Operation name. * * @example * const Compute = require('@google-cloud/compute'); * const compute = new Compute(); * * //- * // Reference a global operation. * //- * const operation = compute.operation('operation-id'); * * //- * // Reference a region operation. * //- * const region = compute.region('us-central1'); * const operation = region.operation('operation-id'); * * //- * // Reference a zone operation. * //- * const zone = compute.zone('us-central1-a'); * const operation = zone.operation('operation-id'); * * //- * // All operations are event emitters. The status of each operation is polled * // continuously, starting only after you register a "complete" listener. * //- * operation.on('complete', function(metadata) { * // The operation is complete. * }); * * //- * // You can register a listener to monitor when the operation begins running. * //- * operation.on('running', function(metadata) { * // The operation is running. * }); * * //- * // Be sure to register an error handler as well to catch any issues which * // impeded the operation. * //- * operation.on('error', function(err) { * // An error occurred during the operation. * }); * * //- * // To force the Operation object to stop polling for updates, simply remove * // any "complete" listeners you've registered. * // * // The easiest way to do this is with `removeAllListeners()`. * //- * operation.removeAllListeners(); */ function Operation(scope, name) { var isCompute = scope.constructor.name === 'Compute'; var methods = { /** * Delete the operation. * * @see [GlobalOperations: delete API Documentation]{@link https://cloud.google.com/compute/docs/reference/v1/globalOperations/delete} * @see [RegionOperations: delete API Documentation]{@link https://cloud.google.com/compute/docs/reference/v1/regionOperations/delete} * @see [ZoneOperations: delete API Documentation]{@link https://cloud.google.com/compute/docs/reference/v1/zoneOperations/delete} * * @method Operation#delete * @param {function=} callback - The callback function. * @param {?error} callback.err - An error returned while making this * request. * @param {object} callback.apiResponse - The full API response. * * @example * const Compute = require('@google-cloud/compute'); * const compute = new Compute(); * const operation = compute.operation('operation-id'); * * operation.delete(function(err, apiResponse) {}); * * //- * // If the callback is omitted, we'll return a Promise. * //- * operation.delete().then(function(data) { * var apiResponse = data[0]; * }); */ delete: true, /** * Check if the operation exists. * * @method Operation#exists * @param {function} callback - The callback function. * @param {?error} callback.err - An error returned while making this * request. * @param {boolean} callback.exists - Whether the operation exists or not. * * @example * const Compute = require('@google-cloud/compute'); * const compute = new Compute(); * const operation = compute.operation('operation-id'); * * operation.exists(function(err, exists) {}); * * //- * // If the callback is omitted, we'll return a Promise. * //- * operation.exists().then(function(data) { * var exists = data[0]; * }); */ exists: true, /** * Get an operation if it exists. * * @method Operation#get * @example * const Compute = require('@google-cloud/compute'); * const compute = new Compute(); * const operation = compute.operation('operation-id'); * * operation.get(function(err, operation, apiResponse) { * // `operation` is an Operation object. * }); * * //- * // If the callback is omitted, we'll return a Promise. * //- * operation.get().then(function(data) { * var operation = data[0]; * var apiResponse = data[1]; * }); */ get: true, }; common.Operation.call(this, { parent: scope, baseUrl: isCompute ? '/global/operations' : '/operations', /** * @name Operation#id * @type {string} */ id: name, methods: methods, }); /** * @name Operation#name * @type {string} */ this.name = name; } util.inherits(Operation, common.Operation); /** * Get the operation's metadata. For a detailed description of metadata see * [Operation resource](https://goo.gl/sWm1rt). * * @see [GlobalOperations: get API Documentation]{@link https://cloud.google.com/compute/docs/reference/v1/globalOperations/get} * @see [RegionOperations: get API Documentation]{@link https://cloud.google.com/compute/docs/reference/v1/regionOperations/get} * @see [ZoneOperations: get API Documentation]{@link https://cloud.google.com/compute/docs/reference/v1/zoneOperations/get} * * @param {function=} callback - The callback function. * @param {?error} callback.err - An error returned while making this request * @param {object} callback.metadata - The disk's metadata. * @param {object} callback.apiResponse - The full API response. * * @example * const Compute = require('@google-cloud/compute'); * const compute = new Compute(); * * operation.getMetadata(function(err, metadata, apiResponse) { * // `metadata.error`: Contains errors if the operation failed. * // `metadata.warnings`: Contains warnings. * }); * * //- * // If the callback is omitted, we'll return a Promise. * //- * operation.getMetadata().then(function(data) { * const metadata = data[0]; * const apiResponse = data[1]; * }); */ Operation.prototype.getMetadata = function(callback) { var self = this; callback = callback || common.util.noop; var getMetadata = common.ServiceObject.prototype.getMetadata; getMetadata.call(this, function(err, apiResponse) { // An Operation entity contains a property named `error`. This makes // `request` think the operation failed, and will return an ApiError to // this callback. We have to make sure this isn't a false error by seeing if // the response body contains a property that wouldn't exist on a failed API // request (`name`). var requestFailed = err && (!apiResponse || apiResponse.name !== self.name); if (requestFailed) { callback(err, null, apiResponse); return; } self.metadata = apiResponse; callback(null, self.metadata, apiResponse); }); }; /** * Poll `getMetadata` to check the operation's status. This runs a loop to ping * the API on an interval. * * Note: This method is automatically called once a "complete" event handler is * registered on the operation. * * @private */ Operation.prototype.poll_ = function(callback) { var self = this; this.getMetadata(function(err, metadata, apiResponse) { // Parsing the response body will automatically create an ApiError object if // the operation failed. var parsedHttpRespBody = common.util.parseHttpRespBody(apiResponse); err = err || parsedHttpRespBody.err; if (err) { callback(err); return; } if (metadata.status === 'RUNNING' && !self.status) { self.status = metadata.status; self.emit('running', metadata); } if (metadata.status !== 'DONE') { callback(); return; } self.status = metadata.status; callback(null, metadata); }); }; /*! Developer Documentation * * All async methods (except for streams) will return a Promise in the event * that a callback is omitted. */ common.util.promisifyAll(Operation); /** * Reference to the {@link Operation} class. * @name module:@google-cloud/compute.Operation * @see Operation */ module.exports = Operation;