UNPKG

typesuite2

Version:

TypeScript client for NetSuite SuiteTalk SOAP API

131 lines 18.4 kB
"use strict"; var __importDefault = (this && this.__importDefault) || function (mod) { return (mod && mod.__esModule) ? mod : { "default": mod }; }; Object.defineProperty(exports, "__esModule", { value: true }); exports.sendSoapRequest = exports.deserializeSoapResponse = exports.serializeSoapRequest = void 0; const crypto_1 = require("crypto"); const axios_1 = __importDefault(require("axios")); const jsonix_1 = require("jsonix"); const platform_core_1 = require("./netsuite_webservices/2019_2/platform_core"); const mappings_1 = __importDefault(require("./netsuite_webservices/2019_2/mappings")); const mappings_2 = __importDefault(require("./xmlsoap/mappings")); const envelope_1 = require("./xmlsoap/envelope"); const com_netsuite_webservices_platform_messages_2019_2_1 = require("./netsuite_webservices/2019_2/__mappings/com_netsuite_webservices_platform_messages_2019_2"); const platform_messages_1 = require("./netsuite_webservices/2019_2/platform_messages"); const ALL_MAPPINGS = [...mappings_2.default, ...mappings_1.default]; const JSONIX_CONTEXT_OPTIONS = { namespacePrefixes: { "http://www.w3.org/2001/XMLSchema-instance": "xsi", "http://schemas.xmlsoap.org/soap/envelope/": "soap", "urn:common_2019_2.platform.webservices.netsuite.com": "platform_common", "urn:core_2019_2.platform.webservices.netsuite.com": "platform_core", "urn:messages_2019_2.platform.webservices.netsuite.com": "platform_messages", }, }; function elementKeyFor(object) { var _a; const elementName = (_a = com_netsuite_webservices_platform_messages_2019_2_1.com_netsuite_webservices_platform_messages_2019_2.elementInfos.find((elementInfo) => { const typeInfo = elementInfo.typeInfo.split(".").slice(-1)[0]; return typeInfo === object.constructor.name; })) === null || _a === void 0 ? void 0 : _a.elementName; return `platform_messages:${ // eslint-disable-next-line @typescript-eslint/restrict-template-expressions elementName !== null && elementName !== void 0 ? elementName : "<<" + object.constructor.name + ">>"}`; } function serializeSoapRequest(passport, body, preferences) { const passportElement = { name: elementKeyFor(passport), value: passport }; const headerElements = [passportElement]; if (preferences) { const prefs = new platform_messages_1.Preferences(preferences); headerElements.push({ name: elementKeyFor(prefs), value: prefs, }); } const bodyElement = { name: elementKeyFor(body), value: body }; const envelope = new envelope_1.Envelope({ header: new envelope_1.Header({ any: headerElements }), body: new envelope_1.Body({ any: [bodyElement] }), }); const data = { "soap:Envelope": envelope }; const context = new jsonix_1.Jsonix.Context(ALL_MAPPINGS, JSONIX_CONTEXT_OPTIONS); const xmlString = context.createMarshaller().marshalString(data); return '<?xml version="1.0" encoding="utf-8"?>' + xmlString; } exports.serializeSoapRequest = serializeSoapRequest; function deserializeSoapResponse(xmlContent) { const context = new jsonix_1.Jsonix.Context(ALL_MAPPINGS); return context.createUnmarshaller().unmarshalString(xmlContent); } exports.deserializeSoapResponse = deserializeSoapResponse; function endpoint(config) { const account = config.account.replace("_", "-"); return `https://${account}.suitetalk.api.netsuite.com/services/NetSuitePort_${config.apiVersion}`; } /** * Serializes the provided request into XML and sends a SOAP request to the configured * endpoint. For successful responses, returns the data extracted the envelope and cast * into the type specified by the generic type parameter R. * * If the SOAP request results in a failed response, returns the SOAP Fault within * the envelope as a rejected promise. * * @param config * @param request * @param soapAction */ async function sendSoapRequest(config, request, soapAction) { const authToken = authenticateRequestWithTokenPassport(config); const soapXML = serializeSoapRequest(authToken, request, config.preferences); const axiosInstance = config.axiosInstance || axios_1.default; try { const response = await axiosInstance.post(endpoint(config), soapXML, { headers: { SOAPAction: soapAction, contentType: "text/xml; charset=UTF-8", }, }); const soapEnvelope = deserializeSoapResponse(response.data); return soapEnvelope.value.body.any[0].value; } catch (error) { if (error.response) { // Non 20x response const soapEnvelope = deserializeSoapResponse(error.response.data); const fault = soapEnvelope.value.body.any[0].value; return Promise.reject(fault); } else { // Network error. throw error; } } } exports.sendSoapRequest = sendSoapRequest; function authenticateRequestWithTokenPassport(config) { const nonce = crypto_1.randomBytes(18).toString("hex"); const timeStamp = Math.round(new Date().getTime() / 1000); const baseString = [ config.account, config.token.consumerKey, config.token.tokenKey, nonce, timeStamp, ].join("&"); const base64hash = crypto_1.createHmac("sha256", `${config.token.consumerSecret}&${config.token.tokenSecret}`) .update(baseString) .digest("base64"); return new platform_core_1.TokenPassport({ account: config.account, consumerKey: config.token.consumerKey, nonce: nonce, timestamp: timeStamp, token: config.token.tokenKey, signature: { algorithm: "HMAC_SHA256", value: base64hash, }, }); } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"soap.js","sourceRoot":"","sources":["../src/soap.ts"],"names":[],"mappings":"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","sourcesContent":["import { createHmac, randomBytes } from \"crypto\";\nimport axios from \"axios\";\nimport { Jsonix } from \"jsonix\";\nimport { TokenPassport } from \"./netsuite_webservices/2019_2/platform_core\";\nimport NetSuiteMappings from \"./netsuite_webservices/2019_2/mappings\";\nimport XmlSoapMappings from \"./xmlsoap/mappings\";\nimport { Body, Envelope, Fault, Header } from \"./xmlsoap/envelope\";\nimport { com_netsuite_webservices_platform_messages_2019_2 as platform } from \"./netsuite_webservices/2019_2/__mappings/com_netsuite_webservices_platform_messages_2019_2\";\nimport { Configuration } from \"./types\";\nimport {\n  Preferences,\n  PreferencesProps,\n} from \"./netsuite_webservices/2019_2/platform_messages\";\n\nconst ALL_MAPPINGS = [...XmlSoapMappings, ...NetSuiteMappings];\n\nconst JSONIX_CONTEXT_OPTIONS = {\n  namespacePrefixes: {\n    \"http://www.w3.org/2001/XMLSchema-instance\": \"xsi\",\n    \"http://schemas.xmlsoap.org/soap/envelope/\": \"soap\",\n    \"urn:common_2019_2.platform.webservices.netsuite.com\": \"platform_common\",\n    \"urn:core_2019_2.platform.webservices.netsuite.com\": \"platform_core\",\n    \"urn:messages_2019_2.platform.webservices.netsuite.com\":\n      \"platform_messages\",\n  },\n};\n\n// eslint-disable-next-line @typescript-eslint/no-empty-interface\ninterface soapObject {}\n\ninterface ElementInfo {\n  typeInfo: string;\n  elementName: string;\n}\n\nfunction elementKeyFor(object: { constructor: { name: string } }) {\n  const elementName = platform.elementInfos.find((elementInfo: ElementInfo) => {\n    const typeInfo = elementInfo.typeInfo.split(\".\").slice(-1)[0];\n    return typeInfo === object.constructor.name;\n  })?.elementName;\n  return `platform_messages:${\n    // eslint-disable-next-line @typescript-eslint/restrict-template-expressions\n    elementName ?? \"<<\" + object.constructor.name + \">>\"\n  }`;\n}\n\nexport function serializeSoapRequest(\n  passport: TokenPassport,\n  body: soapObject,\n  preferences?: PreferencesProps\n): string {\n  const passportElement = { name: elementKeyFor(passport), value: passport };\n  const headerElements: any[] = [passportElement];\n  if (preferences) {\n    const prefs = new Preferences(preferences);\n    headerElements.push({\n      name: elementKeyFor(prefs),\n      value: prefs,\n    });\n  }\n  const bodyElement = { name: elementKeyFor(body), value: body };\n  const envelope = new Envelope({\n    header: new Header({ any: headerElements }),\n    body: new Body({ any: [bodyElement] }),\n  });\n  const data = { \"soap:Envelope\": envelope };\n  const context = new Jsonix.Context(ALL_MAPPINGS, JSONIX_CONTEXT_OPTIONS);\n  const xmlString = context.createMarshaller().marshalString(data);\n\n  return '<?xml version=\"1.0\" encoding=\"utf-8\"?>' + xmlString;\n}\n\nexport function deserializeSoapResponse(xmlContent: string): XmlObject {\n  const context = new Jsonix.Context(ALL_MAPPINGS);\n  return context.createUnmarshaller<XmlObject>().unmarshalString(xmlContent);\n}\n\nexport interface XmlObject {\n  value: {\n    body: {\n      any: [{ value: unknown }];\n    };\n  };\n}\n\nfunction endpoint(config: Configuration): string {\n  const account = config.account.replace(\"_\", \"-\");\n  return `https://${account}.suitetalk.api.netsuite.com/services/NetSuitePort_${config.apiVersion}`;\n}\n\n/**\n * Serializes the provided request into XML and sends a SOAP request to the configured\n * endpoint. For successful responses, returns the data extracted the envelope and cast\n * into the type specified by the generic type parameter R.\n *\n * If the SOAP request results in a failed response, returns the SOAP Fault within\n * the envelope as a rejected promise.\n *\n * @param config\n * @param request\n * @param soapAction\n */\nexport async function sendSoapRequest<T, R>(\n  config: Configuration,\n  request: T,\n  soapAction: string\n): Promise<R> {\n  const authToken = authenticateRequestWithTokenPassport(config);\n  const soapXML = serializeSoapRequest(authToken, request, config.preferences);\n  const axiosInstance = config.axiosInstance || axios;\n  try {\n    const response = await axiosInstance.post(endpoint(config), soapXML, {\n      headers: {\n        SOAPAction: soapAction,\n        contentType: \"text/xml; charset=UTF-8\",\n      },\n    });\n    const soapEnvelope = deserializeSoapResponse(response.data);\n    return soapEnvelope.value.body.any[0].value as R;\n  } catch (error) {\n    if (error.response) {\n      // Non 20x response\n      const soapEnvelope = deserializeSoapResponse(error.response.data);\n      const fault = soapEnvelope.value.body.any[0].value as Fault;\n      return Promise.reject(fault);\n    } else {\n      // Network error.\n      throw error;\n    }\n  }\n}\n\nfunction authenticateRequestWithTokenPassport(\n  config: Configuration\n): TokenPassport {\n  const nonce = randomBytes(18).toString(\"hex\");\n  const timeStamp = Math.round(new Date().getTime() / 1000);\n  const baseString = [\n    config.account,\n    config.token.consumerKey,\n    config.token.tokenKey,\n    nonce,\n    timeStamp,\n  ].join(\"&\");\n  const base64hash = createHmac(\n    \"sha256\",\n    `${config.token.consumerSecret}&${config.token.tokenSecret}`\n  )\n    .update(baseString)\n    .digest(\"base64\");\n\n  return new TokenPassport({\n    account: config.account,\n    consumerKey: config.token.consumerKey,\n    nonce: nonce,\n    timestamp: timeStamp,\n    token: config.token.tokenKey,\n    signature: {\n      algorithm: \"HMAC_SHA256\",\n      value: base64hash,\n    },\n  });\n}\n"]}