jsforce
Version:
Salesforce API Library for JavaScript
777 lines (752 loc) • 80.3 kB
JavaScript
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 (