UNPKG

jsforce

Version:

Salesforce API Library for JavaScript

777 lines (752 loc) 80.3 kB
import _Object$keys from "@babel/runtime-corejs3/core-js-stable/object/keys"; import _Object$getOwnPropertySymbols from "@babel/runtime-corejs3/core-js-stable/object/get-own-property-symbols"; import _filterInstanceProperty from "@babel/runtime-corejs3/core-js-stable/instance/filter"; import _Object$getOwnPropertyDescriptor from "@babel/runtime-corejs3/core-js-stable/object/get-own-property-descriptor"; import _forEachInstanceProperty from "@babel/runtime-corejs3/core-js-stable/instance/for-each"; import _Object$getOwnPropertyDescriptors from "@babel/runtime-corejs3/core-js-stable/object/get-own-property-descriptors"; import _Object$defineProperties from "@babel/runtime-corejs3/core-js-stable/object/define-properties"; import _Object$defineProperty from "@babel/runtime-corejs3/core-js-stable/object/define-property"; import _Reflect$construct from "@babel/runtime-corejs3/core-js-stable/reflect/construct"; import _get from "@babel/runtime-corejs3/helpers/get"; import _possibleConstructorReturn from "@babel/runtime-corejs3/helpers/possibleConstructorReturn"; import _getPrototypeOf from "@babel/runtime-corejs3/helpers/getPrototypeOf"; import _inherits from "@babel/runtime-corejs3/helpers/inherits"; import _asyncToGenerator from "@babel/runtime-corejs3/helpers/asyncToGenerator"; import _classCallCheck from "@babel/runtime-corejs3/helpers/classCallCheck"; import _createClass from "@babel/runtime-corejs3/helpers/createClass"; import _defineProperty from "@babel/runtime-corejs3/helpers/defineProperty"; import _objectWithoutProperties from "@babel/runtime-corejs3/helpers/objectWithoutProperties"; var _excluded = ["$"]; function _callSuper(t, o, e) { return o = _getPrototypeOf(o), _possibleConstructorReturn(t, _isNativeReflectConstruct() ? _Reflect$construct(o, e || [], _getPrototypeOf(t).constructor) : o.apply(t, e)); } function _isNativeReflectConstruct() { try { var t = !Boolean.prototype.valueOf.call(_Reflect$construct(Boolean, [], function () {})); } catch (t) {} return (_isNativeReflectConstruct = function _isNativeReflectConstruct() { return !!t; })(); } import "core-js/modules/es.error.cause.js"; import "core-js/modules/es.array.push.js"; import "core-js/modules/es.object.to-string.js"; import "core-js/modules/es.regexp.to-string.js"; import _regeneratorRuntime from "@babel/runtime-corejs3/regenerator"; import _Array$isArray from "@babel/runtime-corejs3/core-js-stable/array/is-array"; import _mapInstanceProperty from "@babel/runtime-corejs3/core-js-stable/instance/map"; import _concatInstanceProperty from "@babel/runtime-corejs3/core-js-stable/instance/concat"; import _JSON$stringify from "@babel/runtime-corejs3/core-js-stable/json/stringify"; import _Promise from "@babel/runtime-corejs3/core-js-stable/promise"; import _setTimeout from "@babel/runtime-corejs3/core-js-stable/set-timeout"; function ownKeys(e, r) { var t = _Object$keys(e); if (_Object$getOwnPropertySymbols) { var o = _Object$getOwnPropertySymbols(e); r && (o = _filterInstanceProperty(o).call(o, function (r) { return _Object$getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; } function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var _context15, _context16; var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? _forEachInstanceProperty(_context15 = ownKeys(Object(t), !0)).call(_context15, function (r) { _defineProperty(e, r, t[r]); }) : _Object$getOwnPropertyDescriptors ? _Object$defineProperties(e, _Object$getOwnPropertyDescriptors(t)) : _forEachInstanceProperty(_context16 = ownKeys(Object(t))).call(_context16, function (r) { _Object$defineProperty(e, r, _Object$getOwnPropertyDescriptor(t, r)); }); } return e; } /** * @file Manages Salesforce Metadata API * @author Shinichi Tomita <shinichi.tomita@gmail.com> */ import { EventEmitter } from 'events'; import { Readable } from 'stream'; import FormData from 'form-data'; import { registerModule } from '../jsforce'; import SOAP from '../soap'; import { isObject } from '../util/function'; import { ApiSchemas } from './metadata/schema'; export * from './metadata/schema'; /** * */ /** * */ function deallocateTypeWithMetadata(metadata) { var _ref = metadata, $ = _ref.$, md = _objectWithoutProperties(_ref, _excluded); return md; } function assignTypeWithMetadata(metadata, type) { var convert = function convert(md) { return _objectSpread(_defineProperty({}, '@xsi:type', type), md); }; return _Array$isArray(metadata) ? _mapInstanceProperty(metadata).call(metadata, convert) : convert(metadata); } /** * Class for Salesforce Metadata API */ export var MetadataApi = /*#__PURE__*/function () { /** * */ function MetadataApi(conn) { _classCallCheck(this, MetadataApi); /** * Polling interval in milliseconds */ _defineProperty(this, "pollInterval", 1000); /** * Polling timeout in milliseconds */ _defineProperty(this, "pollTimeout", 10000); this._conn = conn; } /** * Call Metadata API SOAP endpoint * * @private */ return _createClass(MetadataApi, [{ key: "_invoke", value: (function () { var _invoke2 = _asyncToGenerator( /*#__PURE__*/_regeneratorRuntime.mark(function _callee(method, message, schema) { var _context; var soapEndpoint, res; return _regeneratorRuntime.wrap(function _callee$(_context2) { while (1) switch (_context2.prev = _context2.next) { case 0: soapEndpoint = new SOAP(this._conn, { xmlns: 'http://soap.sforce.com/2006/04/metadata', endpointUrl: _concatInstanceProperty(_context = "".concat(this._conn.instanceUrl, "/services/Soap/m/")).call(_context, this._conn.version) }); _context2.next = 3; return soapEndpoint.invoke(method, message, schema ? { result: schema } : undefined, ApiSchemas); case 3: res = _context2.sent; return _context2.abrupt("return", res.result); case 5: case "end": return _context2.stop(); } }, _callee, this); })); function _invoke(_x, _x2, _x3) { return _invoke2.apply(this, arguments); } return _invoke; }() /** * Add one or more new metadata components to the organization. */ ) }, { key: "create", value: function create(type, metadata) { var isArray = _Array$isArray(metadata); metadata = assignTypeWithMetadata(metadata, type); var schema = isArray ? [ApiSchemas.SaveResult] : ApiSchemas.SaveResult; return this._invoke('createMetadata', { metadata: metadata }, schema); } /** * Read specified metadata components in the organization. */ }, { key: "read", value: function () { var _read = _asyncToGenerator( /*#__PURE__*/_regeneratorRuntime.mark(function _callee2(type, fullNames) { var _context3; var ReadResultSchema, res; return _regeneratorRuntime.wrap(function _callee2$(_context4) { while (1) switch (_context4.prev = _context4.next) { case 0: ReadResultSchema = type in ApiSchemas ? { type: ApiSchemas.ReadResult.type, props: { records: [type] } } : ApiSchemas.ReadResult; _context4.next = 3; return this._invoke('readMetadata', { type: type, fullNames: fullNames }, ReadResultSchema); case 3: res = _context4.sent; return _context4.abrupt("return", _Array$isArray(fullNames) ? _mapInstanceProperty(_context3 = res.records).call(_context3, deallocateTypeWithMetadata) : deallocateTypeWithMetadata(res.records[0])); case 5: case "end": return _context4.stop(); } }, _callee2, this); })); function read(_x4, _x5) { return _read.apply(this, arguments); } return read; }() /** * Update one or more metadata components in the organization. */ }, { key: "update", value: function update(type, metadata) { var isArray = _Array$isArray(metadata); metadata = assignTypeWithMetadata(metadata, type); var schema = isArray ? [ApiSchemas.SaveResult] : ApiSchemas.SaveResult; return this._invoke('updateMetadata', { metadata: metadata }, schema); } /** * Upsert one or more components in your organization's data. */ }, { key: "upsert", value: function upsert(type, metadata) { var isArray = _Array$isArray(metadata); metadata = assignTypeWithMetadata(metadata, type); var schema = isArray ? [ApiSchemas.UpsertResult] : ApiSchemas.UpsertResult; return this._invoke('upsertMetadata', { metadata: metadata }, schema); } /** * Deletes specified metadata components in the organization. */ }, { key: "delete", value: function _delete(type, fullNames) { var schema = _Array$isArray(fullNames) ? [ApiSchemas.SaveResult] : ApiSchemas.SaveResult; return this._invoke('deleteMetadata', { type: type, fullNames: fullNames }, schema); } /** * Rename fullname of a metadata component in the organization */ }, { key: "rename", value: function rename(type, oldFullName, newFullName) { return this._invoke('renameMetadata', { type: type, oldFullName: oldFullName, newFullName: newFullName }, ApiSchemas.SaveResult); } /** * Retrieves the metadata which describes your organization, including Apex classes and triggers, * custom objects, custom fields on standard objects, tab sets that define an app, * and many other components. */ }, { key: "describe", value: function describe(asOfVersion) { if (!asOfVersion) { asOfVersion = this._conn.version; } return this._invoke('describeMetadata', { asOfVersion: asOfVersion }, ApiSchemas.DescribeMetadataResult); } /** * Retrieves property information about metadata components in your organization */ }, { key: "list", value: function list(queries, asOfVersion) { if (!asOfVersion) { asOfVersion = this._conn.version; } return this._invoke('listMetadata', { queries: queries, asOfVersion: asOfVersion }, [ApiSchemas.FileProperties]); } /** * Checks the status of asynchronous metadata calls */ }, { key: "checkStatus", value: function checkStatus(asyncProcessId) { var res = this._invoke('checkStatus', { asyncProcessId: asyncProcessId }, ApiSchemas.AsyncResult); return new AsyncResultLocator(this, res); } /** * Retrieves XML file representations of components in an organization */ }, { key: "retrieve", value: function retrieve(request) { var res = this._invoke('retrieve', { request: request }, ApiSchemas.RetrieveResult); return new RetrieveResultLocator(this, res); } /** * Checks the status of declarative metadata call retrieve() and returns the zip file contents */ }, { key: "checkRetrieveStatus", value: function checkRetrieveStatus(asyncProcessId) { return this._invoke('checkRetrieveStatus', { asyncProcessId: asyncProcessId }, ApiSchemas.RetrieveResult); } /** * Will deploy a recently validated deploy request * * @param options.id = the deploy ID that's been validated already from a previous checkOnly deploy request * @param options.rest = a boolean whether or not to use the REST API * @returns the deploy ID of the recent validation request */ }, { key: "deployRecentValidation", value: (function () { var _deployRecentValidation = _asyncToGenerator( /*#__PURE__*/_regeneratorRuntime.mark(function _callee3(options) { var id, rest, response, messageBody, requestInfo, requestOptions; return _regeneratorRuntime.wrap(function _callee3$(_context5) { while (1) switch (_context5.prev = _context5.next) { case 0: id = options.id, rest = options.rest; if (!rest) { _context5.next = 10; break; } messageBody = _JSON$stringify({ validatedDeployRequestId: id }); requestInfo = { method: 'POST', url: "".concat(this._conn._baseUrl(), "/metadata/deployRequest"), body: messageBody, headers: { 'content-type': 'application/json' } }; requestOptions = { headers: 'json' }; // This is the deploy ID of the deployRecentValidation response, not // the already validated deploy ID (i.e., validateddeployrequestid). // REST returns an object with an id property, SOAP returns the id as a string directly. _context5.next = 7; return this._conn.request(requestInfo, requestOptions); case 7: response = _context5.sent.id; _context5.next = 13; break; case 10: _context5.next = 12; return this._invoke('deployRecentValidation', { validationId: id }); case 12: response = _context5.sent; case 13: return _context5.abrupt("return", response); case 14: case "end": return _context5.stop(); } }, _callee3, this); })); function deployRecentValidation(_x6) { return _deployRecentValidation.apply(this, arguments); } return deployRecentValidation; }() /** * Deploy components into an organization using zipped file representations * using the REST Metadata API instead of SOAP */ ) }, { key: "deployRest", value: function deployRest(zipInput) { var options = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {}; var form = new FormData(); form.append('file', zipInput, { contentType: 'application/zip', filename: 'package.xml' }); // Add the deploy options form.append('entity_content', _JSON$stringify({ deployOptions: options }), { contentType: 'application/json' }); var request = { url: '/metadata/deployRequest', method: 'POST', headers: _objectSpread({}, form.getHeaders()), body: form.getBuffer() }; var res = this._conn.request(request); return new DeployResultLocator(this, res); } /** * Deploy components into an organization using zipped file representations */ }, { key: "deploy", value: function deploy(zipInput) { var _this = this; var options = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {}; var res = _asyncToGenerator( /*#__PURE__*/_regeneratorRuntime.mark(function _callee4() { var zipContentB64; return _regeneratorRuntime.wrap(function _callee4$(_context6) { while (1) switch (_context6.prev = _context6.next) { case 0: _context6.next = 2; return new _Promise(function (resolve, reject) { if (isObject(zipInput) && 'pipe' in zipInput && typeof zipInput.pipe === 'function') { var bufs = []; zipInput.on('data', function (d) { return bufs.push(d); }); zipInput.on('error', reject); zipInput.on('end', function () { resolve(_concatInstanceProperty(Buffer).call(Buffer, bufs).toString('base64')); }); // zipInput.resume(); } else if (zipInput instanceof Buffer) { resolve(zipInput.toString('base64')); } else if (zipInput instanceof String || typeof zipInput === 'string') { resolve(zipInput); } else { throw 'Unexpected zipInput type'; } }); case 2: zipContentB64 = _context6.sent; return _context6.abrupt("return", _this._invoke('deploy', { ZipFile: zipContentB64, DeployOptions: options }, ApiSchemas.DeployResult)); case 4: case "end": return _context6.stop(); } }, _callee4); }))(); return new DeployResultLocator(this, res); } /** * Checks the status of declarative metadata call deploy(), using either * SOAP or REST APIs. SOAP is the default. */ }, { key: "checkDeployStatus", value: (function () { var _checkDeployStatus = _asyncToGenerator( /*#__PURE__*/_regeneratorRuntime.mark(function _callee5(asyncProcessId) { var includeDetails, rest, _context7, url, _args5 = arguments; return _regeneratorRuntime.wrap(function _callee5$(_context8) { while (1) switch (_context8.prev = _context8.next) { case 0: includeDetails = _args5.length > 1 && _args5[1] !== undefined ? _args5[1] : false; rest = _args5.length > 2 && _args5[2] !== undefined ? _args5[2] : false; if (!rest) { _context8.next = 9; break; } url = _concatInstanceProperty(_context7 = "/metadata/deployRequest/".concat(asyncProcessId)).call(_context7, includeDetails ? '?includeDetails=true' : ''); _context8.next = 6; return this._conn.requestGet(url); case 6: return _context8.abrupt("return", _context8.sent.deployResult); case 9: return _context8.abrupt("return", this._invoke('checkDeployStatus', { asyncProcessId: asyncProcessId, includeDetails: includeDetails }, ApiSchemas.DeployResult)); case 10: case "end": return _context8.stop(); } }, _callee5, this); })); function checkDeployStatus(_x7) { return _checkDeployStatus.apply(this, arguments); } return checkDeployStatus; }()) }, { key: "cancelDeploy", value: function () { var _cancelDeploy = _asyncToGenerator( /*#__PURE__*/_regeneratorRuntime.mark(function _callee6(id) { return _regeneratorRuntime.wrap(function _callee6$(_context9) { while (1) switch (_context9.prev = _context9.next) { case 0: return _context9.abrupt("return", this._invoke('cancelDeploy', { id: id })); case 1: case "end": return _context9.stop(); } }, _callee6, this); })); function cancelDeploy(_x8) { return _cancelDeploy.apply(this, arguments); } return cancelDeploy; }() }]); }(); /*--------------------------------------------*/ /** * The locator class for Metadata API asynchronous call result */ export var AsyncResultLocator = /*#__PURE__*/function (_EventEmitter) { /** * */ function AsyncResultLocator(meta, promise) { var _this2; _classCallCheck(this, AsyncResultLocator); _this2 = _callSuper(this, AsyncResultLocator); _this2._meta = meta; _this2._promise = promise; return _this2; } /** * Promise/A+ interface * http://promises-aplus.github.io/promises-spec/ * * @method Metadata~AsyncResultLocator#then */ _inherits(AsyncResultLocator, _EventEmitter); return _createClass(AsyncResultLocator, [{ key: "then", value: function then(onResolve, onReject) { return this._promise.then(onResolve, onReject); } /** * Check the status of async request */ }, { key: "check", value: (function () { var _check = _asyncToGenerator( /*#__PURE__*/_regeneratorRuntime.mark(function _callee7() { var result; return _regeneratorRuntime.wrap(function _callee7$(_context10) { while (1) switch (_context10.prev = _context10.next) { case 0: _context10.next = 2; return this._promise; case 2: result = _context10.sent; this._id = result.id; return _context10.abrupt("return", this._meta.checkStatus(result.id)); case 5: case "end": return _context10.stop(); } }, _callee7, this); })); function check() { return _check.apply(this, arguments); } return check; }() /** * Polling until async call status becomes complete or error */ ) }, { key: "poll", value: function poll(interval, timeout) { var _this3 = this; var startTime = new Date().getTime(); var poll = /*#__PURE__*/function () { var _ref3 = _asyncToGenerator( /*#__PURE__*/_regeneratorRuntime.mark(function _callee8() { var now, errMsg, result; return _regeneratorRuntime.wrap(function _callee8$(_context11) { while (1) switch (_context11.prev = _context11.next) { case 0: _context11.prev = 0; now = new Date().getTime(); if (!(startTime + timeout < now)) { _context11.next = 7; break; } errMsg = 'Polling time out.'; if (_this3._id) { errMsg += ' Process Id = ' + _this3._id; } _this3.emit('error', new Error(errMsg)); return _context11.abrupt("return"); case 7: _context11.next = 9; return _this3.check(); case 9: result = _context11.sent; if (result.done) { _this3.emit('complete', result); } else { _this3.emit('progress', result); _setTimeout(poll, interval); } _context11.next = 16; break; case 13: _context11.prev = 13; _context11.t0 = _context11["catch"](0); _this3.emit('error', _context11.t0); case 16: case "end": return _context11.stop(); } }, _callee8, null, [[0, 13]]); })); return function poll() { return _ref3.apply(this, arguments); }; }(); _setTimeout(poll, interval); } /** * Check and wait until the async requests become in completed status */ }, { key: "complete", value: function complete() { var _this4 = this; return new _Promise(function (resolve, reject) { _this4.on('complete', resolve); _this4.on('error', reject); _this4.poll(_this4._meta.pollInterval, _this4._meta.pollTimeout); }); } }]); }(EventEmitter); /*--------------------------------------------*/ /** * The locator class to track retreive() Metadata API call result */ export var RetrieveResultLocator = /*#__PURE__*/function (_AsyncResultLocator) { function RetrieveResultLocator() { _classCallCheck(this, RetrieveResultLocator); return _callSuper(this, RetrieveResultLocator, arguments); } _inherits(RetrieveResultLocator, _AsyncResultLocator); return _createClass(RetrieveResultLocator, [{ key: "complete", value: ( /** * Check and wait until the async request becomes in completed status, * and retrieve the result data. */ function () { var _complete = _asyncToGenerator( /*#__PURE__*/_regeneratorRuntime.mark(function _callee9() { var result; return _regeneratorRuntime.wrap(function _callee9$(_context12) { while (1) switch (_context12.prev = _context12.next) { case 0: _context12.next = 2; return _get(_getPrototypeOf(RetrieveResultLocator.prototype), "complete", this).call(this); case 2: result = _context12.sent; return _context12.abrupt("return", this._meta.checkRetrieveStatus(result.id)); case 4: case "end": return _context12.stop(); } }, _callee9, this); })); function complete() { return _complete.apply(this, arguments); } return complete; }() /** * Change the retrieved result to Node.js readable stream */ ) }, { key: "stream", value: function stream() { var _this5 = this; var resultStream = new Readable(); var reading = false; resultStream._read = /*#__PURE__*/_asyncToGenerator( /*#__PURE__*/_regeneratorRuntime.mark(function _callee10() { var result; return _regeneratorRuntime.wrap(function _callee10$(_context13) { while (1) switch (_context13.prev = _context13.next) { case 0: if (!reading) { _context13.next = 2; break; } return _context13.abrupt("return"); case 2: reading = true; _context13.prev = 3; _context13.next = 6; return _this5.complete(); case 6: result = _context13.sent; resultStream.push(Buffer.from(result.zipFile, 'base64')); resultStream.push(null); _context13.next = 14; break; case 11: _context13.prev = 11; _context13.t0 = _context13["catch"](3); resultStream.emit('error', _context13.t0); case 14: case "end": return _context13.stop(); } }, _callee10, null, [[3, 11]]); })); return resultStream; } }]); }(AsyncResultLocator); /*--------------------------------------------*/ /** * The locator class to track deploy() Metadata API call result * * @protected * @class Metadata~DeployResultLocator * @extends Metadata~AsyncResultLocator * @param {Metadata} meta - Metadata API object * @param {Promise.<Metadata~AsyncResult>} result - Promise object for async result of deploy() call */ export var DeployResultLocator = /*#__PURE__*/function (_AsyncResultLocator2) { function DeployResultLocator() { _classCallCheck(this, DeployResultLocator); return _callSuper(this, DeployResultLocator, arguments); } _inherits(DeployResultLocator, _AsyncResultLocator2); return _createClass(DeployResultLocator, [{ key: "complete", value: ( /** * Check and wait until the async request becomes in completed status, * and retrieve the result data. */ function () { var _complete2 = _asyncToGenerator( /*#__PURE__*/_regeneratorRuntime.mark(function _callee11(includeDetails) { var result; return _regeneratorRuntime.wrap(function _callee11$(_context14) { while (1) switch (_context14.prev = _context14.next) { case 0: _context14.next = 2; return _get(_getPrototypeOf(DeployResultLocator.prototype), "complete", this).call(this); case 2: result = _context14.sent; return _context14.abrupt("return", this._meta.checkDeployStatus(result.id, includeDetails)); case 4: case "end": return _context14.stop(); } }, _callee11, this); })); function complete(_x9) { return _complete2.apply(this, arguments); } return complete; }()) }]); }(AsyncResultLocator); /*--------------------------------------------*/ /* * Register hook in connection instantiation for dynamically adding this API module features */ registerModule('metadata', function (conn) { return new MetadataApi(conn); }); export default MetadataApi; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["EventEmitter","Readable","FormData","registerModule","SOAP","isObject","ApiSchemas","deallocateTypeWithMetadata","metadata","_ref","$","md","_objectWithoutProperties","_excluded","assignTypeWithMetadata","type","convert","_objectSpread","_defineProperty","_Array$isArray","_mapInstanceProperty","call","MetadataApi","conn","_classCallCheck","_conn","_createClass","key","value","_invoke2","_asyncToGenerator","_regeneratorRuntime","mark","_callee","method","message","schema","_context","soapEndpoint","res","wrap","_callee$","_context2","prev","next","xmlns","endpointUrl","_concatInstanceProperty","concat","instanceUrl","version","invoke","result","undefined","sent","abrupt","stop","_invoke","_x","_x2","_x3","apply","arguments","create","isArray","SaveResult","_read","_callee2","fullNames","_context3","ReadResultSchema","_callee2$","_context4","ReadResult","props","records","read","_x4","_x5","update","upsert","UpsertResult","_delete","rename","oldFullName","newFullName","describe","asOfVersion","DescribeMetadataResult","list","queries","FileProperties","checkStatus","asyncProcessId","AsyncResult","AsyncResultLocator","retrieve","request","RetrieveResult","RetrieveResultLocator","checkRetrieveStatus","_deployRecentValidation","_callee3","options","id","rest","response","messageBody","requestInfo","requestOptions","_callee3$","_context5","_JSON$stringify","validatedDeployRequestId","url","_baseUrl","body","headers","validationId","deployRecentValidation","_x6","deployRest","zipInput","length","form","append","contentType","filename","deployOptions","getHeaders","getBuffer","DeployResultLocator","deploy","_this","_callee4","zipContentB64","_callee4$","_context6","_Promise","resolve","reject","pipe","bufs","on","d","push","Buffer","toString","String","ZipFile","DeployOptions","DeployResult","_checkDeployStatus","_callee5","includeDetails","_context7","_args5","_callee5$","_context8","requestGet","deployResult","checkDeployStatus","_x7","_cancelDeploy","_callee6","_callee6$","_context9","cancelDeploy","_x8","_EventEmitter","meta","promise","_this2","_callSuper","_meta","_promise","_inherits","then","onResolve","onReject","_check","_callee7","_callee7$","_context10","_id","check","poll","interval","timeout","_this3","startTime","Date","getTime","_ref3","_callee8","now","errMsg","_callee8$","_context11","emit","Error","done","_setTimeout","t0","complete","_this4","pollInterval","pollTimeout","_AsyncResultLocator","_complete","_callee9","_callee9$","_context12","_get","_getPrototypeOf","prototype","stream","_this5","resultStream","reading","_callee10","_callee10$","_context13","from","zipFile","_AsyncResultLocator2","_complete2","_callee11","_callee11$","_context14","_x9"],"sources":["../../src/api/metadata.ts"],"sourcesContent":["/**\n * @file Manages Salesforce Metadata API\n * @author Shinichi Tomita <shinichi.tomita@gmail.com>\n */\nimport { EventEmitter } from 'events';\nimport { Readable } from 'stream';\nimport FormData from 'form-data';\nimport { registerModule } from '../jsforce';\nimport Connection from '../connection';\nimport SOAP from '../soap';\nimport { isObject } from '../util/function';\nimport { Schema, SoapSchemaDef, SoapSchema, HttpRequest } from '../types';\nimport {\n  ApiSchemas,\n  Metadata,\n  ReadResult,\n  SaveResult,\n  UpsertResult,\n  ListMetadataQuery,\n  FileProperties,\n  DescribeMetadataResult,\n  RetrieveRequest,\n  DeployOptions,\n  RetrieveResult,\n  DeployResult,\n  AsyncResult,\n  ApiSchemaTypes, CancelDeployResult,\n} from './metadata/schema';\nexport * from './metadata/schema';\n\n/**\n *\n */\ntype MetadataType_<\n  K extends keyof ApiSchemaTypes = keyof ApiSchemaTypes\n> = K extends keyof ApiSchemaTypes\n  ? ApiSchemaTypes[K] extends Metadata\n    ? K\n    : never\n  : never;\n\nexport type MetadataType = MetadataType_;\n\nexport type MetadataDefinition<\n  T extends string,\n  M extends Metadata = Metadata\n> = Metadata extends M\n  ? T extends keyof ApiSchemaTypes & MetadataType\n    ? ApiSchemaTypes[T] extends Metadata\n      ? ApiSchemaTypes[T]\n      : Metadata\n    : Metadata\n  : M;\n\ntype DeepPartial<T> = T extends any[]\n  ? Array<DeepPartial<T[number]>>\n  : T extends object\n  ? { [K in keyof T]?: DeepPartial<T[K]> }\n  : T;\n\nexport type InputMetadataDefinition<\n  T extends string,\n  M extends Metadata = Metadata\n> = DeepPartial<MetadataDefinition<T, M>>;\n\n/**\n *\n */\nfunction deallocateTypeWithMetadata<M extends Metadata>(metadata: M): M {\n  const { $, ...md } = metadata as any;\n  return md;\n}\n\nfunction assignTypeWithMetadata(metadata: Metadata | Metadata[], type: string) {\n  const convert = (md: Metadata) => ({ ['@xsi:type']: type, ...md });\n  return Array.isArray(metadata) ? metadata.map(convert) : convert(metadata);\n}\n\n/**\n * Class for Salesforce Metadata API\n */\nexport class MetadataApi<S extends Schema> {\n  _conn: Connection<S>;\n\n  /**\n   * Polling interval in milliseconds\n   */\n  pollInterval: number = 1000;\n\n  /**\n   * Polling timeout in milliseconds\n   */\n  pollTimeout: number = 10000;\n\n  /**\n   *\n   */\n  constructor(conn: Connection<S>) {\n    this._conn = conn;\n  }\n\n  /**\n   * Call Metadata API SOAP endpoint\n   *\n   * @private\n   */\n  async _invoke(\n    method: string,\n    message: object,\n    schema?: SoapSchema | SoapSchemaDef,\n  ) {\n    const soapEndpoint = new SOAP(this._conn, {\n      xmlns: 'http://soap.sforce.com/2006/04/metadata',\n      endpointUrl: `${this._conn.instanceUrl}/services/Soap/m/${this._conn.version}`,\n    });\n    const res = await soapEndpoint.invoke(\n      method,\n      message,\n      schema ? ({ result: schema } as SoapSchema) : undefined,\n      ApiSchemas,\n    );\n    return res.result;\n  }\n\n  /**\n   * Add one or more new metadata components to the organization.\n   */\n  create<\n    M extends Metadata = Metadata,\n    T extends MetadataType = MetadataType,\n    MD extends InputMetadataDefinition<T, M> = InputMetadataDefinition<T, M>\n  >(type: T, metadata: MD[]): Promise<SaveResult[]>;\n  create<\n    M extends Metadata = Metadata,\n    T extends MetadataType = MetadataType,\n    MD extends InputMetadataDefinition<T, M> = InputMetadataDefinition<T, M>\n  >(type: T, metadata: MD): Promise<SaveResult>;\n  create<\n    M extends Metadata = Metadata,\n    T extends MetadataType = MetadataType,\n    MD extends InputMetadataDefinition<T, M> = InputMetadataDefinition<T, M>\n  >(type: T, metadata: MD | MD[]): Promise<SaveResult | SaveResult[]>;\n  create(type: string, metadata: Metadata | Metadata[]) {\n    const isArray = Array.isArray(metadata);\n    metadata = assignTypeWithMetadata(metadata, type);\n    const schema = isArray ? [ApiSchemas.SaveResult] : ApiSchemas.SaveResult;\n    return this._invoke('createMetadata', { metadata }, schema);\n  }\n\n  /**\n   * Read specified metadata components in the organization.\n   */\n  read<\n    M extends Metadata = Metadata,\n    T extends MetadataType = MetadataType,\n    MD extends MetadataDefinition<T, M> = MetadataDefinition<T, M>\n  >(type: T, fullNames: string[]): Promise<MD[]>;\n  read<\n    M extends Metadata = Metadata,\n    T extends MetadataType = MetadataType,\n    MD extends MetadataDefinition<T, M> = MetadataDefinition<T, M>\n  >(type: T, fullNames: string): Promise<MD>;\n  read<\n    M extends Metadata = Metadata,\n    T extends MetadataType = MetadataType,\n    MD extends MetadataDefinition<T, M> = MetadataDefinition<T, M>\n  >(type: T, fullNames: string | string[]): Promise<MD | MD[]>;\n  async read(type: string, fullNames: string | string[]) {\n    const ReadResultSchema =\n      type in ApiSchemas\n        ? ({\n            type: ApiSchemas.ReadResult.type,\n            props: {\n              records: [type],\n            },\n          } as const)\n        : ApiSchemas.ReadResult;\n    const res: ReadResult = await this._invoke(\n      'readMetadata',\n      { type, fullNames },\n      ReadResultSchema,\n    );\n    return Array.isArray(fullNames)\n      ? res.records.map(deallocateTypeWithMetadata)\n      : deallocateTypeWithMetadata(res.records[0]);\n  }\n\n  /**\n   * Update one or more metadata components in the organization.\n   */\n  update<\n    M extends Metadata = Metadata,\n    T extends string = string,\n    MD extends InputMetadataDefinition<T, M> = InputMetadataDefinition<T, M>\n  >(type: T, metadata: Array<Partial<MD>>): Promise<SaveResult[]>;\n  update<\n    M extends Metadata = Metadata,\n    T extends string = string,\n    MD extends InputMetadataDefinition<T, M> = InputMetadataDefinition<T, M>\n  >(type: T, metadata: Partial<MD>): Promise<SaveResult>;\n  update<\n    M extends Metadata = Metadata,\n    T extends string = string,\n    MD extends InputMetadataDefinition<T, M> = InputMetadataDefinition<T, M>\n  >(\n    type: T,\n    metadata: Partial<MD> | Array<Partial<MD>>,\n  ): Promise<SaveResult | SaveResult[]>;\n  update(type: string, metadata: Metadata | Metadata[]) {\n    const isArray = Array.isArray(metadata);\n    metadata = assignTypeWithMetadata(metadata, type);\n    const schema = isArray ? [ApiSchemas.SaveResult] : ApiSchemas.SaveResult;\n    return this._invoke('updateMetadata', { metadata }, schema);\n  }\n\n  /**\n   * Upsert one or more components in your organization's data.\n   */\n  upsert<\n    M extends Metadata = Metadata,\n    T extends string = string,\n    MD extends InputMetadataDefinition<T, M> = InputMetadataDefinition<T, M>\n  >(type: T, metadata: MD[]): Promise<UpsertResult[]>;\n  upsert<\n    M extends Metadata = Metadata,\n    T extends string = string,\n    MD extends InputMetadataDefinition<T, M> = InputMetadataDefinition<T, M>\n  >(type: T, metadata: MD): Promise<UpsertResult>;\n  upsert<\n    M extends Metadata = Metadata,\n    T extends string = string,\n    MD extends InputMetadataDefinition<T, M> = InputMetadataDefinition<T, M>\n  >(type: T, metadata: MD | MD[]): Promise<UpsertResult | UpsertResult[]>;\n  upsert(type: string, metadata: Metadata | Metadata[]) {\n    const isArray = Array.isArray(metadata);\n    metadata = assignTypeWithMetadata(metadata, type);\n    const schema = isArray\n      ? [ApiSchemas.UpsertResult]\n      : ApiSchemas.UpsertResult;\n    return this._invoke('upsertMetadata', { metadata }, schema);\n  }\n\n  /**\n   * Deletes specified metadata components in the organization.\n   */\n  delete(type: string, fullNames: string[]): Promise<SaveResult[]>;\n  delete(type: string, fullNames: string): Promise<SaveResult>;\n  delete(\n    type: string,\n    fullNames: string | string[],\n  ): Promise<SaveResult | SaveResult[]>;\n  delete(type: string, fullNames: string | string[]) {\n    const schema = Array.isArray(fullNames)\n      ? [ApiSchemas.SaveResult]\n      : ApiSchemas.SaveResult;\n    return this._invoke('deleteMetadata', { type, fullNames }, schema);\n  }\n\n  /**\n   * Rename fullname of a metadata component in the organization\n   */\n  rename(\n    type: string,\n    oldFullName: string,\n    newFullName: string,\n  ): Promise<SaveResult> {\n    return this._invoke(\n      'renameMetadata',\n      { type, oldFullName, newFullName },\n      ApiSchemas.SaveResult,\n    );\n  }\n\n  /**\n   * Retrieves the metadata which describes your organization, including Apex classes and triggers,\n   * custom objects, custom fields on standard objects, tab sets that define an app,\n   * and many other components.\n   */\n  describe(asOfVersion?: string): Promise<DescribeMetadataResult> {\n    if (!asOfVersion) {\n      asOfVersion = this._conn.version;\n    }\n    return this._invoke(\n      'describeMetadata',\n      { asOfVersion },\n      ApiSchemas.DescribeMetadataResult,\n    );\n  }\n\n  /**\n   * Retrieves property information about metadata components in your organization\n   */\n  list(\n    queries: ListMetadataQuery | ListMetadataQuery[],\n    asOfVersion?: string,\n  ): Promise<FileProperties[]> {\n    if (!asOfVersion) {\n      asOfVersion = this._conn.version;\n    }\n    return this._invoke('listMetadata', { queries, asOfVersion }, [\n      ApiSchemas.FileProperties,\n    ]);\n  }\n\n  /**\n   * Checks the status of asynchronous metadata calls\n   */\n  checkStatus(asyncProcessId: string) {\n    const res = this._invoke(\n      'checkStatus',\n      { asyncProcessId },\n      ApiSchemas.AsyncResult,\n    );\n    return new AsyncResultLocator(this, res);\n  }\n\n  /**\n   * Retrieves XML file representations of components in an organization\n   */\n  retrieve(request: Partial<RetrieveRequest>) {\n    const res = this._invoke(\n      'retrieve',\n      { request },\n      ApiSchemas.RetrieveResult,\n    );\n    return new RetrieveResultLocator(this, res);\n  }\n\n  /**\n   * Checks the status of declarative metadata call retrieve() and returns the zip file contents\n   */\n  checkRetrieveStatus(asyncProcessId: string): Promise<RetrieveResult> {\n    return this._invoke(\n      'checkRetrieveStatus',\n      { asyncProcessId },\n      ApiSchemas.RetrieveResult,\n    );\n  }\n\n  /**\n   * Will deploy a recently validated deploy request\n   *\n   * @param options.id = the deploy ID that's been validated already from a previous checkOnly deploy request\n   * @param options.rest = a boolean whether or not to use the REST API\n   * @returns the deploy ID of the recent validation request\n   */\n  public async deployRecentValidation(options: {\n    id: string;\n    rest?: boolean;\n  }): Promise<string> {\n    const { id, rest } = options;\n    let response: string;\n    if (rest) {\n      const messageBody = JSON.stringify({\n        validatedDeployRequestId: id,\n      });\n\n      const requestInfo: HttpRequest = {\n        method: 'POST',\n        url: `${this._conn._baseUrl()}/metadata/deployRequest`,\n        body: messageBody,\n        headers: {\n          'content-type': 'application/json',\n        },\n      };\n      const requestOptions = { headers: 'json' };\n      // This is the deploy ID of the deployRecentValidation response, not\n      // the already validated deploy ID (i.e., validateddeployrequestid).\n      // REST returns an object with an id property, SOAP returns the id as a string directly.\n      response = (\n        await this._conn.request<{ id: string }>(requestInfo, requestOptions)\n      ).id;\n    } else {\n      response = await this._invoke('deployRecentValidation', {\n        validationId: id,\n      });\n    }\n\n    return response;\n  }\n\n  /**\n   * Deploy components into an organization using zipped file representations\n   * using the REST Metadata API instead of SOAP\n   */\n  deployRest(\n    zipInput: Buffer,\n    options: Partial<DeployOptions> = {},\n  ): DeployResultLocator<S> {\n    const form = new FormData();\n    form.append('file', zipInput, {\n      contentType: 'application/zip',\n      filename: 'package.xml',\n    });\n\n    // Add the deploy options\n    form.append('entity_content', JSON.stringify({ deployOptions: options }), {\n      contentType: 'application/json',\n    });\n\n    const request: HttpRequest = {\n      url: '/metadata/deployRequest',\n      method: 'POST',\n      headers: { ...form.getHeaders() },\n      body: form.getBuffer(),\n    };\n    const res = this._conn.request<AsyncResult>(request);\n\n    return new DeployResultLocator(this, res);\n  }\n\n  /**\n   * Deploy components into an organization using zipped file representations\n   */\n  deploy(\n    zipInput: Readable | Buffer | string,\n    options: Partial<DeployOptions> = {},\n  ): DeployResultLocator<S> {\n    const res = (async () => {\n      const zipContentB64 = await new Promise((resolve, reject) => {\n        if (\n          isObject(zipInput) &&\n          'pipe' in zipInput &&\n          typeof zipInput.pipe === 'function'\n        ) {\n          const bufs: Buffer[] = [];\n          zipInput.on('data', (d) => bufs.push(d));\n          zipInput.on('error', reject);\n          zipInput.on('end', () => {\n            resolve(Buffer.concat(bufs).toString('base64'));\n          });\n          // zipInput.resume();\n        } else if (