@ripwords/myinvois-client 0.2.23 → 0.2.25

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (35) hide show
  1. package/dist/api/documentManagement.d.ts +2 -2
  2. package/dist/api/documentSubmission.d.ts +2 -2
  3. package/dist/api/documentSubmission.js +2 -2
  4. package/dist/api/documentTypeManagement.d.ts +2 -2
  5. package/dist/api/notificationManagement.d.ts +2 -2
  6. package/dist/api/platformLogin.d.ts +2 -2
  7. package/dist/api/taxpayerValidation.d.ts +2 -2
  8. package/dist/{document-Dvonhaqt.cjs → document-BWReWH1w.cjs} +62 -20
  9. package/dist/document-BWReWH1w.cjs.map +1 -0
  10. package/dist/{document-BRxLmvIU.js → document-BbggZ72T.js} +50 -20
  11. package/dist/{documentSubmission-CWscnXdN.cjs → documentSubmission-DQy30z4E.cjs} +4 -4
  12. package/dist/documentSubmission-DQy30z4E.cjs.map +1 -0
  13. package/dist/{documentSubmission-D9hVtze6.js → documentSubmission-DRbswZAD.js} +3 -3
  14. package/dist/{documents-Dmn5FIRZ.d.cts → documents-BWpJ0B3C.d.cts} +7 -1
  15. package/dist/{documents-AlZIGjw2.d.ts → documents-DRQ6_toX.d.ts} +6 -0
  16. package/dist/index.cjs +2 -2
  17. package/dist/index.d.ts +2 -2
  18. package/dist/index.js +2 -2
  19. package/dist/index18.cjs +2 -2
  20. package/dist/index26.cjs +4 -2
  21. package/dist/index29.cjs +1 -1
  22. package/dist/index71.cts.map +1 -1
  23. package/dist/{taxpayer-TB35wh00.d.cts → taxpayer-CApK7x_O.d.cts} +2 -2
  24. package/dist/{taxpayer-DZmqFn6t.d.ts → taxpayer-DtyHXKwz.d.ts} +1 -1
  25. package/dist/types/documents.d.ts +1 -1
  26. package/dist/types/index.d.ts +2 -2
  27. package/dist/types/taxpayer.d.ts +2 -2
  28. package/dist/utils/document.d.ts +16 -6
  29. package/dist/utils/document.js +2 -2
  30. package/dist/utils/signature-diagnostics.d.ts +2 -2
  31. package/dist/utils/signature-diagnostics.js +1 -1
  32. package/dist/utils/validation.d.ts +2 -2
  33. package/package.json +1 -1
  34. package/dist/document-Dvonhaqt.cjs.map +0 -1
  35. package/dist/documentSubmission-CWscnXdN.cjs.map +0 -1
@@ -51,11 +51,11 @@ import "../YX-F34sJ7Ik.js";
51
51
  import "../ZX-CDQOfsHh.js";
52
52
  import "../XX-DOA-10JW.js";
53
53
  import "../unit-types-VgYXIwTT.js";
54
- import { DocumentStatus, DocumentSummary, DocumentValidationResult, DocumentValidationStepResult } from "../documents-AlZIGjw2.js";
54
+ import { DocumentStatus, DocumentSummary, DocumentValidationResult, DocumentValidationStepResult } from "../documents-DRQ6_toX.js";
55
55
  import "../payment-modes-g3DzLmWb.js";
56
56
  import "../signatures-CerHUrj3.js";
57
57
  import "../notifications-sFhgh3rJ.js";
58
- import "../taxpayer-DZmqFn6t.js";
58
+ import "../taxpayer-DtyHXKwz.js";
59
59
  import { Fetch } from "../utils-C4FoVKLq.js";
60
60
  import "../index-CygwSf0x.js";
61
61
 
@@ -51,11 +51,11 @@ import "../YX-F34sJ7Ik.js";
51
51
  import "../ZX-CDQOfsHh.js";
52
52
  import "../XX-DOA-10JW.js";
53
53
  import "../unit-types-VgYXIwTT.js";
54
- import { AllDocumentsV1_1, DocumentSummary, SigningCredentials, StandardError, SubmissionResponse, SubmissionStatus } from "../documents-AlZIGjw2.js";
54
+ import { AllDocumentsV1_1, DocumentSummary, SigningCredentials, StandardError, SubmissionResponse, SubmissionStatus } from "../documents-DRQ6_toX.js";
55
55
  import "../payment-modes-g3DzLmWb.js";
56
56
  import "../signatures-CerHUrj3.js";
57
57
  import "../notifications-sFhgh3rJ.js";
58
- import "../taxpayer-DZmqFn6t.js";
58
+ import "../taxpayer-DtyHXKwz.js";
59
59
  import { Fetch } from "../utils-C4FoVKLq.js";
60
60
  import "../index-CygwSf0x.js";
61
61
 
@@ -1,5 +1,5 @@
1
1
  import "../formatIdValue-qTxJqj9o.js";
2
- import "../document-BRxLmvIU.js";
3
- import { getSubmissionStatus, performDocumentAction, submitDocument } from "../documentSubmission-D9hVtze6.js";
2
+ import "../document-BbggZ72T.js";
3
+ import { getSubmissionStatus, performDocumentAction, submitDocument } from "../documentSubmission-DRbswZAD.js";
4
4
 
5
5
  export { getSubmissionStatus, performDocumentAction, submitDocument };
@@ -51,11 +51,11 @@ import "../YX-F34sJ7Ik.js";
51
51
  import "../ZX-CDQOfsHh.js";
52
52
  import "../XX-DOA-10JW.js";
53
53
  import "../unit-types-VgYXIwTT.js";
54
- import { DocumentTypeResponse, DocumentTypeVersionResponse, DocumentTypesResponse } from "../documents-AlZIGjw2.js";
54
+ import { DocumentTypeResponse, DocumentTypeVersionResponse, DocumentTypesResponse } from "../documents-DRQ6_toX.js";
55
55
  import "../payment-modes-g3DzLmWb.js";
56
56
  import "../signatures-CerHUrj3.js";
57
57
  import "../notifications-sFhgh3rJ.js";
58
- import "../taxpayer-DZmqFn6t.js";
58
+ import "../taxpayer-DtyHXKwz.js";
59
59
  import { Fetch } from "../utils-C4FoVKLq.js";
60
60
  import "../index-CygwSf0x.js";
61
61
 
@@ -51,11 +51,11 @@ import "../YX-F34sJ7Ik.js";
51
51
  import "../ZX-CDQOfsHh.js";
52
52
  import "../XX-DOA-10JW.js";
53
53
  import "../unit-types-VgYXIwTT.js";
54
- import "../documents-AlZIGjw2.js";
54
+ import "../documents-DRQ6_toX.js";
55
55
  import "../payment-modes-g3DzLmWb.js";
56
56
  import "../signatures-CerHUrj3.js";
57
57
  import { NotificationResponse, NotificationSearchParams } from "../notifications-sFhgh3rJ.js";
58
- import "../taxpayer-DZmqFn6t.js";
58
+ import "../taxpayer-DtyHXKwz.js";
59
59
  import { Fetch } from "../utils-C4FoVKLq.js";
60
60
  import "../index-CygwSf0x.js";
61
61
 
@@ -51,11 +51,11 @@ import "../YX-F34sJ7Ik.js";
51
51
  import "../ZX-CDQOfsHh.js";
52
52
  import "../XX-DOA-10JW.js";
53
53
  import "../unit-types-VgYXIwTT.js";
54
- import "../documents-AlZIGjw2.js";
54
+ import "../documents-DRQ6_toX.js";
55
55
  import "../payment-modes-g3DzLmWb.js";
56
56
  import "../signatures-CerHUrj3.js";
57
57
  import "../notifications-sFhgh3rJ.js";
58
- import "../taxpayer-DZmqFn6t.js";
58
+ import "../taxpayer-DtyHXKwz.js";
59
59
  import "../utils-C4FoVKLq.js";
60
60
  import { ClientCredentials } from "../index-CygwSf0x.js";
61
61
 
@@ -51,11 +51,11 @@ import "../YX-F34sJ7Ik.js";
51
51
  import "../ZX-CDQOfsHh.js";
52
52
  import "../XX-DOA-10JW.js";
53
53
  import "../unit-types-VgYXIwTT.js";
54
- import { RegistrationType } from "../documents-AlZIGjw2.js";
54
+ import { RegistrationType } from "../documents-DRQ6_toX.js";
55
55
  import "../payment-modes-g3DzLmWb.js";
56
56
  import "../signatures-CerHUrj3.js";
57
57
  import "../notifications-sFhgh3rJ.js";
58
- import { TaxpayerQRCodeResponse, TinSearchParams, TinSearchResponse } from "../taxpayer-DZmqFn6t.js";
58
+ import { TaxpayerQRCodeResponse, TinSearchParams, TinSearchResponse } from "../taxpayer-DtyHXKwz.js";
59
59
  import { Fetch } from "../utils-C4FoVKLq.js";
60
60
  import "../index-CygwSf0x.js";
61
61
 
@@ -254,18 +254,26 @@ const generateCleanUBLDocument = (invoices) => {
254
254
  };
255
255
  };
256
256
  /**
257
- * Step 2: Calculate Document Digest
258
- * FIXED: Remove UBLExtensions and Signature before hashing (DS322)
259
- * Based on working implementation pattern
257
+ * Step 1: Transform the document for hashing or transmission
258
+ * Removes UBLExtensions and Signature, and minifies the JSON
259
+ * Returns the minified, cleaned JSON string
260
260
  */
261
- const calculateDocumentDigest = (invoices) => {
261
+ const transformDocumentForHashing = (invoices) => {
262
262
  const cleanDocument = generateCleanUBLDocument(invoices);
263
- const documentForHashing = JSON.parse(JSON.stringify(cleanDocument));
264
- if (documentForHashing.Invoice && Array.isArray(documentForHashing.Invoice)) documentForHashing.Invoice.forEach((invoice) => {
263
+ const documentForTransform = JSON.parse(JSON.stringify(cleanDocument));
264
+ if (documentForTransform.Invoice && Array.isArray(documentForTransform.Invoice)) documentForTransform.Invoice.forEach((invoice) => {
265
265
  delete invoice.UBLExtensions;
266
266
  delete invoice.Signature;
267
267
  });
268
- const documentString = JSON.stringify(documentForHashing);
268
+ return JSON.stringify(documentForTransform);
269
+ };
270
+ /**
271
+ * Step 2: Calculate Document Digest
272
+ * FIXED: Remove UBLExtensions and Signature before hashing (DS322)
273
+ * Based on working implementation pattern
274
+ */
275
+ const calculateDocumentDigest = (invoices) => {
276
+ const documentString = transformDocumentForHashing(invoices);
269
277
  const hash = crypto.default.createHash("sha256");
270
278
  hash.update(documentString, "utf8");
271
279
  return hash.digest("base64");
@@ -290,8 +298,8 @@ const extractCertificateInfo = (certificatePem) => {
290
298
  const cert = new crypto.X509Certificate(certificatePem);
291
299
  const serialNumberHex = cert.serialNumber;
292
300
  const normalizeDistinguishedName = (dn) => {
293
- const normalized = dn.split("\n").map((line) => line.trim()).filter((line) => line.length > 0).join(", ");
294
- return normalized.replace(/\s*=\s*/g, "=").replace(/,\s+/g, ", ").replace(/\r/g, "").replace(/\s{2,}/g, " ").trim();
301
+ const parts = dn.split(/\r?\n|,\s*/).map((part) => part.trim()).filter((part) => part.length > 0).reverse();
302
+ return parts.join(", ").replace(/\s*=\s*/g, "=");
295
303
  };
296
304
  const formatSerialNumber = (serialHex) => {
297
305
  const decimal = BigInt("0x" + serialHex).toString();
@@ -318,7 +326,10 @@ const createSignedProperties = (certificateDigest, signingTime, issuerName, seri
318
326
  SigningTime: [{ _: signingTime }],
319
327
  SigningCertificate: [{ Cert: [{
320
328
  CertDigest: [{
321
- DigestMethod: [{ Algorithm: "http://www.w3.org/2001/04/xmlenc#sha256" }],
329
+ DigestMethod: [{
330
+ _: "",
331
+ Algorithm: "http://www.w3.org/2001/04/xmlenc#sha256"
332
+ }],
322
333
  DigestValue: [{ _: certificateDigest }]
323
334
  }],
324
335
  IssuerSerial: [{
@@ -331,11 +342,17 @@ const createSignedProperties = (certificateDigest, signingTime, issuerName, seri
331
342
  };
332
343
  /**
333
344
  * Step 6: Calculate SignedProperties Digest
334
- * FIXED: Add Target wrapper and use direct JSON stringify (DS320)
335
- * Based on working implementation pattern
345
+ * Calculates the digest over the correct structure for validator compliance.
336
346
  */
337
- const calculateSignedPropertiesDigest = (signedProperties) => {
338
- const signedPropertiesString = canonicalizeJSON(signedProperties);
347
+ const calculateSignedPropertiesDigest = (signedProperties, useTargetWrapper = true) => {
348
+ let signedPropertiesString;
349
+ if (useTargetWrapper) {
350
+ const digestObj = {
351
+ Target: "signature",
352
+ SignedProperties: signedProperties.SignedProperties
353
+ };
354
+ signedPropertiesString = JSON.stringify(digestObj);
355
+ } else signedPropertiesString = JSON.stringify(signedProperties.SignedProperties);
339
356
  const hash = crypto.default.createHash("sha256");
340
357
  hash.update(signedPropertiesString, "utf8");
341
358
  return hash.digest("base64");
@@ -374,10 +391,10 @@ const createSignedInfoAndSign = (docDigest, propsDigest, privateKeyPem) => {
374
391
  DigestValue: [{ _: propsDigest }]
375
392
  }]
376
393
  };
377
- const signedInfoString = JSON.stringify(signedInfo);
394
+ const signedInfoRaw = JSON.stringify(signedInfo);
378
395
  try {
379
396
  const signer = crypto.default.createSign("RSA-SHA256");
380
- signer.update(signedInfoString, "utf8");
397
+ signer.update(signedInfoRaw, "utf8");
381
398
  const signatureValue = signer.sign(privateKeyPem, "base64");
382
399
  return {
383
400
  signedInfo,
@@ -388,18 +405,31 @@ const createSignedInfoAndSign = (docDigest, propsDigest, privateKeyPem) => {
388
405
  }
389
406
  };
390
407
  /**
408
+ * Signs the minified document string using the provided private key PEM
409
+ * Returns the signature as a base64 string
410
+ */
411
+ const signDocumentString = (documentString, privateKeyPem) => {
412
+ const signature = crypto.default.sign("sha256", Buffer.from(documentString, "utf8"), {
413
+ key: privateKeyPem,
414
+ padding: crypto.default.constants.RSA_PKCS1_PADDING
415
+ });
416
+ return signature.toString("base64");
417
+ };
418
+ /**
391
419
  * Complete document generation with signatures
392
420
  * Follows the complete MyInvois JSON signature creation process (Steps 1-7)
393
421
  */
394
422
  const generateCompleteDocument = (invoices, signingCredentials) => {
395
423
  try {
396
424
  const docDigest = calculateDocumentDigest(invoices);
425
+ const documentString = transformDocumentForHashing(invoices);
426
+ const docSignature = signDocumentString(documentString, signingCredentials.privateKeyPem);
397
427
  const signingTime = (/* @__PURE__ */ new Date()).toISOString();
398
428
  const certInfo = extractCertificateInfo(signingCredentials.certificatePem);
399
429
  const certificateDigest = calculateCertificateDigest(signingCredentials.certificatePem);
400
430
  const signedProperties = createSignedProperties(certificateDigest, signingTime, certInfo.issuerName, certInfo.serialNumber);
401
- const propsDigest = calculateSignedPropertiesDigest(signedProperties);
402
- const { signedInfo, signatureValue } = createSignedInfoAndSign(docDigest, propsDigest, signingCredentials.privateKeyPem);
431
+ const propsDigest = calculateSignedPropertiesDigest(signedProperties, true);
432
+ const { signedInfo } = createSignedInfoAndSign(docDigest, propsDigest, signingCredentials.privateKeyPem);
403
433
  const certificate = signingCredentials.certificatePem.replace(/-----BEGIN CERTIFICATE-----/g, "").replace(/-----END CERTIFICATE-----/g, "").replace(/\s+/g, "");
404
434
  const signedInvoices = invoices.map((invoice) => {
405
435
  const cleanInvoice = generateCleanInvoiceObject(invoice);
@@ -424,7 +454,7 @@ const generateCompleteDocument = (invoices, signingCredentials) => {
424
454
  X509SerialNumber: [{ _: certInfo.serialNumber }]
425
455
  }]
426
456
  }] }],
427
- SignatureValue: [{ _: signatureValue }],
457
+ SignatureValue: [{ _: docSignature }],
428
458
  SignedInfo: [signedInfo]
429
459
  }]
430
460
  }] }] }]
@@ -597,10 +627,22 @@ Object.defineProperty(exports, 'isPercentageTax', {
597
627
  return isPercentageTax;
598
628
  }
599
629
  });
630
+ Object.defineProperty(exports, 'signDocumentString', {
631
+ enumerable: true,
632
+ get: function () {
633
+ return signDocumentString;
634
+ }
635
+ });
600
636
  Object.defineProperty(exports, 'sortObjectKeys', {
601
637
  enumerable: true,
602
638
  get: function () {
603
639
  return sortObjectKeys;
604
640
  }
605
641
  });
606
- //# sourceMappingURL=document-Dvonhaqt.cjs.map
642
+ Object.defineProperty(exports, 'transformDocumentForHashing', {
643
+ enumerable: true,
644
+ get: function () {
645
+ return transformDocumentForHashing;
646
+ }
647
+ });
648
+ //# sourceMappingURL=document-BWReWH1w.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"document-BWReWH1w.cjs","names":["item: InvoiceLineItem","doc: AllDocumentsV1_1","obj: unknown","sortedObj: Record<string, unknown>","invoice: AllDocumentsV1_1","invoices: AllDocumentsV1_1[]","invoice: InvoiceSubmission","certificatePem: string","X509Certificate","dn: string","serialHex: string","error: unknown","certificateDigest: string","signingTime: string","issuerName: string","serialNumber: string","signedProperties: SignedPropertiesObject","useTargetWrapper: boolean","signedPropertiesString: string","digestObj: {\n Target: string\n SignedProperties: SignedPropertiesObject['SignedProperties']\n }","docDigest: string","propsDigest: string","privateKeyPem: string","signedInfo: SignedInfoObject","documentString: string","signingCredentials: SigningCredentials","params: {\n itemClassificationCode: ClassificationCode\n itemDescription: string\n unitPrice: number\n quantity?: number\n taxType: TaxTypeCode\n taxRate: number\n totalTaxableAmountPerLine?: number\n}","params: {\n itemClassificationCode: ClassificationCode\n itemDescription: string\n unitPrice: number\n quantity?: number\n taxType: TaxTypeCode\n taxPerUnitAmount: number\n baseUnitMeasure: number\n baseUnitMeasureCode: UnitTypeCode\n totalTaxableAmountPerLine?: number\n}","lineItems: InvoiceLineItem[]"],"sources":["../src/utils/document.ts"],"sourcesContent":["import crypto, { X509Certificate } from 'crypto'\nimport {\n InvoiceSubmission,\n AllDocumentsV1_1,\n SigningCredentials,\n SignedPropertiesObject,\n UBLDocument,\n CompleteInvoice,\n SignedInfoObject,\n UnitTypeCode,\n InvoiceLineItem,\n} from '../types'\nimport type { ClassificationCode } from '../types'\nimport type { TaxTypeCode } from '../types'\nimport { formatIdValue } from './formatIdValue'\n\n/**\n * MyInvois v1.1 Document Generation and Signing Utilities\n * Strictly follows: https://sdk.myinvois.hasil.gov.my/documents/invoice-v1-1\n * JSON Signature Guide: https://sdk.myinvois.hasil.gov.my/signature-creation-json/\n */\n\n/**\n * Determines if a line item uses fixed rate taxation\n */\nexport const isFixedRateTax = (item: InvoiceLineItem): boolean => {\n return (\n item.taxPerUnitAmount !== undefined && item.baseUnitMeasure !== undefined\n )\n}\n\n/**\n * Determines if a line item uses percentage taxation\n */\nexport const isPercentageTax = (item: InvoiceLineItem): boolean => {\n return item.taxRate !== undefined && !isFixedRateTax(item)\n}\n\n/**\n * Calculates expected tax amount for a line item based on its tax type\n */\nexport const calculateExpectedTaxAmount = (item: InvoiceLineItem): number => {\n if (isFixedRateTax(item)) {\n return item.taxPerUnitAmount! * item.baseUnitMeasure!\n } else if (isPercentageTax(item)) {\n return (item.totalTaxableAmountPerLine * item.taxRate!) / 100\n }\n return 0\n}\n\n/**\n * Extracts the line-item array from any document variant\n */\nconst getLineItems = (doc: AllDocumentsV1_1): InvoiceLineItem[] => {\n if ('invoiceLineItems' in doc) return doc.invoiceLineItems\n if ('creditNoteLineItems' in doc) return doc.creditNoteLineItems\n if ('debitNoteLineItems' in doc) return doc.debitNoteLineItems\n if ('refundNoteLineItems' in doc) return doc.refundNoteLineItems\n if ('selfBilledCreditNoteLineItems' in doc)\n return doc.selfBilledCreditNoteLineItems\n if ('selfBilledRefundNoteLineItems' in doc)\n return doc.selfBilledRefundNoteLineItems\n // Fallback (should never happen with exhaustive types)\n return []\n}\n\n/**\n * Helper function to recursively sort object keys for JSON canonicalization\n */\nexport function sortObjectKeys(obj: unknown): unknown {\n if (obj === null || typeof obj !== 'object') {\n return obj\n }\n\n if (Array.isArray(obj)) {\n return obj.map(sortObjectKeys)\n }\n\n const sortedObj: Record<string, unknown> = {}\n const keys = Object.keys(obj as Record<string, unknown>).sort()\n\n for (const key of keys) {\n sortedObj[key] = sortObjectKeys((obj as Record<string, unknown>)[key])\n }\n\n return sortedObj\n}\n\n/**\n * Enhanced canonicalization following MyInvois specification exactly\n * Key changes: ensure consistent ordering and formatting\n */\nexport const canonicalizeJSON = (obj: unknown): string => {\n const sortedObj = sortObjectKeys(obj)\n // Use compact JSON with no extra whitespace\n return JSON.stringify(sortedObj, null, 0)\n}\n\n/**\n * Generates a clean invoice object following MyInvois v1.1 specification exactly\n * This is the base invoice structure WITHOUT signature elements (for hash calculation)\n *\n * Key requirements from working documents:\n * - All mandatory fields must be present\n * - Many optional fields must be present even if empty\n * - Specific field ordering and structure\n * - Correct listID values (e.g., \"3166-1\" not \"ISO3166-1\")\n */\nexport const generateCleanInvoiceObject = (\n invoice: AllDocumentsV1_1,\n): InvoiceSubmission => {\n const lineItems = getLineItems(invoice)\n return {\n // === MANDATORY CORE FIELDS ===\n ID: [{ _: invoice.eInvoiceCodeOrNumber }],\n IssueDate: [{ _: invoice.eInvoiceDate }],\n IssueTime: [{ _: invoice.eInvoiceTime }],\n InvoiceTypeCode: [\n {\n _: invoice.eInvoiceTypeCode,\n listVersionID: invoice.eInvoiceVersion || '1.1',\n },\n ],\n DocumentCurrencyCode: [{ _: invoice.invoiceCurrencyCode }],\n\n // === BILLING REFERENCE (only for credit/debit/refund notes) ===\n ...('originalEInvoiceReferenceNumber' in invoice &&\n 'originalEInvoiceInternalId' in invoice &&\n invoice.originalEInvoiceReferenceNumber\n ? {\n BillingReference: [\n {\n InvoiceDocumentReference: [\n {\n UUID: [\n {\n _: invoice.originalEInvoiceReferenceNumber,\n },\n ],\n ID: [\n {\n _: invoice.originalEInvoiceInternalId,\n },\n ],\n },\n ],\n },\n ],\n }\n : {}),\n\n // === SUPPLIER PARTY (AccountingSupplierParty) ===\n AccountingSupplierParty: [\n {\n Party: [\n {\n // Industry Classification - required field\n IndustryClassificationCode: [\n {\n _: invoice.supplier.industryClassificationCode,\n name: invoice.supplier.industryClassificationDescription,\n },\n ],\n\n // Party Identifications\n PartyIdentification: [\n {\n ID: [\n {\n _: invoice.supplier.tin,\n schemeID: 'TIN',\n },\n ],\n },\n {\n ID: [\n {\n _: formatIdValue(invoice.supplier.registrationNumber),\n schemeID: invoice.supplier.registrationType || 'NRIC',\n },\n ],\n },\n ],\n\n // Postal Address - FIXED listID format\n PostalAddress: [\n {\n CityName: [{ _: invoice.supplier.address.cityName }],\n CountrySubentityCode: [{ _: invoice.supplier.address.state }],\n AddressLine: [\n {\n Line: [{ _: invoice.supplier.address.addressLine0 }],\n },\n ],\n Country: [\n {\n IdentificationCode: [\n {\n _: invoice.supplier.address.country || 'MYS',\n listID: '3166-1', // FIXED: was \"ISO3166-1\"\n listAgencyID: 'ISO',\n },\n ],\n },\n ],\n },\n ],\n\n // Party Legal Entity\n PartyLegalEntity: [\n {\n RegistrationName: [{ _: invoice.supplier.name }],\n },\n ],\n\n // Contact Information\n Contact: [\n {\n Telephone: [{ _: invoice.supplier.contactNumber || '' }],\n },\n ],\n },\n ],\n },\n ],\n\n // === BUYER PARTY (AccountingCustomerParty) ===\n AccountingCustomerParty: [\n {\n Party: [\n {\n // Party Identifications\n PartyIdentification: [\n {\n ID: [\n {\n _: invoice.buyer.tin,\n schemeID: 'TIN',\n },\n ],\n },\n {\n ID: [\n {\n _: formatIdValue(invoice.buyer.registrationNumber),\n schemeID: invoice.buyer.registrationType || 'NRIC',\n },\n ],\n },\n {\n ID: [\n {\n _: invoice.buyer.sstRegistrationNumber || 'NA',\n schemeID: 'SST',\n },\n ],\n },\n ],\n\n // Postal Address - FIXED listID format\n PostalAddress: [\n {\n CityName: [{ _: invoice.buyer.address.cityName }],\n CountrySubentityCode: [{ _: invoice.buyer.address.state }],\n AddressLine: [\n {\n Line: [{ _: invoice.buyer.address.addressLine0 }],\n },\n ],\n Country: [\n {\n IdentificationCode: [\n {\n _: invoice.buyer.address.country || 'MYS',\n listID: '3166-1', // FIXED: was \"ISO3166-1\"\n listAgencyID: 'ISO',\n },\n ],\n },\n ],\n },\n ],\n\n // Party Legal Entity\n PartyLegalEntity: [\n {\n RegistrationName: [{ _: invoice.buyer.name }],\n },\n ],\n\n // Contact Information\n Contact: [\n {\n Telephone: [{ _: invoice.buyer.contactNumber || '' }],\n },\n ],\n },\n ],\n },\n ],\n\n // === TAX TOTAL ===\n TaxTotal: [\n {\n TaxAmount: [\n {\n _: invoice.taxTotal.taxAmount,\n currencyID: invoice.invoiceCurrencyCode,\n },\n ],\n TaxSubtotal: [\n // Generate basic tax subtotal from invoice line items\n {\n TaxableAmount: [\n {\n _: invoice.legalMonetaryTotal.taxExclusiveAmount,\n currencyID: invoice.invoiceCurrencyCode,\n },\n ],\n TaxAmount: [\n {\n _: invoice.taxTotal.taxAmount,\n currencyID: invoice.invoiceCurrencyCode,\n },\n ],\n TaxCategory: [\n {\n ID: [{ _: lineItems[0]?.taxType || '01' }],\n TaxScheme: [\n {\n ID: [\n {\n _: 'OTH',\n schemeAgencyID: '6',\n schemeID: 'UN/ECE 5153',\n },\n ],\n },\n ],\n },\n ],\n },\n ],\n },\n ],\n\n // === LEGAL MONETARY TOTAL ===\n LegalMonetaryTotal: [\n {\n LineExtensionAmount: [\n {\n _: invoice.legalMonetaryTotal.taxExclusiveAmount,\n currencyID: invoice.invoiceCurrencyCode,\n },\n ],\n TaxExclusiveAmount: [\n {\n _: invoice.legalMonetaryTotal.taxExclusiveAmount,\n currencyID: invoice.invoiceCurrencyCode,\n },\n ],\n TaxInclusiveAmount: [\n {\n _: invoice.legalMonetaryTotal.taxInclusiveAmount,\n currencyID: invoice.invoiceCurrencyCode,\n },\n ],\n PayableAmount: [\n {\n _: invoice.legalMonetaryTotal.payableAmount,\n currencyID: invoice.invoiceCurrencyCode,\n },\n ],\n },\n ],\n\n // === INVOICE LINES ===\n InvoiceLine: lineItems.map((item, index) => ({\n ID: [{ _: (index + 1).toString() }],\n\n // Item Information\n Item: [\n {\n CommodityClassification: [\n {\n ItemClassificationCode: [\n {\n _: item.itemClassificationCode,\n listID: 'CLASS',\n },\n ],\n },\n ],\n Description: [{ _: item.itemDescription }],\n },\n ],\n\n ItemPriceExtension: [\n {\n Amount: [\n {\n _: item.totalTaxableAmountPerLine,\n currencyID: invoice.invoiceCurrencyCode,\n },\n ],\n },\n ],\n\n LineExtensionAmount: [\n {\n _: item.totalTaxableAmountPerLine,\n currencyID: invoice.invoiceCurrencyCode,\n },\n ],\n\n // Price Information\n Price: [\n {\n PriceAmount: [\n {\n _: item.unitPrice,\n currencyID: invoice.invoiceCurrencyCode,\n },\n ],\n },\n ],\n\n // Tax Information for line\n TaxTotal: [\n {\n TaxAmount: [\n {\n _: item.taxAmount,\n currencyID: invoice.invoiceCurrencyCode,\n },\n ],\n TaxSubtotal: [\n {\n TaxableAmount: [\n {\n _: item.totalTaxableAmountPerLine,\n currencyID: invoice.invoiceCurrencyCode,\n },\n ],\n TaxAmount: [\n {\n _: item.taxAmount,\n currencyID: invoice.invoiceCurrencyCode,\n },\n ],\n // Conditional tax fields based on taxation type\n ...(item.taxPerUnitAmount !== undefined &&\n item.baseUnitMeasure !== undefined\n ? {\n // Fixed Rate Taxation\n PerUnitAmount: [\n {\n _: item.taxPerUnitAmount,\n currencyID: invoice.invoiceCurrencyCode,\n },\n ],\n BaseUnitMeasure: [\n {\n _: item.baseUnitMeasure,\n unitCode: item.baseUnitMeasureCode || 'C62',\n },\n ],\n }\n : item.taxRate !== undefined\n ? {\n // Percentage Taxation\n Percent: [{ _: item.taxRate }],\n }\n : {}),\n TaxCategory: [\n {\n ID: [{ _: item.taxType }],\n TaxScheme: [\n {\n ID: [\n {\n _: 'OTH',\n schemeAgencyID: '6',\n schemeID: 'UN/ECE 5153',\n },\n ],\n },\n ],\n },\n ],\n },\n ],\n },\n ],\n })),\n\n // === TAX EXCHANGE RATE (mandatory where applicable) ===\n TaxExchangeRate: invoice.currencyExchangeRate\n ? [\n {\n SourceCurrencyCode: [\n {\n _: invoice.invoiceCurrencyCode,\n },\n ],\n TargetCurrencyCode: [\n {\n _: 'MYR',\n },\n ],\n CalculationRate: [\n {\n _: invoice.currencyExchangeRate,\n },\n ],\n },\n ]\n : undefined,\n }\n}\n\n/**\n * Generates the complete UBL document structure with namespace declarations\n */\nexport const generateCleanUBLDocument = (\n invoices: AllDocumentsV1_1[],\n): UBLDocument => {\n return {\n _D: 'urn:oasis:names:specification:ubl:schema:xsd:Invoice-2',\n _A: 'urn:oasis:names:specification:ubl:schema:xsd:CommonAggregateComponents-2',\n _B: 'urn:oasis:names:specification:ubl:schema:xsd:CommonBasicComponents-2',\n Invoice: invoices.map(generateCleanInvoiceObject),\n }\n}\n\n/**\n * Step 1: Transform the document for hashing or transmission\n * Removes UBLExtensions and Signature, and minifies the JSON\n * Returns the minified, cleaned JSON string\n */\nexport const transformDocumentForHashing = (\n invoices: AllDocumentsV1_1[],\n): string => {\n // Generate clean UBL document structure\n const cleanDocument = generateCleanUBLDocument(invoices)\n\n // Deep clone to avoid mutating input\n const documentForTransform = JSON.parse(JSON.stringify(cleanDocument))\n if (\n documentForTransform.Invoice &&\n Array.isArray(documentForTransform.Invoice)\n ) {\n documentForTransform.Invoice.forEach((invoice: InvoiceSubmission) => {\n delete invoice.UBLExtensions\n delete invoice.Signature\n })\n }\n\n // Return minified JSON string\n return JSON.stringify(documentForTransform)\n}\n\n/**\n * Step 2: Calculate Document Digest\n * FIXED: Remove UBLExtensions and Signature before hashing (DS322)\n * Based on working implementation pattern\n */\nexport const calculateDocumentDigest = (\n invoices: AllDocumentsV1_1[],\n): string => {\n // Use the transformation function to get the minified, cleaned JSON string\n const documentString = transformDocumentForHashing(invoices)\n\n // Calculate SHA-256 hash\n const hash = crypto.createHash('sha256')\n hash.update(documentString, 'utf8')\n\n // Return as Base64 (DocDigest)\n return hash.digest('base64')\n}\n\n/**\n * Step 4: Calculate Certificate Digest\n * Enhanced to handle certificate content properly\n */\nexport const calculateCertificateDigest = (certificatePem: string): string => {\n // Extract certificate content (Base64 without PEM headers/footers)\n const certificateContent = certificatePem\n .replace(/-----BEGIN CERTIFICATE-----/g, '')\n .replace(/-----END CERTIFICATE-----/g, '')\n .replace(/\\s+/g, '') // Remove all whitespace\n\n // Convert Base64 to binary\n const certificateBinary = Buffer.from(certificateContent, 'base64')\n\n // Calculate SHA-256 hash of binary content\n const hash = crypto.createHash('sha256')\n hash.update(certificateBinary)\n\n // Return as Base64\n return hash.digest('base64')\n}\n\n/**\n * Enhanced certificate info extraction with better error handling\n * FIXED: Normalize issuer name format to match MyInvois expectations (DS326)\n */\nexport const extractCertificateInfo = (\n certificatePem: string,\n): {\n issuerName: string\n serialNumber: string\n subjectName: string\n} => {\n try {\n const cert = new X509Certificate(certificatePem)\n\n // Extract serial number and convert to decimal string\n const serialNumberHex = cert.serialNumber\n\n // Keep the DN formatting exactly as it appears in the certificate to avoid\n // mismatches when the signing service validates the X509IssuerName fields.\n // We only replace raw new-lines with \", \" so that the DN remains a single-line\n // string while preserving all other whitespace and ordering.\n const normalizeDistinguishedName = (dn: string): string => {\n // Node returns issuer DN in reverse RDN order (C, O, ... , CN).\n // The MyInvois validator expects forward order (CN first).\n // 1. Break DN into components separated by newline or commas.\n // 2. Reverse to get CN → ... → C ordering.\n // 3. Join with \", \" and ensure single '=' spacing.\n const parts = dn\n .split(/\\r?\\n|,\\s*/)\n .map(part => part.trim())\n .filter(part => part.length > 0)\n .reverse()\n return parts.join(', ').replace(/\\s*=\\s*/g, '=')\n }\n\n // Enhanced serial number formatting\n const formatSerialNumber = (serialHex: string): string => {\n // Convert hex to decimal and ensure it's a string\n const decimal = BigInt('0x' + serialHex).toString()\n return decimal\n }\n\n return {\n issuerName: normalizeDistinguishedName(cert.issuer),\n serialNumber: formatSerialNumber(serialNumberHex),\n subjectName: normalizeDistinguishedName(cert.subject),\n }\n } catch (error: unknown) {\n throw new Error(\n `Failed to extract certificate info: ${error instanceof Error ? error.message : String(error)}`,\n )\n }\n}\n\n/**\n * Step 5: Create SignedProperties with enhanced structure\n * FIXED: Simplified structure to match MyInvois expectations (DS320)\n * Following MyInvois JSON signature specification exactly\n */\nexport const createSignedProperties = (\n certificateDigest: string,\n signingTime: string,\n issuerName: string,\n serialNumber: string,\n): SignedPropertiesObject => {\n return {\n SignedProperties: [\n {\n Id: 'id-xades-signed-props',\n SignedSignatureProperties: [\n {\n SigningTime: [{ _: signingTime }],\n SigningCertificate: [\n {\n Cert: [\n {\n CertDigest: [\n {\n DigestMethod: [\n {\n _: '',\n Algorithm:\n 'http://www.w3.org/2001/04/xmlenc#sha256',\n },\n ],\n DigestValue: [{ _: certificateDigest }],\n },\n ],\n IssuerSerial: [\n {\n X509IssuerName: [{ _: issuerName }],\n X509SerialNumber: [{ _: serialNumber }],\n },\n ],\n },\n ],\n },\n ],\n },\n ],\n },\n ],\n }\n}\n\n/**\n * Step 6: Calculate SignedProperties Digest\n * Calculates the digest over the correct structure for validator compliance.\n */\nexport const calculateSignedPropertiesDigest = (\n signedProperties: SignedPropertiesObject,\n useTargetWrapper: boolean = true, // now default to wrapper for compliance\n): string => {\n let signedPropertiesString: string\n if (useTargetWrapper) {\n const digestObj: {\n Target: string\n SignedProperties: SignedPropertiesObject['SignedProperties']\n } = {\n Target: 'signature',\n SignedProperties: signedProperties.SignedProperties,\n }\n signedPropertiesString = JSON.stringify(digestObj)\n } else {\n signedPropertiesString = JSON.stringify(signedProperties.SignedProperties)\n }\n const hash = crypto.createHash('sha256')\n hash.update(signedPropertiesString, 'utf8')\n return hash.digest('base64')\n}\n\n/**\n * Step 3: Create SignedInfo and calculate signature\n * Enhanced with better structure and signature generation\n */\nexport const createSignedInfoAndSign = (\n docDigest: string,\n propsDigest: string,\n privateKeyPem: string,\n): { signedInfo: SignedInfoObject; signatureValue: string } => {\n // Create SignedInfo structure following specification exactly\n const signedInfo: SignedInfoObject = {\n CanonicalizationMethod: [\n {\n _: '',\n Algorithm: 'http://www.w3.org/2006/12/xml-c14n11',\n },\n ],\n SignatureMethod: [\n {\n _: '',\n Algorithm: 'http://www.w3.org/2001/04/xmldsig-more#rsa-sha256',\n },\n ],\n Reference: [\n {\n Id: 'id-doc-signed-data',\n Type: '',\n URI: '',\n DigestMethod: [\n {\n _: '',\n Algorithm: 'http://www.w3.org/2001/04/xmlenc#sha256',\n },\n ],\n DigestValue: [{ _: docDigest }],\n },\n {\n Id: 'id-xades-signed-props',\n Type: 'http://uri.etsi.org/01903/v1.3.2#SignedProperties',\n URI: '#id-xades-signed-props',\n DigestMethod: [\n {\n _: '',\n Algorithm: 'http://www.w3.org/2001/04/xmlenc#sha256',\n },\n ],\n DigestValue: [{ _: propsDigest }],\n },\n ],\n }\n\n // Serialize the SignedInfo exactly as it will be embedded (no reordering)\n const signedInfoRaw = JSON.stringify(signedInfo)\n\n try {\n const signer = crypto.createSign('RSA-SHA256')\n signer.update(signedInfoRaw, 'utf8')\n const signatureValue = signer.sign(privateKeyPem, 'base64')\n\n // Reuse the original object so ordering is preserved\n return { signedInfo, signatureValue }\n } catch (error) {\n throw new Error(\n `Signature generation failed: ${error instanceof Error ? error.message : String(error)}`,\n )\n }\n}\n\n/**\n * Signs the minified document string using the provided private key PEM\n * Returns the signature as a base64 string\n */\nexport const signDocumentString = (\n documentString: string,\n privateKeyPem: string,\n): string => {\n const signature = crypto.sign('sha256', Buffer.from(documentString, 'utf8'), {\n key: privateKeyPem,\n padding: crypto.constants.RSA_PKCS1_PADDING,\n })\n return signature.toString('base64')\n}\n\n/**\n * Complete document generation with signatures\n * Follows the complete MyInvois JSON signature creation process (Steps 1-7)\n */\nexport const generateCompleteDocument = (\n invoices: AllDocumentsV1_1[],\n signingCredentials: SigningCredentials,\n): CompleteInvoice => {\n try {\n // Step 1: Generate clean document (done in calculateDocumentDigest)\n // Step 2: Calculate document digest\n const docDigest = calculateDocumentDigest(invoices)\n\n // Get the minified, cleaned JSON string for signing\n const documentString = transformDocumentForHashing(invoices)\n\n // Step 3: Sign the minified document string (not the digest)\n const docSignature = signDocumentString(\n documentString,\n signingCredentials.privateKeyPem,\n )\n\n // Generate signing time in proper ISO format\n const signingTime = new Date().toISOString()\n\n // Extract certificate information (enhanced)\n const certInfo = extractCertificateInfo(signingCredentials.certificatePem)\n\n // Step 4: Calculate certificate digest\n const certificateDigest = calculateCertificateDigest(\n signingCredentials.certificatePem,\n )\n\n // Step 5: Create SignedProperties using extracted cert info\n const signedProperties = createSignedProperties(\n certificateDigest,\n signingTime,\n certInfo.issuerName,\n certInfo.serialNumber,\n )\n\n // Step 6: Calculate SignedProperties digest (using Target wrapper for compliance)\n const propsDigest = calculateSignedPropertiesDigest(signedProperties, true)\n\n // Step 3 (new): Create SignedInfo structure (but do not sign it, just include for type compliance)\n const { signedInfo } = createSignedInfoAndSign(\n docDigest,\n propsDigest,\n signingCredentials.privateKeyPem,\n )\n\n // Extract certificate content (Base64 without PEM headers)\n const certificate = signingCredentials.certificatePem\n .replace(/-----BEGIN CERTIFICATE-----/g, '')\n .replace(/-----END CERTIFICATE-----/g, '')\n .replace(/\\s+/g, '')\n\n // Step 7: Create final signed document\n const signedInvoices = invoices.map(invoice => {\n const cleanInvoice = generateCleanInvoiceObject(invoice)\n\n return {\n ...cleanInvoice,\n\n // Add UBLExtensions with complete signature structure\n UBLExtensions: [\n {\n UBLExtension: [\n {\n ExtensionURI: [\n {\n _: 'urn:oasis:names:specification:ubl:dsig:enveloped:xades',\n },\n ],\n ExtensionContent: [\n {\n UBLDocumentSignatures: [\n {\n SignatureInformation: [\n {\n ID: [\n {\n _: 'urn:oasis:names:specification:ubl:signature:1',\n },\n ],\n ReferencedSignatureID: [\n {\n _: 'urn:oasis:names:specification:ubl:signature:Invoice',\n },\n ],\n Signature: [\n {\n Id: 'signature',\n Object: [\n {\n QualifyingProperties: [\n {\n Target: 'signature',\n SignedProperties:\n signedProperties.SignedProperties,\n },\n ],\n },\n ],\n KeyInfo: [\n {\n X509Data: [\n {\n X509Certificate: [{ _: certificate }],\n X509SubjectName: [\n { _: certInfo.subjectName },\n ],\n X509IssuerSerial: [\n {\n X509IssuerName: [\n {\n _: certInfo.issuerName,\n },\n ],\n X509SerialNumber: [\n {\n _: certInfo.serialNumber,\n },\n ],\n },\n ],\n },\n ],\n },\n ],\n SignatureValue: [{ _: docSignature }],\n SignedInfo: [signedInfo],\n },\n ],\n },\n ],\n },\n ],\n },\n ],\n },\n ],\n },\n ],\n\n // Add simple Signature reference\n Signature: [\n {\n ID: [\n {\n _: 'urn:oasis:names:specification:ubl:signature:Invoice',\n },\n ],\n SignatureMethod: [\n {\n _: 'urn:oasis:names:specification:ubl:dsig:enveloped:xades',\n },\n ],\n },\n ],\n }\n })\n\n return {\n _D: 'urn:oasis:names:specification:ubl:schema:xsd:Invoice-2',\n _A: 'urn:oasis:names:specification:ubl:schema:xsd:CommonAggregateComponents-2',\n _B: 'urn:oasis:names:specification:ubl:schema:xsd:CommonBasicComponents-2',\n Invoice: signedInvoices,\n }\n } catch (error) {\n throw new Error(\n `Document generation failed: ${error instanceof Error ? error.message : String(error)}`,\n )\n }\n}\n\n/**\n * Creates a line item with percentage-based taxation (e.g., SST, GST)\n */\nexport const createPercentageTaxLineItem = (params: {\n itemClassificationCode: ClassificationCode\n itemDescription: string\n unitPrice: number\n quantity?: number\n taxType: TaxTypeCode\n taxRate: number\n totalTaxableAmountPerLine?: number\n}): InvoiceLineItem => {\n const quantity = params.quantity || 1\n const totalTaxableAmount =\n params.totalTaxableAmountPerLine || params.unitPrice * quantity\n const taxAmount = (totalTaxableAmount * params.taxRate) / 100\n\n return {\n itemClassificationCode: params.itemClassificationCode,\n itemDescription: params.itemDescription,\n unitPrice: params.unitPrice,\n taxType: params.taxType,\n taxRate: params.taxRate,\n taxAmount: Math.round(taxAmount * 100) / 100, // Round to 2 decimal places\n totalTaxableAmountPerLine: totalTaxableAmount,\n totalAmountPerLine: totalTaxableAmount + taxAmount,\n }\n}\n\n/**\n * Creates a line item with fixed rate taxation (e.g., Tourism Tax)\n */\nexport const createFixedRateTaxLineItem = (params: {\n itemClassificationCode: ClassificationCode\n itemDescription: string\n unitPrice: number\n quantity?: number\n taxType: TaxTypeCode\n taxPerUnitAmount: number\n baseUnitMeasure: number\n baseUnitMeasureCode: UnitTypeCode\n totalTaxableAmountPerLine?: number\n}): InvoiceLineItem => {\n const quantity = params.quantity || 1\n const totalTaxableAmount =\n params.totalTaxableAmountPerLine || params.unitPrice * quantity\n const taxAmount = params.taxPerUnitAmount * params.baseUnitMeasure\n\n return {\n itemClassificationCode: params.itemClassificationCode,\n itemDescription: params.itemDescription,\n unitPrice: params.unitPrice,\n taxType: params.taxType,\n taxPerUnitAmount: params.taxPerUnitAmount,\n baseUnitMeasure: params.baseUnitMeasure,\n baseUnitMeasureCode: params.baseUnitMeasureCode,\n taxAmount: Math.round(taxAmount * 100) / 100, // Round to 2 decimal places\n totalTaxableAmountPerLine: totalTaxableAmount,\n totalAmountPerLine: totalTaxableAmount + taxAmount,\n }\n}\n\n/**\n * Calculates invoice totals from line items\n */\nexport const calculateInvoiceTotals = (\n lineItems: InvoiceLineItem[],\n): {\n legalMonetaryTotal: {\n taxExclusiveAmount: number\n taxInclusiveAmount: number\n payableAmount: number\n }\n taxTotal: {\n taxAmount: number\n }\n} => {\n const taxExclusiveAmount = lineItems.reduce(\n (sum, item) => sum + item.totalTaxableAmountPerLine,\n 0,\n )\n const totalTaxAmount = lineItems.reduce(\n (sum, item) => sum + item.taxAmount,\n 0,\n )\n const taxInclusiveAmount = taxExclusiveAmount + totalTaxAmount\n\n return {\n legalMonetaryTotal: {\n taxExclusiveAmount: Math.round(taxExclusiveAmount * 100) / 100,\n taxInclusiveAmount: Math.round(taxInclusiveAmount * 100) / 100,\n payableAmount: Math.round(taxInclusiveAmount * 100) / 100,\n },\n taxTotal: {\n taxAmount: Math.round(totalTaxAmount * 100) / 100,\n },\n }\n}\n"],"mappings":"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"}
@@ -253,18 +253,26 @@ const generateCleanUBLDocument = (invoices) => {
253
253
  };
254
254
  };
255
255
  /**
256
- * Step 2: Calculate Document Digest
257
- * FIXED: Remove UBLExtensions and Signature before hashing (DS322)
258
- * Based on working implementation pattern
256
+ * Step 1: Transform the document for hashing or transmission
257
+ * Removes UBLExtensions and Signature, and minifies the JSON
258
+ * Returns the minified, cleaned JSON string
259
259
  */
260
- const calculateDocumentDigest = (invoices) => {
260
+ const transformDocumentForHashing = (invoices) => {
261
261
  const cleanDocument = generateCleanUBLDocument(invoices);
262
- const documentForHashing = JSON.parse(JSON.stringify(cleanDocument));
263
- if (documentForHashing.Invoice && Array.isArray(documentForHashing.Invoice)) documentForHashing.Invoice.forEach((invoice) => {
262
+ const documentForTransform = JSON.parse(JSON.stringify(cleanDocument));
263
+ if (documentForTransform.Invoice && Array.isArray(documentForTransform.Invoice)) documentForTransform.Invoice.forEach((invoice) => {
264
264
  delete invoice.UBLExtensions;
265
265
  delete invoice.Signature;
266
266
  });
267
- const documentString = JSON.stringify(documentForHashing);
267
+ return JSON.stringify(documentForTransform);
268
+ };
269
+ /**
270
+ * Step 2: Calculate Document Digest
271
+ * FIXED: Remove UBLExtensions and Signature before hashing (DS322)
272
+ * Based on working implementation pattern
273
+ */
274
+ const calculateDocumentDigest = (invoices) => {
275
+ const documentString = transformDocumentForHashing(invoices);
268
276
  const hash = crypto.createHash("sha256");
269
277
  hash.update(documentString, "utf8");
270
278
  return hash.digest("base64");
@@ -289,8 +297,8 @@ const extractCertificateInfo = (certificatePem) => {
289
297
  const cert = new X509Certificate(certificatePem);
290
298
  const serialNumberHex = cert.serialNumber;
291
299
  const normalizeDistinguishedName = (dn) => {
292
- const normalized = dn.split("\n").map((line) => line.trim()).filter((line) => line.length > 0).join(", ");
293
- return normalized.replace(/\s*=\s*/g, "=").replace(/,\s+/g, ", ").replace(/\r/g, "").replace(/\s{2,}/g, " ").trim();
300
+ const parts = dn.split(/\r?\n|,\s*/).map((part) => part.trim()).filter((part) => part.length > 0).reverse();
301
+ return parts.join(", ").replace(/\s*=\s*/g, "=");
294
302
  };
295
303
  const formatSerialNumber = (serialHex) => {
296
304
  const decimal = BigInt("0x" + serialHex).toString();
@@ -317,7 +325,10 @@ const createSignedProperties = (certificateDigest, signingTime, issuerName, seri
317
325
  SigningTime: [{ _: signingTime }],
318
326
  SigningCertificate: [{ Cert: [{
319
327
  CertDigest: [{
320
- DigestMethod: [{ Algorithm: "http://www.w3.org/2001/04/xmlenc#sha256" }],
328
+ DigestMethod: [{
329
+ _: "",
330
+ Algorithm: "http://www.w3.org/2001/04/xmlenc#sha256"
331
+ }],
321
332
  DigestValue: [{ _: certificateDigest }]
322
333
  }],
323
334
  IssuerSerial: [{
@@ -330,11 +341,17 @@ const createSignedProperties = (certificateDigest, signingTime, issuerName, seri
330
341
  };
331
342
  /**
332
343
  * Step 6: Calculate SignedProperties Digest
333
- * FIXED: Add Target wrapper and use direct JSON stringify (DS320)
334
- * Based on working implementation pattern
344
+ * Calculates the digest over the correct structure for validator compliance.
335
345
  */
336
- const calculateSignedPropertiesDigest = (signedProperties) => {
337
- const signedPropertiesString = canonicalizeJSON(signedProperties);
346
+ const calculateSignedPropertiesDigest = (signedProperties, useTargetWrapper = true) => {
347
+ let signedPropertiesString;
348
+ if (useTargetWrapper) {
349
+ const digestObj = {
350
+ Target: "signature",
351
+ SignedProperties: signedProperties.SignedProperties
352
+ };
353
+ signedPropertiesString = JSON.stringify(digestObj);
354
+ } else signedPropertiesString = JSON.stringify(signedProperties.SignedProperties);
338
355
  const hash = crypto.createHash("sha256");
339
356
  hash.update(signedPropertiesString, "utf8");
340
357
  return hash.digest("base64");
@@ -373,10 +390,10 @@ const createSignedInfoAndSign = (docDigest, propsDigest, privateKeyPem) => {
373
390
  DigestValue: [{ _: propsDigest }]
374
391
  }]
375
392
  };
376
- const signedInfoString = JSON.stringify(signedInfo);
393
+ const signedInfoRaw = JSON.stringify(signedInfo);
377
394
  try {
378
395
  const signer = crypto.createSign("RSA-SHA256");
379
- signer.update(signedInfoString, "utf8");
396
+ signer.update(signedInfoRaw, "utf8");
380
397
  const signatureValue = signer.sign(privateKeyPem, "base64");
381
398
  return {
382
399
  signedInfo,
@@ -387,18 +404,31 @@ const createSignedInfoAndSign = (docDigest, propsDigest, privateKeyPem) => {
387
404
  }
388
405
  };
389
406
  /**
407
+ * Signs the minified document string using the provided private key PEM
408
+ * Returns the signature as a base64 string
409
+ */
410
+ const signDocumentString = (documentString, privateKeyPem) => {
411
+ const signature = crypto.sign("sha256", Buffer.from(documentString, "utf8"), {
412
+ key: privateKeyPem,
413
+ padding: crypto.constants.RSA_PKCS1_PADDING
414
+ });
415
+ return signature.toString("base64");
416
+ };
417
+ /**
390
418
  * Complete document generation with signatures
391
419
  * Follows the complete MyInvois JSON signature creation process (Steps 1-7)
392
420
  */
393
421
  const generateCompleteDocument = (invoices, signingCredentials) => {
394
422
  try {
395
423
  const docDigest = calculateDocumentDigest(invoices);
424
+ const documentString = transformDocumentForHashing(invoices);
425
+ const docSignature = signDocumentString(documentString, signingCredentials.privateKeyPem);
396
426
  const signingTime = (/* @__PURE__ */ new Date()).toISOString();
397
427
  const certInfo = extractCertificateInfo(signingCredentials.certificatePem);
398
428
  const certificateDigest = calculateCertificateDigest(signingCredentials.certificatePem);
399
429
  const signedProperties = createSignedProperties(certificateDigest, signingTime, certInfo.issuerName, certInfo.serialNumber);
400
- const propsDigest = calculateSignedPropertiesDigest(signedProperties);
401
- const { signedInfo, signatureValue } = createSignedInfoAndSign(docDigest, propsDigest, signingCredentials.privateKeyPem);
430
+ const propsDigest = calculateSignedPropertiesDigest(signedProperties, true);
431
+ const { signedInfo } = createSignedInfoAndSign(docDigest, propsDigest, signingCredentials.privateKeyPem);
402
432
  const certificate = signingCredentials.certificatePem.replace(/-----BEGIN CERTIFICATE-----/g, "").replace(/-----END CERTIFICATE-----/g, "").replace(/\s+/g, "");
403
433
  const signedInvoices = invoices.map((invoice) => {
404
434
  const cleanInvoice = generateCleanInvoiceObject(invoice);
@@ -423,7 +453,7 @@ const generateCompleteDocument = (invoices, signingCredentials) => {
423
453
  X509SerialNumber: [{ _: certInfo.serialNumber }]
424
454
  }]
425
455
  }] }],
426
- SignatureValue: [{ _: signatureValue }],
456
+ SignatureValue: [{ _: docSignature }],
427
457
  SignedInfo: [signedInfo]
428
458
  }]
429
459
  }] }] }]
@@ -500,4 +530,4 @@ const calculateInvoiceTotals = (lineItems) => {
500
530
  };
501
531
 
502
532
  //#endregion
503
- export { calculateCertificateDigest, calculateDocumentDigest, calculateExpectedTaxAmount, calculateInvoiceTotals, calculateSignedPropertiesDigest, canonicalizeJSON, createFixedRateTaxLineItem, createPercentageTaxLineItem, createSignedInfoAndSign, createSignedProperties, extractCertificateInfo, generateCleanInvoiceObject, generateCleanUBLDocument, generateCompleteDocument, isFixedRateTax, isPercentageTax, sortObjectKeys };
533
+ export { calculateCertificateDigest, calculateDocumentDigest, calculateExpectedTaxAmount, calculateInvoiceTotals, calculateSignedPropertiesDigest, canonicalizeJSON, createFixedRateTaxLineItem, createPercentageTaxLineItem, createSignedInfoAndSign, createSignedProperties, extractCertificateInfo, generateCleanInvoiceObject, generateCleanUBLDocument, generateCompleteDocument, isFixedRateTax, isPercentageTax, signDocumentString, sortObjectKeys, transformDocumentForHashing };
@@ -1,4 +1,4 @@
1
- const require_document = require('./document-Dvonhaqt.cjs');
1
+ const require_document = require('./document-BWReWH1w.cjs');
2
2
 
3
3
  //#region src/api/documentSubmission.ts
4
4
  async function submitDocument(context, documents) {
@@ -89,9 +89,9 @@ async function getSubmissionStatus(context, submissionUid, pollInterval = 1e3, m
89
89
  return await getSubmissionStatus(context, submissionUid, pollInterval, maxRetries - 1);
90
90
  }
91
91
  return {
92
- status: "TimedOut",
92
+ status: data.overallStatus || "TimedOut",
93
93
  documentSummary: data.documentSummary,
94
- error: {
94
+ error: data.error ?? {
95
95
  code: "Timeout",
96
96
  message: "Submission timed out",
97
97
  target: "submission",
@@ -147,4 +147,4 @@ Object.defineProperty(exports, 'submitDocument', {
147
147
  return submitDocument;
148
148
  }
149
149
  });
150
- //# sourceMappingURL=documentSubmission-CWscnXdN.cjs.map
150
+ //# sourceMappingURL=documentSubmission-DQy30z4E.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"documentSubmission-DQy30z4E.cjs","names":["context: SubmissionContext","documents: AllDocumentsV1_1[]","context: Pick<SubmissionContext, 'fetch' | 'debug'>","submissionUid: string","pollInterval: number","maxRetries: number","documentUid: string","status: 'rejected' | 'cancelled'","reason: string"],"sources":["../src/api/documentSubmission.ts"],"sourcesContent":["import type {\n AllDocumentsV1_1,\n SubmissionResponse,\n SigningCredentials,\n SubmissionStatus,\n DocumentSummary,\n Fetch,\n StandardError,\n GetSubmissionResponse,\n} from '../types'\nimport { generateCompleteDocument } from '../utils/document'\n\ninterface SubmissionContext {\n fetch: Fetch\n debug: boolean\n signingCredentials: SigningCredentials\n}\n\nexport async function submitDocument(\n context: SubmissionContext,\n documents: AllDocumentsV1_1[],\n): Promise<{\n data: SubmissionResponse\n status: number\n}> {\n const { fetch, debug, signingCredentials } = context\n\n // 🔒 Hard enforcement of platform submission limits\n if (documents.length > 100) {\n throw new Error(\n 'Submission rejected: Cannot submit more than 100 documents at once',\n )\n }\n\n if (debug) {\n console.log(`📦 Preparing to submit ${documents.length} document(s)...`)\n }\n\n // For batch submission, each document must be signed and encoded separately\n // Build the submission payload according to MyInvois API format\n const crypto = await import('crypto')\n\n const submissionPayload = {\n documents: await Promise.all(\n documents.map(async doc => {\n // 1️⃣ Sign the single document (generateCompleteDocument expects an array)\n const signedDocument = generateCompleteDocument(\n [doc],\n signingCredentials,\n )\n\n // 2️⃣ Serialize\n const docJson = JSON.stringify(signedDocument)\n\n // 3️⃣ Hash\n const docHash = crypto\n .createHash('sha256')\n .update(docJson, 'utf8')\n .digest('hex')\n\n // 4️⃣ Base64 encode\n const docBase64 = Buffer.from(docJson, 'utf8').toString('base64')\n\n // 🚨 Enforce 300 KB per-document limit\n const rawSize = Buffer.byteLength(docBase64, 'base64')\n if (rawSize > 300 * 1024) {\n throw new Error(\n `Submission rejected: Document ${doc.eInvoiceCodeOrNumber} is ${rawSize} bytes – exceeds 300KB limit`,\n )\n }\n\n if (debug) {\n console.log('—'.repeat(60))\n console.log(`📄 Prepared document: ${doc.eInvoiceCodeOrNumber}`)\n console.log(` • JSON size : ${docJson.length} bytes`)\n console.log(` • Base64 size: ${docBase64.length} bytes`)\n }\n\n return {\n format: 'JSON',\n document: docBase64,\n documentHash: docHash,\n codeNumber: doc.eInvoiceCodeOrNumber,\n }\n }),\n ),\n }\n\n const payloadSize = Buffer.byteLength(JSON.stringify(submissionPayload))\n\n if (payloadSize > 5 * 1024 * 1024) {\n throw new Error(\n `Submission rejected: Payload is ${payloadSize} bytes – exceeds 5MB limit`,\n )\n }\n\n if (debug) {\n console.log('🚀 Submission payload structure:')\n console.log('- Format: JSON')\n console.log('- Documents count:', submissionPayload.documents.length)\n console.log('- Total payload size:', payloadSize, 'bytes')\n }\n\n // Submit to MyInvois API with proper headers\n const response = await fetch('/api/v1.0/documentsubmissions', {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n },\n body: JSON.stringify(submissionPayload),\n })\n\n const responseData = (await response.json()) as SubmissionResponse\n\n if (debug) {\n console.log(`📡 API Response status: ${response.status}`)\n\n if (responseData.rejectedDocuments?.length > 0) {\n responseData.rejectedDocuments.forEach((doc, index) => {\n console.log(` Document ${index + 1}:`, doc.invoiceCodeNumber)\n if (doc.error) {\n console.log(` Error:`, doc.error.message)\n if (doc.error.details) {\n doc.error.details.forEach((detail, detailIndex) => {\n console.log(` Detail ${detailIndex + 1}:`, detail.message)\n })\n }\n }\n })\n }\n }\n\n const data = responseData as SubmissionResponse\n\n if (debug) {\n if (response.status !== 202) {\n console.error('❌ Submission failed with status:', response.status)\n console.error('❌ Response data:', JSON.stringify(data, null, 2))\n } else {\n console.log('✅ Submission successful!')\n console.log(`📋 Submission UID: ${data.submissionUid}`)\n console.log(\n `✅ Accepted documents: ${data.acceptedDocuments?.length || 0}`,\n )\n console.log(\n `❌ Rejected documents: ${data.rejectedDocuments?.length || 0}`,\n )\n }\n }\n\n return {\n data,\n status: response.status,\n }\n}\n\nexport async function getSubmissionStatus(\n context: Pick<SubmissionContext, 'fetch' | 'debug'>,\n submissionUid: string,\n pollInterval: number = 1000,\n maxRetries: number = 10,\n): Promise<{\n status: SubmissionStatus\n documentSummary?: DocumentSummary[]\n error?: {\n code: string\n message: string | null\n target: string\n details: {\n code: string\n message: string\n target: string\n }[]\n }\n}> {\n const { fetch, debug } = context\n\n try {\n const response = await fetch(\n `/api/v1.0/documentsubmissions/${submissionUid}`,\n )\n\n const data = (await response.json()) as GetSubmissionResponse\n\n if (debug) {\n console.log('Submission:', data)\n if (data.error) {\n console.log('Submission error details:', data.error.details)\n }\n }\n\n // If we have a successful response and status is completed, return success\n if (data.overallStatus === 'Valid') {\n return {\n status: data.overallStatus,\n documentSummary: data.documentSummary,\n }\n }\n if (data.overallStatus === 'Invalid') {\n return {\n status: 'Invalid',\n documentSummary: data.documentSummary,\n }\n }\n\n // If we have retries left, continue polling for any non-completed status or errors\n if (maxRetries > 0) {\n await new Promise(resolve => setTimeout(resolve, pollInterval))\n return await getSubmissionStatus(\n context,\n submissionUid,\n pollInterval,\n maxRetries - 1,\n )\n }\n\n return {\n status: data.overallStatus || 'TimedOut',\n documentSummary: data.documentSummary,\n error:\n data.error ??\n ({\n code: 'Timeout',\n message: 'Submission timed out',\n target: 'submission',\n details: [],\n } satisfies StandardError),\n }\n } catch (error) {\n // Handle any request errors by retrying if we have retries left\n if (maxRetries > 0) {\n if (debug) {\n console.log('Request error, retrying...', error)\n }\n await new Promise(resolve => setTimeout(resolve, pollInterval))\n return await getSubmissionStatus(\n context,\n submissionUid,\n pollInterval,\n maxRetries - 1,\n )\n }\n\n // No retries left - return timeout\n return {\n status: 'TimedOut',\n documentSummary: [],\n error: {\n code: 'Timeout',\n message: 'Submission timed out after request errors',\n target: 'submission',\n details: [],\n },\n }\n }\n}\n\nexport async function performDocumentAction(\n documentUid: string,\n status: 'rejected' | 'cancelled',\n reason: string,\n): Promise<{\n uuid: string\n status: string\n error: StandardError\n}> {\n const response = await fetch(\n `/api/v1.0/documents/state/${documentUid}/state`,\n {\n method: 'POST',\n body: JSON.stringify({\n status,\n reason,\n }),\n },\n )\n\n const data = (await response.json()) as {\n uuid: string\n status: string\n error: StandardError\n }\n\n return data\n}\n"],"mappings":"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"}
@@ -1,4 +1,4 @@
1
- import { generateCompleteDocument } from "./document-BRxLmvIU.js";
1
+ import { generateCompleteDocument } from "./document-BbggZ72T.js";
2
2
 
3
3
  //#region src/api/documentSubmission.ts
4
4
  async function submitDocument(context, documents) {
@@ -89,9 +89,9 @@ async function getSubmissionStatus(context, submissionUid, pollInterval = 1e3, m
89
89
  return await getSubmissionStatus(context, submissionUid, pollInterval, maxRetries - 1);
90
90
  }
91
91
  return {
92
- status: "TimedOut",
92
+ status: data.overallStatus || "TimedOut",
93
93
  documentSummary: data.documentSummary,
94
- error: {
94
+ error: data.error ?? {
95
95
  code: "Timeout",
96
96
  message: "Submission timed out",
97
97
  target: "submission",
@@ -832,6 +832,8 @@ interface InvoiceSubmission {
832
832
  Cert: {
833
833
  CertDigest: {
834
834
  DigestMethod: {
835
+ /** Optional text content field (required by MyInvois JSON signature spec). Typically blank string */
836
+ _?: string;
835
837
  Algorithm: string;
836
838
  }[];
837
839
  DigestValue: {
@@ -937,6 +939,8 @@ interface SignedPropertiesObject {
937
939
  CertDigest: {
938
940
  /** Digest method algorithm */
939
941
  DigestMethod: {
942
+ /** Optional text content field (required by MyInvois JSON signature spec). Typically blank string */
943
+ _?: string;
940
944
  Algorithm: string;
941
945
  }[];
942
946
  /** Base64 encoded digest value */
@@ -1020,6 +1024,8 @@ interface SignedInvoiceSubmission extends InvoiceSubmission {
1020
1024
  Cert: {
1021
1025
  CertDigest: {
1022
1026
  DigestMethod: {
1027
+ /** Optional text content field (required by MyInvois JSON signature spec). Typically blank string */
1028
+ _?: string;
1023
1029
  Algorithm: string;
1024
1030
  }[];
1025
1031
  DigestValue: {
@@ -1128,4 +1134,4 @@ interface DocumentTypeVersionResponse {
1128
1134
  }
1129
1135
  //#endregion
1130
1136
  export { Address, AllDocumentsV1_1, Buyer, CompleteInvoice, CreditNoteV1_1, DebitNoteV1_1, DocumentStatus, DocumentSummary, DocumentTypeResponse, DocumentTypeVersion, DocumentTypeVersionResponse, DocumentTypesResponse, DocumentValidationResult, DocumentValidationStepResult, FinalDocumentData, GetSubmissionResponse, InvoiceLineItem, InvoiceSubmission, InvoiceV1_1, LegalMonetaryTotal, RefundNoteV1_1, RegistrationType, ResponseDocument, SelfBilledCreditNoteV1_1, SelfBilledInvoiceV1_1, SelfBilledRefundNoteV1_1, SignedInfoObject, SignedInvoiceSubmission, SignedPropertiesData, SignedPropertiesObject, SigningCredentials, StandardError, SubmissionResponse, SubmissionStatus, Supplier, TaxCategory, TaxSubtotal, TaxTotal, UBLDocument, WorkflowParameter };
1131
- //# sourceMappingURL=documents-Dmn5FIRZ.d.cts.map
1137
+ //# sourceMappingURL=documents-BWpJ0B3C.d.cts.map
@@ -832,6 +832,8 @@ interface InvoiceSubmission {
832
832
  Cert: {
833
833
  CertDigest: {
834
834
  DigestMethod: {
835
+ /** Optional text content field (required by MyInvois JSON signature spec). Typically blank string */
836
+ _?: string;
835
837
  Algorithm: string;
836
838
  }[];
837
839
  DigestValue: {
@@ -937,6 +939,8 @@ interface SignedPropertiesObject {
937
939
  CertDigest: {
938
940
  /** Digest method algorithm */
939
941
  DigestMethod: {
942
+ /** Optional text content field (required by MyInvois JSON signature spec). Typically blank string */
943
+ _?: string;
940
944
  Algorithm: string;
941
945
  }[];
942
946
  /** Base64 encoded digest value */
@@ -1020,6 +1024,8 @@ interface SignedInvoiceSubmission extends InvoiceSubmission {
1020
1024
  Cert: {
1021
1025
  CertDigest: {
1022
1026
  DigestMethod: {
1027
+ /** Optional text content field (required by MyInvois JSON signature spec). Typically blank string */
1028
+ _?: string;
1023
1029
  Algorithm: string;
1024
1030
  }[];
1025
1031
  DigestValue: {
package/dist/index.cjs CHANGED
@@ -1,7 +1,7 @@
1
1
  const require_documentManagement = require('./documentManagement-DQ7JEcBq.cjs');
2
2
  require('./formatIdValue-i67o4kyD.cjs');
3
- require('./document-Dvonhaqt.cjs');
4
- const require_documentSubmission = require('./documentSubmission-CWscnXdN.cjs');
3
+ require('./document-BWReWH1w.cjs');
4
+ const require_documentSubmission = require('./documentSubmission-DQy30z4E.cjs');
5
5
  const require_documentTypeManagement = require('./documentTypeManagement-D_-LiQVg.cjs');
6
6
  const require_notificationManagement = require('./notificationManagement-DLBDn77E.cjs');
7
7
  const require_platformLogin = require('./platformLogin-Ch6hFKoU.cjs');
package/dist/index.d.ts CHANGED
@@ -51,11 +51,11 @@ import "./YX-F34sJ7Ik.js";
51
51
  import "./ZX-CDQOfsHh.js";
52
52
  import "./XX-DOA-10JW.js";
53
53
  import { UnitType, UnitTypeCode } from "./unit-types-VgYXIwTT.js";
54
- import { Address, AllDocumentsV1_1, Buyer, CompleteInvoice, CreditNoteV1_1, DebitNoteV1_1, DocumentStatus, DocumentSummary, DocumentTypeResponse, DocumentTypeVersion, DocumentTypeVersionResponse, DocumentTypesResponse, DocumentValidationResult, DocumentValidationStepResult, FinalDocumentData, GetSubmissionResponse, InvoiceLineItem, InvoiceSubmission, InvoiceV1_1, LegalMonetaryTotal, RefundNoteV1_1, RegistrationType, ResponseDocument, SelfBilledCreditNoteV1_1, SelfBilledInvoiceV1_1, SelfBilledRefundNoteV1_1, SignedInfoObject, SignedInvoiceSubmission, SignedPropertiesData, SignedPropertiesObject, SigningCredentials, StandardError, SubmissionResponse, SubmissionStatus, Supplier, TaxCategory, TaxSubtotal, TaxTotal, UBLDocument, WorkflowParameter } from "./documents-AlZIGjw2.js";
54
+ import { Address, AllDocumentsV1_1, Buyer, CompleteInvoice, CreditNoteV1_1, DebitNoteV1_1, DocumentStatus, DocumentSummary, DocumentTypeResponse, DocumentTypeVersion, DocumentTypeVersionResponse, DocumentTypesResponse, DocumentValidationResult, DocumentValidationStepResult, FinalDocumentData, GetSubmissionResponse, InvoiceLineItem, InvoiceSubmission, InvoiceV1_1, LegalMonetaryTotal, RefundNoteV1_1, RegistrationType, ResponseDocument, SelfBilledCreditNoteV1_1, SelfBilledInvoiceV1_1, SelfBilledRefundNoteV1_1, SignedInfoObject, SignedInvoiceSubmission, SignedPropertiesData, SignedPropertiesObject, SigningCredentials, StandardError, SubmissionResponse, SubmissionStatus, Supplier, TaxCategory, TaxSubtotal, TaxTotal, UBLDocument, WorkflowParameter } from "./documents-DRQ6_toX.js";
55
55
  import { PaymentMode, PaymentModeCode, PaymentModeCodeEnum } from "./payment-modes-g3DzLmWb.js";
56
56
  import { CanonicalizationMethod, Cert, CertDigest, DigestMethod, IssuerDigitalSignature, IssuerSerial, KeyInfo, QualifyingProperties, Reference, SignatureMethod, SignedInfo, SignedProperties, SignedSignatureProperties, SigningCertificate, Transform, X509Data } from "./signatures-CerHUrj3.js";
57
57
  import { Notification, NotificationDeliveryAttempt, NotificationMetadata, NotificationResponse, NotificationSearchParams, NotificationStatus, NotificationStatusEnum, NotificationType, NotificationTypeEnum } from "./notifications-sFhgh3rJ.js";
58
- import { TaxpayerQRCodeResponse, TinSearchParams, TinSearchResponse } from "./taxpayer-DZmqFn6t.js";
58
+ import { TaxpayerQRCodeResponse, TinSearchParams, TinSearchResponse } from "./taxpayer-DtyHXKwz.js";
59
59
  import { Fetch } from "./utils-C4FoVKLq.js";
60
60
  import { ClientCredentials, TokenResponse } from "./index-CygwSf0x.js";
61
61
 
package/dist/index.js CHANGED
@@ -1,7 +1,7 @@
1
1
  import { getDocument, getDocumentDetails, searchDocuments } from "./documentManagement-CIQPkmyb.js";
2
2
  import "./formatIdValue-qTxJqj9o.js";
3
- import "./document-BRxLmvIU.js";
4
- import { getSubmissionStatus, performDocumentAction, submitDocument } from "./documentSubmission-D9hVtze6.js";
3
+ import "./document-BbggZ72T.js";
4
+ import { getSubmissionStatus, performDocumentAction, submitDocument } from "./documentSubmission-DRbswZAD.js";
5
5
  import { getDocumentType, getDocumentTypeVersion, getDocumentTypes } from "./documentTypeManagement-cBtVCOY3.js";
6
6
  import { getNotifications } from "./notificationManagement-n4Z5e-My.js";
7
7
  import { platformLogin } from "./platformLogin-CqI9OLYP.js";
package/dist/index18.cjs CHANGED
@@ -1,6 +1,6 @@
1
1
  require('./formatIdValue-i67o4kyD.cjs');
2
- require('./document-Dvonhaqt.cjs');
3
- const require_documentSubmission = require('./documentSubmission-CWscnXdN.cjs');
2
+ require('./document-BWReWH1w.cjs');
3
+ const require_documentSubmission = require('./documentSubmission-DQy30z4E.cjs');
4
4
 
5
5
  exports.getSubmissionStatus = require_documentSubmission.getSubmissionStatus;
6
6
  exports.performDocumentAction = require_documentSubmission.performDocumentAction;
package/dist/index26.cjs CHANGED
@@ -1,5 +1,5 @@
1
1
  require('./formatIdValue-i67o4kyD.cjs');
2
- const require_document = require('./document-Dvonhaqt.cjs');
2
+ const require_document = require('./document-BWReWH1w.cjs');
3
3
 
4
4
  exports.calculateCertificateDigest = require_document.calculateCertificateDigest;
5
5
  exports.calculateDocumentDigest = require_document.calculateDocumentDigest;
@@ -17,4 +17,6 @@ exports.generateCleanUBLDocument = require_document.generateCleanUBLDocument;
17
17
  exports.generateCompleteDocument = require_document.generateCompleteDocument;
18
18
  exports.isFixedRateTax = require_document.isFixedRateTax;
19
19
  exports.isPercentageTax = require_document.isPercentageTax;
20
- exports.sortObjectKeys = require_document.sortObjectKeys;
20
+ exports.signDocumentString = require_document.signDocumentString;
21
+ exports.sortObjectKeys = require_document.sortObjectKeys;
22
+ exports.transformDocumentForHashing = require_document.transformDocumentForHashing;
package/dist/index29.cjs CHANGED
@@ -1,6 +1,6 @@
1
1
  const require_chunk = require('./chunk-CUT6urMc.cjs');
2
2
  require('./formatIdValue-i67o4kyD.cjs');
3
- const require_document = require('./document-Dvonhaqt.cjs');
3
+ const require_document = require('./document-BWReWH1w.cjs');
4
4
  const crypto = require_chunk.__toESM(require("crypto"));
5
5
 
6
6
  //#region src/utils/signature-diagnostics.ts
@@ -1 +1 @@
1
- {"version":3,"file":"index71.cts","names":[],"sources":["../src/utils/document.d.ts"],"sourcesContent":["import { InvoiceSubmission, AllDocumentsV1_1, SigningCredentials, SignedPropertiesObject, UBLDocument, CompleteInvoice, SignedInfoObject, UnitTypeCode, InvoiceLineItem } from '../types';\nimport type { ClassificationCode } from '../types';\nimport type { TaxTypeCode } from '../types';\n/**\n * MyInvois v1.1 Document Generation and Signing Utilities\n * Strictly follows: https://sdk.myinvois.hasil.gov.my/documents/invoice-v1-1\n * JSON Signature Guide: https://sdk.myinvois.hasil.gov.my/signature-creation-json/\n */\n/**\n * Determines if a line item uses fixed rate taxation\n */\nexport declare const isFixedRateTax: (item: InvoiceLineItem) => boolean;\n/**\n * Determines if a line item uses percentage taxation\n */\nexport declare const isPercentageTax: (item: InvoiceLineItem) => boolean;\n/**\n * Calculates expected tax amount for a line item based on its tax type\n */\nexport declare const calculateExpectedTaxAmount: (item: InvoiceLineItem) => number;\n/**\n * Helper function to recursively sort object keys for JSON canonicalization\n */\nexport declare function sortObjectKeys(obj: unknown): unknown;\n/**\n * Enhanced canonicalization following MyInvois specification exactly\n * Key changes: ensure consistent ordering and formatting\n */\nexport declare const canonicalizeJSON: (obj: unknown) => string;\n/**\n * Generates a clean invoice object following MyInvois v1.1 specification exactly\n * This is the base invoice structure WITHOUT signature elements (for hash calculation)\n *\n * Key requirements from working documents:\n * - All mandatory fields must be present\n * - Many optional fields must be present even if empty\n * - Specific field ordering and structure\n * - Correct listID values (e.g., \"3166-1\" not \"ISO3166-1\")\n */\nexport declare const generateCleanInvoiceObject: (invoice: AllDocumentsV1_1) => InvoiceSubmission;\n/**\n * Generates the complete UBL document structure with namespace declarations\n */\nexport declare const generateCleanUBLDocument: (invoices: AllDocumentsV1_1[]) => UBLDocument;\n/**\n * Step 2: Calculate Document Digest\n * FIXED: Remove UBLExtensions and Signature before hashing (DS322)\n * Based on working implementation pattern\n */\nexport declare const calculateDocumentDigest: (invoices: AllDocumentsV1_1[]) => string;\n/**\n * Step 4: Calculate Certificate Digest\n * Enhanced to handle certificate content properly\n */\nexport declare const calculateCertificateDigest: (certificatePem: string) => string;\n/**\n * Enhanced certificate info extraction with better error handling\n * FIXED: Normalize issuer name format to match MyInvois expectations (DS326)\n */\nexport declare const extractCertificateInfo: (certificatePem: string) => {\n issuerName: string;\n serialNumber: string;\n subjectName: string;\n};\n/**\n * Step 5: Create SignedProperties with enhanced structure\n * FIXED: Simplified structure to match MyInvois expectations (DS320)\n * Following MyInvois JSON signature specification exactly\n */\nexport declare const createSignedProperties: (certificateDigest: string, signingTime: string, issuerName: string, serialNumber: string) => SignedPropertiesObject;\n/**\n * Step 6: Calculate SignedProperties Digest\n * FIXED: Add Target wrapper and use direct JSON stringify (DS320)\n * Based on working implementation pattern\n */\nexport declare const calculateSignedPropertiesDigest: (signedProperties: SignedPropertiesObject) => string;\n/**\n * Step 3: Create SignedInfo and calculate signature\n * Enhanced with better structure and signature generation\n */\nexport declare const createSignedInfoAndSign: (docDigest: string, propsDigest: string, privateKeyPem: string) => {\n signedInfo: SignedInfoObject;\n signatureValue: string;\n};\n/**\n * Complete document generation with signatures\n * Follows the complete MyInvois JSON signature creation process (Steps 1-7)\n */\nexport declare const generateCompleteDocument: (invoices: AllDocumentsV1_1[], signingCredentials: SigningCredentials) => CompleteInvoice;\n/**\n * Creates a line item with percentage-based taxation (e.g., SST, GST)\n */\nexport declare const createPercentageTaxLineItem: (params: {\n itemClassificationCode: ClassificationCode;\n itemDescription: string;\n unitPrice: number;\n quantity?: number;\n taxType: TaxTypeCode;\n taxRate: number;\n totalTaxableAmountPerLine?: number;\n}) => InvoiceLineItem;\n/**\n * Creates a line item with fixed rate taxation (e.g., Tourism Tax)\n */\nexport declare const createFixedRateTaxLineItem: (params: {\n itemClassificationCode: ClassificationCode;\n itemDescription: string;\n unitPrice: number;\n quantity?: number;\n taxType: TaxTypeCode;\n taxPerUnitAmount: number;\n baseUnitMeasure: number;\n baseUnitMeasureCode: UnitTypeCode;\n totalTaxableAmountPerLine?: number;\n}) => InvoiceLineItem;\n/**\n * Calculates invoice totals from line items\n */\nexport declare const calculateInvoiceTotals: (lineItems: InvoiceLineItem[]) => {\n legalMonetaryTotal: {\n taxExclusiveAmount: number;\n taxInclusiveAmount: number;\n payableAmount: number;\n };\n taxTotal: {\n taxAmount: number;\n };\n};\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAGA,IAAE,iBAAA,CAAA,KAAA,MAAA,eAAA;AACF,IAAW,kBAAkB,CAAC,KAAK,MAAM,eAAgB;AACzD,IAAW,6BAA6B,CAAC,KAAK,MAAM,eAAgB;AACpE,IAAW,iBAAiB,CAAC,GAAI;AACjC,IAAE,mBAAA,CAAA,GAAA;AACF,IAAE,6BAAA;CAAA;CAAA,MAAA;CAAA,MAAA;AAAA;AACF,IAAW,2BAA2B;CAAC;CAAK,MAAM;CAAE,MAAA;AAAA;AACpD,IAAE,0BAAA,CAAA,KAAA,MAAA,gBAAA;AACF,IAAW,6BAA6B,CAAC,GAAI;AAC7C,IAAE,yBAAA,CAAA,GAAA;AACF,IAAW,yBAAyB,CAAC,KAAK,MAAM,sBAAI;AACpD,IAAE,kCAAA,CAAA,KAAA,MAAA,sBAAA;AACF,IAAW,0BAA0B,CAAC,KAAK,MAAM,gBAAiB;AAClE,IAAE,2BAAA;CAAA;CAAA,MAAA;CAAA,MAAA;CAAA,MAAA;AAAA;AACF,IAAW,8BAA8B;CAAC;CAAK,MAAM;CAAiB,MAAA;CAAA,MAAA;AAAA;AACtE,IAAE,6BAAA;CAAA;CAAA,MAAA;CAAA,MAAA;CAAA,MAAA;CAAA,MAAA;AAAA;AACF,IAAW,yBAAyB,CAAC,KAAK,MAAM,eAAgB"}
1
+ {"version":3,"file":"index71.cts","names":[],"sources":["../src/utils/document.d.ts"],"sourcesContent":["import { InvoiceSubmission, AllDocumentsV1_1, SigningCredentials, SignedPropertiesObject, UBLDocument, CompleteInvoice, SignedInfoObject, UnitTypeCode, InvoiceLineItem } from '../types';\nimport type { ClassificationCode } from '../types';\nimport type { TaxTypeCode } from '../types';\n/**\n * MyInvois v1.1 Document Generation and Signing Utilities\n * Strictly follows: https://sdk.myinvois.hasil.gov.my/documents/invoice-v1-1\n * JSON Signature Guide: https://sdk.myinvois.hasil.gov.my/signature-creation-json/\n */\n/**\n * Determines if a line item uses fixed rate taxation\n */\nexport declare const isFixedRateTax: (item: InvoiceLineItem) => boolean;\n/**\n * Determines if a line item uses percentage taxation\n */\nexport declare const isPercentageTax: (item: InvoiceLineItem) => boolean;\n/**\n * Calculates expected tax amount for a line item based on its tax type\n */\nexport declare const calculateExpectedTaxAmount: (item: InvoiceLineItem) => number;\n/**\n * Helper function to recursively sort object keys for JSON canonicalization\n */\nexport declare function sortObjectKeys(obj: unknown): unknown;\n/**\n * Enhanced canonicalization following MyInvois specification exactly\n * Key changes: ensure consistent ordering and formatting\n */\nexport declare const canonicalizeJSON: (obj: unknown) => string;\n/**\n * Generates a clean invoice object following MyInvois v1.1 specification exactly\n * This is the base invoice structure WITHOUT signature elements (for hash calculation)\n *\n * Key requirements from working documents:\n * - All mandatory fields must be present\n * - Many optional fields must be present even if empty\n * - Specific field ordering and structure\n * - Correct listID values (e.g., \"3166-1\" not \"ISO3166-1\")\n */\nexport declare const generateCleanInvoiceObject: (invoice: AllDocumentsV1_1) => InvoiceSubmission;\n/**\n * Generates the complete UBL document structure with namespace declarations\n */\nexport declare const generateCleanUBLDocument: (invoices: AllDocumentsV1_1[]) => UBLDocument;\n/**\n * Step 1: Transform the document for hashing or transmission\n * Removes UBLExtensions and Signature, and minifies the JSON\n * Returns the minified, cleaned JSON string\n */\nexport declare const transformDocumentForHashing: (invoices: AllDocumentsV1_1[]) => string;\n/**\n * Step 2: Calculate Document Digest\n * FIXED: Remove UBLExtensions and Signature before hashing (DS322)\n * Based on working implementation pattern\n */\nexport declare const calculateDocumentDigest: (invoices: AllDocumentsV1_1[]) => string;\n/**\n * Step 4: Calculate Certificate Digest\n * Enhanced to handle certificate content properly\n */\nexport declare const calculateCertificateDigest: (certificatePem: string) => string;\n/**\n * Enhanced certificate info extraction with better error handling\n * FIXED: Normalize issuer name format to match MyInvois expectations (DS326)\n */\nexport declare const extractCertificateInfo: (certificatePem: string) => {\n issuerName: string;\n serialNumber: string;\n subjectName: string;\n};\n/**\n * Step 5: Create SignedProperties with enhanced structure\n * FIXED: Simplified structure to match MyInvois expectations (DS320)\n * Following MyInvois JSON signature specification exactly\n */\nexport declare const createSignedProperties: (certificateDigest: string, signingTime: string, issuerName: string, serialNumber: string) => SignedPropertiesObject;\n/**\n * Step 6: Calculate SignedProperties Digest\n * Calculates the digest over the correct structure for validator compliance.\n */\nexport declare const calculateSignedPropertiesDigest: (signedProperties: SignedPropertiesObject, useTargetWrapper?: boolean) => string;\n/**\n * Step 3: Create SignedInfo and calculate signature\n * Enhanced with better structure and signature generation\n */\nexport declare const createSignedInfoAndSign: (docDigest: string, propsDigest: string, privateKeyPem: string) => {\n signedInfo: SignedInfoObject;\n signatureValue: string;\n};\n/**\n * Signs the minified document string using the provided private key PEM\n * Returns the signature as a base64 string\n */\nexport declare const signDocumentString: (documentString: string, privateKeyPem: string) => string;\n/**\n * Complete document generation with signatures\n * Follows the complete MyInvois JSON signature creation process (Steps 1-7)\n */\nexport declare const generateCompleteDocument: (invoices: AllDocumentsV1_1[], signingCredentials: SigningCredentials) => CompleteInvoice;\n/**\n * Creates a line item with percentage-based taxation (e.g., SST, GST)\n */\nexport declare const createPercentageTaxLineItem: (params: {\n itemClassificationCode: ClassificationCode;\n itemDescription: string;\n unitPrice: number;\n quantity?: number;\n taxType: TaxTypeCode;\n taxRate: number;\n totalTaxableAmountPerLine?: number;\n}) => InvoiceLineItem;\n/**\n * Creates a line item with fixed rate taxation (e.g., Tourism Tax)\n */\nexport declare const createFixedRateTaxLineItem: (params: {\n itemClassificationCode: ClassificationCode;\n itemDescription: string;\n unitPrice: number;\n quantity?: number;\n taxType: TaxTypeCode;\n taxPerUnitAmount: number;\n baseUnitMeasure: number;\n baseUnitMeasureCode: UnitTypeCode;\n totalTaxableAmountPerLine?: number;\n}) => InvoiceLineItem;\n/**\n * Calculates invoice totals from line items\n */\nexport declare const calculateInvoiceTotals: (lineItems: InvoiceLineItem[]) => {\n legalMonetaryTotal: {\n taxExclusiveAmount: number;\n taxInclusiveAmount: number;\n payableAmount: number;\n };\n taxTotal: {\n taxAmount: number;\n };\n};\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAGA,IAAE,iBAAA,CAAA,KAAA,MAAA,eAAA;AACF,IAAW,kBAAkB,CAAC,KAAK,MAAM,eAAgB;AACzD,IAAW,6BAA6B,CAAC,KAAK,MAAM,eAAgB;AACpE,IAAW,iBAAiB,CAAC,GAAI;AACjC,IAAE,mBAAA,CAAA,GAAA;AACF,IAAE,6BAAA;CAAA;CAAA,MAAA;CAAA,MAAA;AAAA;AACF,IAAW,2BAA2B;CAAC;CAAK,MAAM;CAAE,MAAA;AAAA;AACpD,IAAE,8BAAA,CAAA,KAAA,MAAA,gBAAA;AACF,IAAW,0BAA0B,CAAC,KAAK,MAAM,gBAAiB;AAClE,IAAE,6BAAA,CAAA,GAAA;AACF,IAAW,yBAAyB,CAAC,GAAI;AACzC,IAAE,yBAAA,CAAA,KAAA,MAAA,sBAAA;AACF,IAAW,kCAAkC,CAAC,KAAK,MAAM,sBAAe;AACxE,IAAE,0BAAA,CAAA,KAAA,MAAA,gBAAA;AACF,IAAW,qBAAqB,CAAC,GAAI;AACrC,IAAE,2BAAA;CAAA;CAAA,MAAA;CAAA,MAAA;CAAA,MAAA;AAAA;AACF,IAAW,8BAA8B;CAAC;CAAK,MAAM;CAAoB,MAAM;CAAG,MAAA;AAAA;AAClF,IAAE,6BAAA;CAAA;CAAA,MAAA;CAAA,MAAA;CAAA,MAAA;CAAA,MAAA;AAAA;AACF,IAAW,yBAAyB,CAAC,KAAK,MAAM,eAAgB"}
@@ -1,5 +1,5 @@
1
1
  import { MSICCode } from "./msic-codes-CIKdPqag.cjs";
2
- import { RegistrationType } from "./documents-Dmn5FIRZ.cjs";
2
+ import { RegistrationType } from "./documents-BWpJ0B3C.cjs";
3
3
 
4
4
  //#region src/types/taxpayer.d.ts
5
5
  interface TaxpayerQRCodeResponse {
@@ -32,4 +32,4 @@ interface TinSearchResponse {
32
32
  }
33
33
  //#endregion
34
34
  export { TaxpayerQRCodeResponse, TinSearchParams, TinSearchResponse };
35
- //# sourceMappingURL=taxpayer-TB35wh00.d.cts.map
35
+ //# sourceMappingURL=taxpayer-CApK7x_O.d.cts.map
@@ -1,5 +1,5 @@
1
1
  import { MSICCode } from "./msic-codes-CjrrJmzb.js";
2
- import { RegistrationType } from "./documents-AlZIGjw2.js";
2
+ import { RegistrationType } from "./documents-DRQ6_toX.js";
3
3
 
4
4
  //#region src/types/taxpayer.d.ts
5
5
  interface TaxpayerQRCodeResponse {
@@ -51,5 +51,5 @@ import "../YX-F34sJ7Ik.js";
51
51
  import "../ZX-CDQOfsHh.js";
52
52
  import "../XX-DOA-10JW.js";
53
53
  import "../unit-types-VgYXIwTT.js";
54
- import { Address, AllDocumentsV1_1, Buyer, CompleteInvoice, CreditNoteV1_1, DebitNoteV1_1, DocumentStatus, DocumentSummary, DocumentTypeResponse, DocumentTypeVersion, DocumentTypeVersionResponse, DocumentTypesResponse, DocumentValidationResult, DocumentValidationStepResult, FinalDocumentData, GetSubmissionResponse, InvoiceLineItem, InvoiceSubmission, InvoiceV1_1, LegalMonetaryTotal, RefundNoteV1_1, RegistrationType, ResponseDocument, SelfBilledCreditNoteV1_1, SelfBilledInvoiceV1_1, SelfBilledRefundNoteV1_1, SignedInfoObject, SignedInvoiceSubmission, SignedPropertiesData, SignedPropertiesObject, SigningCredentials, StandardError, SubmissionResponse, SubmissionStatus, Supplier, TaxCategory, TaxSubtotal, TaxTotal, UBLDocument, WorkflowParameter } from "../documents-AlZIGjw2.js";
54
+ import { Address, AllDocumentsV1_1, Buyer, CompleteInvoice, CreditNoteV1_1, DebitNoteV1_1, DocumentStatus, DocumentSummary, DocumentTypeResponse, DocumentTypeVersion, DocumentTypeVersionResponse, DocumentTypesResponse, DocumentValidationResult, DocumentValidationStepResult, FinalDocumentData, GetSubmissionResponse, InvoiceLineItem, InvoiceSubmission, InvoiceV1_1, LegalMonetaryTotal, RefundNoteV1_1, RegistrationType, ResponseDocument, SelfBilledCreditNoteV1_1, SelfBilledInvoiceV1_1, SelfBilledRefundNoteV1_1, SignedInfoObject, SignedInvoiceSubmission, SignedPropertiesData, SignedPropertiesObject, SigningCredentials, StandardError, SubmissionResponse, SubmissionStatus, Supplier, TaxCategory, TaxSubtotal, TaxTotal, UBLDocument, WorkflowParameter } from "../documents-DRQ6_toX.js";
55
55
  export { Address, AllDocumentsV1_1, Buyer, CompleteInvoice, CreditNoteV1_1, DebitNoteV1_1, DocumentStatus, DocumentSummary, DocumentTypeResponse, DocumentTypeVersion, DocumentTypeVersionResponse, DocumentTypesResponse, DocumentValidationResult, DocumentValidationStepResult, FinalDocumentData, GetSubmissionResponse, InvoiceLineItem, InvoiceSubmission, InvoiceV1_1, LegalMonetaryTotal, RefundNoteV1_1, RegistrationType, ResponseDocument, SelfBilledCreditNoteV1_1, SelfBilledInvoiceV1_1, SelfBilledRefundNoteV1_1, SignedInfoObject, SignedInvoiceSubmission, SignedPropertiesData, SignedPropertiesObject, SigningCredentials, StandardError, SubmissionResponse, SubmissionStatus, Supplier, TaxCategory, TaxSubtotal, TaxTotal, UBLDocument, WorkflowParameter };
@@ -51,11 +51,11 @@ import "../YX-F34sJ7Ik.js";
51
51
  import "../ZX-CDQOfsHh.js";
52
52
  import "../XX-DOA-10JW.js";
53
53
  import { UnitType, UnitTypeCode } from "../unit-types-VgYXIwTT.js";
54
- import { Address, AllDocumentsV1_1, Buyer, CompleteInvoice, CreditNoteV1_1, DebitNoteV1_1, DocumentStatus, DocumentSummary, DocumentTypeResponse, DocumentTypeVersion, DocumentTypeVersionResponse, DocumentTypesResponse, DocumentValidationResult, DocumentValidationStepResult, FinalDocumentData, GetSubmissionResponse, InvoiceLineItem, InvoiceSubmission, InvoiceV1_1, LegalMonetaryTotal, RefundNoteV1_1, RegistrationType, ResponseDocument, SelfBilledCreditNoteV1_1, SelfBilledInvoiceV1_1, SelfBilledRefundNoteV1_1, SignedInfoObject, SignedInvoiceSubmission, SignedPropertiesData, SignedPropertiesObject, SigningCredentials, StandardError, SubmissionResponse, SubmissionStatus, Supplier, TaxCategory, TaxSubtotal, TaxTotal, UBLDocument, WorkflowParameter } from "../documents-AlZIGjw2.js";
54
+ import { Address, AllDocumentsV1_1, Buyer, CompleteInvoice, CreditNoteV1_1, DebitNoteV1_1, DocumentStatus, DocumentSummary, DocumentTypeResponse, DocumentTypeVersion, DocumentTypeVersionResponse, DocumentTypesResponse, DocumentValidationResult, DocumentValidationStepResult, FinalDocumentData, GetSubmissionResponse, InvoiceLineItem, InvoiceSubmission, InvoiceV1_1, LegalMonetaryTotal, RefundNoteV1_1, RegistrationType, ResponseDocument, SelfBilledCreditNoteV1_1, SelfBilledInvoiceV1_1, SelfBilledRefundNoteV1_1, SignedInfoObject, SignedInvoiceSubmission, SignedPropertiesData, SignedPropertiesObject, SigningCredentials, StandardError, SubmissionResponse, SubmissionStatus, Supplier, TaxCategory, TaxSubtotal, TaxTotal, UBLDocument, WorkflowParameter } from "../documents-DRQ6_toX.js";
55
55
  import { PaymentMode, PaymentModeCode, PaymentModeCodeEnum } from "../payment-modes-g3DzLmWb.js";
56
56
  import { CanonicalizationMethod, Cert, CertDigest, DigestMethod, IssuerDigitalSignature, IssuerSerial, KeyInfo, QualifyingProperties, Reference, SignatureMethod, SignedInfo, SignedProperties, SignedSignatureProperties, SigningCertificate, Transform, X509Data } from "../signatures-CerHUrj3.js";
57
57
  import { Notification, NotificationDeliveryAttempt, NotificationMetadata, NotificationResponse, NotificationSearchParams, NotificationStatus, NotificationStatusEnum, NotificationType, NotificationTypeEnum } from "../notifications-sFhgh3rJ.js";
58
- import { TaxpayerQRCodeResponse, TinSearchParams, TinSearchResponse } from "../taxpayer-DZmqFn6t.js";
58
+ import { TaxpayerQRCodeResponse, TinSearchParams, TinSearchResponse } from "../taxpayer-DtyHXKwz.js";
59
59
  import { Fetch } from "../utils-C4FoVKLq.js";
60
60
  import { ClientCredentials, TokenResponse } from "../index-CygwSf0x.js";
61
61
  export { Address, AllDocumentsV1_1, Buyer, CanonicalizationMethod, Cert, CertDigest, Classification, ClassificationCode, ClassificationCodeEnum, ClientCredentials, CompleteInvoice, Country, CountryCode, CountryCodeEnum, CountryNameEnum, CreditNoteV1_1, Currency, CurrencyCode, CurrencyCodeEnum, DebitNoteV1_1, DigestMethod, DocumentStatus, DocumentSummary, DocumentTypeResponse, DocumentTypeVersion, DocumentTypeVersionResponse, DocumentTypesResponse, DocumentValidationResult, DocumentValidationStepResult, EInvoiceType, EInvoiceTypeCode, EInvoiceTypeCodeEnum, Fetch, FinalDocumentData, GetSubmissionResponse, InvoiceLineItem, InvoiceSubmission, InvoiceV1_1, IssuerDigitalSignature, IssuerSerial, KeyInfo, LegalMonetaryTotal, MSICCode, Notification, NotificationDeliveryAttempt, NotificationMetadata, NotificationResponse, NotificationSearchParams, NotificationStatus, NotificationStatusEnum, NotificationType, NotificationTypeEnum, PaymentMode, PaymentModeCode, PaymentModeCodeEnum, QualifyingProperties, Reference, RefundNoteV1_1, RegistrationType, ResponseDocument, SelfBilledCreditNoteV1_1, SelfBilledInvoiceV1_1, SelfBilledRefundNoteV1_1, SignatureMethod, SignedInfo, SignedInfoObject, SignedInvoiceSubmission, SignedProperties, SignedPropertiesData, SignedPropertiesObject, SignedSignatureProperties, SigningCertificate, SigningCredentials, StandardError, State, StateCode, StateCodeEnum, SubmissionResponse, SubmissionStatus, Supplier, TaxCategory, TaxSubtotal, TaxTotal, TaxType, TaxTypeCode, TaxTypeCodeEnum, TaxpayerQRCodeResponse, TinSearchParams, TinSearchResponse, TokenResponse, Transform, UBLDocument, UnitType, UnitTypeCode, WorkflowParameter, X509Data };
@@ -51,6 +51,6 @@ import "../YX-F34sJ7Ik.js";
51
51
  import "../ZX-CDQOfsHh.js";
52
52
  import "../XX-DOA-10JW.js";
53
53
  import "../unit-types-VgYXIwTT.js";
54
- import "../documents-AlZIGjw2.js";
55
- import { TaxpayerQRCodeResponse, TinSearchParams, TinSearchResponse } from "../taxpayer-DZmqFn6t.js";
54
+ import "../documents-DRQ6_toX.js";
55
+ import { TaxpayerQRCodeResponse, TinSearchParams, TinSearchResponse } from "../taxpayer-DtyHXKwz.js";
56
56
  export { TaxpayerQRCodeResponse, TinSearchParams, TinSearchResponse };
@@ -51,11 +51,11 @@ import "../YX-F34sJ7Ik.js";
51
51
  import "../ZX-CDQOfsHh.js";
52
52
  import "../XX-DOA-10JW.js";
53
53
  import { UnitTypeCode } from "../unit-types-VgYXIwTT.js";
54
- import { AllDocumentsV1_1, CompleteInvoice, InvoiceLineItem, InvoiceSubmission, SignedInfoObject, SignedPropertiesObject, SigningCredentials, UBLDocument } from "../documents-AlZIGjw2.js";
54
+ import { AllDocumentsV1_1, CompleteInvoice, InvoiceLineItem, InvoiceSubmission, SignedInfoObject, SignedPropertiesObject, SigningCredentials, UBLDocument } from "../documents-DRQ6_toX.js";
55
55
  import "../payment-modes-g3DzLmWb.js";
56
56
  import "../signatures-CerHUrj3.js";
57
57
  import "../notifications-sFhgh3rJ.js";
58
- import "../taxpayer-DZmqFn6t.js";
58
+ import "../taxpayer-DtyHXKwz.js";
59
59
  import "../utils-C4FoVKLq.js";
60
60
  import "../index-CygwSf0x.js";
61
61
 
@@ -102,6 +102,12 @@ declare const generateCleanInvoiceObject: (invoice: AllDocumentsV1_1) => Invoice
102
102
  */
103
103
  declare const generateCleanUBLDocument: (invoices: AllDocumentsV1_1[]) => UBLDocument;
104
104
  /**
105
+ * Step 1: Transform the document for hashing or transmission
106
+ * Removes UBLExtensions and Signature, and minifies the JSON
107
+ * Returns the minified, cleaned JSON string
108
+ */
109
+ declare const transformDocumentForHashing: (invoices: AllDocumentsV1_1[]) => string;
110
+ /**
105
111
  * Step 2: Calculate Document Digest
106
112
  * FIXED: Remove UBLExtensions and Signature before hashing (DS322)
107
113
  * Based on working implementation pattern
@@ -129,10 +135,9 @@ declare const extractCertificateInfo: (certificatePem: string) => {
129
135
  declare const createSignedProperties: (certificateDigest: string, signingTime: string, issuerName: string, serialNumber: string) => SignedPropertiesObject;
130
136
  /**
131
137
  * Step 6: Calculate SignedProperties Digest
132
- * FIXED: Add Target wrapper and use direct JSON stringify (DS320)
133
- * Based on working implementation pattern
138
+ * Calculates the digest over the correct structure for validator compliance.
134
139
  */
135
- declare const calculateSignedPropertiesDigest: (signedProperties: SignedPropertiesObject) => string;
140
+ declare const calculateSignedPropertiesDigest: (signedProperties: SignedPropertiesObject, useTargetWrapper?: boolean) => string;
136
141
  /**
137
142
  * Step 3: Create SignedInfo and calculate signature
138
143
  * Enhanced with better structure and signature generation
@@ -142,6 +147,11 @@ declare const createSignedInfoAndSign: (docDigest: string, propsDigest: string,
142
147
  signatureValue: string;
143
148
  };
144
149
  /**
150
+ * Signs the minified document string using the provided private key PEM
151
+ * Returns the signature as a base64 string
152
+ */
153
+ declare const signDocumentString: (documentString: string, privateKeyPem: string) => string;
154
+ /**
145
155
  * Complete document generation with signatures
146
156
  * Follows the complete MyInvois JSON signature creation process (Steps 1-7)
147
157
  */
@@ -186,4 +196,4 @@ declare const calculateInvoiceTotals: (lineItems: InvoiceLineItem[]) => {
186
196
  };
187
197
  };
188
198
  //#endregion
189
- export { calculateCertificateDigest, calculateDocumentDigest, calculateExpectedTaxAmount, calculateInvoiceTotals, calculateSignedPropertiesDigest, canonicalizeJSON, createFixedRateTaxLineItem, createPercentageTaxLineItem, createSignedInfoAndSign, createSignedProperties, extractCertificateInfo, generateCleanInvoiceObject, generateCleanUBLDocument, generateCompleteDocument, isFixedRateTax, isPercentageTax, sortObjectKeys };
199
+ export { calculateCertificateDigest, calculateDocumentDigest, calculateExpectedTaxAmount, calculateInvoiceTotals, calculateSignedPropertiesDigest, canonicalizeJSON, createFixedRateTaxLineItem, createPercentageTaxLineItem, createSignedInfoAndSign, createSignedProperties, extractCertificateInfo, generateCleanInvoiceObject, generateCleanUBLDocument, generateCompleteDocument, isFixedRateTax, isPercentageTax, signDocumentString, sortObjectKeys, transformDocumentForHashing };
@@ -1,4 +1,4 @@
1
1
  import "../formatIdValue-qTxJqj9o.js";
2
- import { calculateCertificateDigest, calculateDocumentDigest, calculateExpectedTaxAmount, calculateInvoiceTotals, calculateSignedPropertiesDigest, canonicalizeJSON, createFixedRateTaxLineItem, createPercentageTaxLineItem, createSignedInfoAndSign, createSignedProperties, extractCertificateInfo, generateCleanInvoiceObject, generateCleanUBLDocument, generateCompleteDocument, isFixedRateTax, isPercentageTax, sortObjectKeys } from "../document-BRxLmvIU.js";
2
+ import { calculateCertificateDigest, calculateDocumentDigest, calculateExpectedTaxAmount, calculateInvoiceTotals, calculateSignedPropertiesDigest, canonicalizeJSON, createFixedRateTaxLineItem, createPercentageTaxLineItem, createSignedInfoAndSign, createSignedProperties, extractCertificateInfo, generateCleanInvoiceObject, generateCleanUBLDocument, generateCompleteDocument, isFixedRateTax, isPercentageTax, signDocumentString, sortObjectKeys, transformDocumentForHashing } from "../document-BbggZ72T.js";
3
3
 
4
- export { calculateCertificateDigest, calculateDocumentDigest, calculateExpectedTaxAmount, calculateInvoiceTotals, calculateSignedPropertiesDigest, canonicalizeJSON, createFixedRateTaxLineItem, createPercentageTaxLineItem, createSignedInfoAndSign, createSignedProperties, extractCertificateInfo, generateCleanInvoiceObject, generateCleanUBLDocument, generateCompleteDocument, isFixedRateTax, isPercentageTax, sortObjectKeys };
4
+ export { calculateCertificateDigest, calculateDocumentDigest, calculateExpectedTaxAmount, calculateInvoiceTotals, calculateSignedPropertiesDigest, canonicalizeJSON, createFixedRateTaxLineItem, createPercentageTaxLineItem, createSignedInfoAndSign, createSignedProperties, extractCertificateInfo, generateCleanInvoiceObject, generateCleanUBLDocument, generateCompleteDocument, isFixedRateTax, isPercentageTax, signDocumentString, sortObjectKeys, transformDocumentForHashing };
@@ -51,11 +51,11 @@ import "../YX-F34sJ7Ik.js";
51
51
  import "../ZX-CDQOfsHh.js";
52
52
  import "../XX-DOA-10JW.js";
53
53
  import "../unit-types-VgYXIwTT.js";
54
- import { InvoiceV1_1 } from "../documents-AlZIGjw2.js";
54
+ import { InvoiceV1_1 } from "../documents-DRQ6_toX.js";
55
55
  import "../payment-modes-g3DzLmWb.js";
56
56
  import "../signatures-CerHUrj3.js";
57
57
  import "../notifications-sFhgh3rJ.js";
58
- import "../taxpayer-DZmqFn6t.js";
58
+ import "../taxpayer-DtyHXKwz.js";
59
59
  import "../utils-C4FoVKLq.js";
60
60
  import "../index-CygwSf0x.js";
61
61
 
@@ -1,5 +1,5 @@
1
1
  import "../formatIdValue-qTxJqj9o.js";
2
- import { calculateCertificateDigest, calculateDocumentDigest, calculateSignedPropertiesDigest, createSignedProperties, extractCertificateInfo } from "../document-BRxLmvIU.js";
2
+ import { calculateCertificateDigest, calculateDocumentDigest, calculateSignedPropertiesDigest, createSignedProperties, extractCertificateInfo } from "../document-BbggZ72T.js";
3
3
  import crypto from "crypto";
4
4
 
5
5
  //#region src/utils/signature-diagnostics.ts
@@ -51,11 +51,11 @@ import "../YX-F34sJ7Ik.js";
51
51
  import "../ZX-CDQOfsHh.js";
52
52
  import "../XX-DOA-10JW.js";
53
53
  import "../unit-types-VgYXIwTT.js";
54
- import { InvoiceLineItem, InvoiceV1_1 } from "../documents-AlZIGjw2.js";
54
+ import { InvoiceLineItem, InvoiceV1_1 } from "../documents-DRQ6_toX.js";
55
55
  import "../payment-modes-g3DzLmWb.js";
56
56
  import "../signatures-CerHUrj3.js";
57
57
  import "../notifications-sFhgh3rJ.js";
58
- import "../taxpayer-DZmqFn6t.js";
58
+ import "../taxpayer-DtyHXKwz.js";
59
59
  import "../utils-C4FoVKLq.js";
60
60
  import "../index-CygwSf0x.js";
61
61
 
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@ripwords/myinvois-client",
3
- "version": "0.2.23",
3
+ "version": "0.2.25",
4
4
  "description": "MyInvois client",
5
5
  "scripts": {
6
6
  "build": "tsdown",
@@ -1 +0,0 @@
1
- {"version":3,"file":"document-Dvonhaqt.cjs","names":["item: InvoiceLineItem","doc: AllDocumentsV1_1","obj: unknown","sortedObj: Record<string, unknown>","invoice: AllDocumentsV1_1","invoices: AllDocumentsV1_1[]","invoice: any","certificatePem: string","X509Certificate","dn: string","serialHex: string","error: unknown","certificateDigest: string","signingTime: string","issuerName: string","serialNumber: string","signedProperties: SignedPropertiesObject","docDigest: string","propsDigest: string","privateKeyPem: string","signingCredentials: SigningCredentials","params: {\n itemClassificationCode: ClassificationCode\n itemDescription: string\n unitPrice: number\n quantity?: number\n taxType: TaxTypeCode\n taxRate: number\n totalTaxableAmountPerLine?: number\n}","params: {\n itemClassificationCode: ClassificationCode\n itemDescription: string\n unitPrice: number\n quantity?: number\n taxType: TaxTypeCode\n taxPerUnitAmount: number\n baseUnitMeasure: number\n baseUnitMeasureCode: UnitTypeCode\n totalTaxableAmountPerLine?: number\n}","lineItems: InvoiceLineItem[]"],"sources":["../src/utils/document.ts"],"sourcesContent":["import crypto, { X509Certificate } from 'crypto'\nimport {\n InvoiceSubmission,\n AllDocumentsV1_1,\n SigningCredentials,\n SignedPropertiesObject,\n UBLDocument,\n CompleteInvoice,\n SignedInfoObject,\n UnitTypeCode,\n InvoiceLineItem,\n} from '../types'\nimport type { ClassificationCode } from '../types'\nimport type { TaxTypeCode } from '../types'\nimport { formatIdValue } from './formatIdValue'\n\n/**\n * MyInvois v1.1 Document Generation and Signing Utilities\n * Strictly follows: https://sdk.myinvois.hasil.gov.my/documents/invoice-v1-1\n * JSON Signature Guide: https://sdk.myinvois.hasil.gov.my/signature-creation-json/\n */\n\n/**\n * Determines if a line item uses fixed rate taxation\n */\nexport const isFixedRateTax = (item: InvoiceLineItem): boolean => {\n return (\n item.taxPerUnitAmount !== undefined && item.baseUnitMeasure !== undefined\n )\n}\n\n/**\n * Determines if a line item uses percentage taxation\n */\nexport const isPercentageTax = (item: InvoiceLineItem): boolean => {\n return item.taxRate !== undefined && !isFixedRateTax(item)\n}\n\n/**\n * Calculates expected tax amount for a line item based on its tax type\n */\nexport const calculateExpectedTaxAmount = (item: InvoiceLineItem): number => {\n if (isFixedRateTax(item)) {\n return item.taxPerUnitAmount! * item.baseUnitMeasure!\n } else if (isPercentageTax(item)) {\n return (item.totalTaxableAmountPerLine * item.taxRate!) / 100\n }\n return 0\n}\n\n/**\n * Extracts the line-item array from any document variant\n */\nconst getLineItems = (doc: AllDocumentsV1_1): InvoiceLineItem[] => {\n if ('invoiceLineItems' in doc) return doc.invoiceLineItems\n if ('creditNoteLineItems' in doc) return doc.creditNoteLineItems\n if ('debitNoteLineItems' in doc) return doc.debitNoteLineItems\n if ('refundNoteLineItems' in doc) return doc.refundNoteLineItems\n if ('selfBilledCreditNoteLineItems' in doc)\n return doc.selfBilledCreditNoteLineItems\n if ('selfBilledRefundNoteLineItems' in doc)\n return doc.selfBilledRefundNoteLineItems\n // Fallback (should never happen with exhaustive types)\n return []\n}\n\n/**\n * Helper function to recursively sort object keys for JSON canonicalization\n */\nexport function sortObjectKeys(obj: unknown): unknown {\n if (obj === null || typeof obj !== 'object') {\n return obj\n }\n\n if (Array.isArray(obj)) {\n return obj.map(sortObjectKeys)\n }\n\n const sortedObj: Record<string, unknown> = {}\n const keys = Object.keys(obj as Record<string, unknown>).sort()\n\n for (const key of keys) {\n sortedObj[key] = sortObjectKeys((obj as Record<string, unknown>)[key])\n }\n\n return sortedObj\n}\n\n/**\n * Enhanced canonicalization following MyInvois specification exactly\n * Key changes: ensure consistent ordering and formatting\n */\nexport const canonicalizeJSON = (obj: unknown): string => {\n const sortedObj = sortObjectKeys(obj)\n // Use compact JSON with no extra whitespace\n return JSON.stringify(sortedObj, null, 0)\n}\n\n/**\n * Generates a clean invoice object following MyInvois v1.1 specification exactly\n * This is the base invoice structure WITHOUT signature elements (for hash calculation)\n *\n * Key requirements from working documents:\n * - All mandatory fields must be present\n * - Many optional fields must be present even if empty\n * - Specific field ordering and structure\n * - Correct listID values (e.g., \"3166-1\" not \"ISO3166-1\")\n */\nexport const generateCleanInvoiceObject = (\n invoice: AllDocumentsV1_1,\n): InvoiceSubmission => {\n const lineItems = getLineItems(invoice)\n return {\n // === MANDATORY CORE FIELDS ===\n ID: [{ _: invoice.eInvoiceCodeOrNumber }],\n IssueDate: [{ _: invoice.eInvoiceDate }],\n IssueTime: [{ _: invoice.eInvoiceTime }],\n InvoiceTypeCode: [\n {\n _: invoice.eInvoiceTypeCode,\n listVersionID: invoice.eInvoiceVersion || '1.1',\n },\n ],\n DocumentCurrencyCode: [{ _: invoice.invoiceCurrencyCode }],\n\n // === BILLING REFERENCE (only for credit/debit/refund notes) ===\n ...('originalEInvoiceReferenceNumber' in invoice &&\n 'originalEInvoiceInternalId' in invoice &&\n invoice.originalEInvoiceReferenceNumber\n ? {\n BillingReference: [\n {\n InvoiceDocumentReference: [\n {\n UUID: [\n {\n _: invoice.originalEInvoiceReferenceNumber,\n },\n ],\n ID: [\n {\n _: invoice.originalEInvoiceInternalId,\n },\n ],\n },\n ],\n },\n ],\n }\n : {}),\n\n // === SUPPLIER PARTY (AccountingSupplierParty) ===\n AccountingSupplierParty: [\n {\n Party: [\n {\n // Industry Classification - required field\n IndustryClassificationCode: [\n {\n _: invoice.supplier.industryClassificationCode,\n name: invoice.supplier.industryClassificationDescription,\n },\n ],\n\n // Party Identifications\n PartyIdentification: [\n {\n ID: [\n {\n _: invoice.supplier.tin,\n schemeID: 'TIN',\n },\n ],\n },\n {\n ID: [\n {\n _: formatIdValue(invoice.supplier.registrationNumber),\n schemeID: invoice.supplier.registrationType || 'NRIC',\n },\n ],\n },\n ],\n\n // Postal Address - FIXED listID format\n PostalAddress: [\n {\n CityName: [{ _: invoice.supplier.address.cityName }],\n CountrySubentityCode: [{ _: invoice.supplier.address.state }],\n AddressLine: [\n {\n Line: [{ _: invoice.supplier.address.addressLine0 }],\n },\n ],\n Country: [\n {\n IdentificationCode: [\n {\n _: invoice.supplier.address.country || 'MYS',\n listID: '3166-1', // FIXED: was \"ISO3166-1\"\n listAgencyID: 'ISO',\n },\n ],\n },\n ],\n },\n ],\n\n // Party Legal Entity\n PartyLegalEntity: [\n {\n RegistrationName: [{ _: invoice.supplier.name }],\n },\n ],\n\n // Contact Information\n Contact: [\n {\n Telephone: [{ _: invoice.supplier.contactNumber || '' }],\n },\n ],\n },\n ],\n },\n ],\n\n // === BUYER PARTY (AccountingCustomerParty) ===\n AccountingCustomerParty: [\n {\n Party: [\n {\n // Party Identifications\n PartyIdentification: [\n {\n ID: [\n {\n _: invoice.buyer.tin,\n schemeID: 'TIN',\n },\n ],\n },\n {\n ID: [\n {\n _: formatIdValue(invoice.buyer.registrationNumber),\n schemeID: invoice.buyer.registrationType || 'NRIC',\n },\n ],\n },\n {\n ID: [\n {\n _: invoice.buyer.sstRegistrationNumber || 'NA',\n schemeID: 'SST',\n },\n ],\n },\n ],\n\n // Postal Address - FIXED listID format\n PostalAddress: [\n {\n CityName: [{ _: invoice.buyer.address.cityName }],\n CountrySubentityCode: [{ _: invoice.buyer.address.state }],\n AddressLine: [\n {\n Line: [{ _: invoice.buyer.address.addressLine0 }],\n },\n ],\n Country: [\n {\n IdentificationCode: [\n {\n _: invoice.buyer.address.country || 'MYS',\n listID: '3166-1', // FIXED: was \"ISO3166-1\"\n listAgencyID: 'ISO',\n },\n ],\n },\n ],\n },\n ],\n\n // Party Legal Entity\n PartyLegalEntity: [\n {\n RegistrationName: [{ _: invoice.buyer.name }],\n },\n ],\n\n // Contact Information\n Contact: [\n {\n Telephone: [{ _: invoice.buyer.contactNumber || '' }],\n },\n ],\n },\n ],\n },\n ],\n\n // === TAX TOTAL ===\n TaxTotal: [\n {\n TaxAmount: [\n {\n _: invoice.taxTotal.taxAmount,\n currencyID: invoice.invoiceCurrencyCode,\n },\n ],\n TaxSubtotal: [\n // Generate basic tax subtotal from invoice line items\n {\n TaxableAmount: [\n {\n _: invoice.legalMonetaryTotal.taxExclusiveAmount,\n currencyID: invoice.invoiceCurrencyCode,\n },\n ],\n TaxAmount: [\n {\n _: invoice.taxTotal.taxAmount,\n currencyID: invoice.invoiceCurrencyCode,\n },\n ],\n TaxCategory: [\n {\n ID: [{ _: lineItems[0]?.taxType || '01' }],\n TaxScheme: [\n {\n ID: [\n {\n _: 'OTH',\n schemeAgencyID: '6',\n schemeID: 'UN/ECE 5153',\n },\n ],\n },\n ],\n },\n ],\n },\n ],\n },\n ],\n\n // === LEGAL MONETARY TOTAL ===\n LegalMonetaryTotal: [\n {\n LineExtensionAmount: [\n {\n _: invoice.legalMonetaryTotal.taxExclusiveAmount,\n currencyID: invoice.invoiceCurrencyCode,\n },\n ],\n TaxExclusiveAmount: [\n {\n _: invoice.legalMonetaryTotal.taxExclusiveAmount,\n currencyID: invoice.invoiceCurrencyCode,\n },\n ],\n TaxInclusiveAmount: [\n {\n _: invoice.legalMonetaryTotal.taxInclusiveAmount,\n currencyID: invoice.invoiceCurrencyCode,\n },\n ],\n PayableAmount: [\n {\n _: invoice.legalMonetaryTotal.payableAmount,\n currencyID: invoice.invoiceCurrencyCode,\n },\n ],\n },\n ],\n\n // === INVOICE LINES ===\n InvoiceLine: lineItems.map((item, index) => ({\n ID: [{ _: (index + 1).toString() }],\n\n // Item Information\n Item: [\n {\n CommodityClassification: [\n {\n ItemClassificationCode: [\n {\n _: item.itemClassificationCode,\n listID: 'CLASS',\n },\n ],\n },\n ],\n Description: [{ _: item.itemDescription }],\n },\n ],\n\n ItemPriceExtension: [\n {\n Amount: [\n {\n _: item.totalTaxableAmountPerLine,\n currencyID: invoice.invoiceCurrencyCode,\n },\n ],\n },\n ],\n\n LineExtensionAmount: [\n {\n _: item.totalTaxableAmountPerLine,\n currencyID: invoice.invoiceCurrencyCode,\n },\n ],\n\n // Price Information\n Price: [\n {\n PriceAmount: [\n {\n _: item.unitPrice,\n currencyID: invoice.invoiceCurrencyCode,\n },\n ],\n },\n ],\n\n // Tax Information for line\n TaxTotal: [\n {\n TaxAmount: [\n {\n _: item.taxAmount,\n currencyID: invoice.invoiceCurrencyCode,\n },\n ],\n TaxSubtotal: [\n {\n TaxableAmount: [\n {\n _: item.totalTaxableAmountPerLine,\n currencyID: invoice.invoiceCurrencyCode,\n },\n ],\n TaxAmount: [\n {\n _: item.taxAmount,\n currencyID: invoice.invoiceCurrencyCode,\n },\n ],\n // Conditional tax fields based on taxation type\n ...(item.taxPerUnitAmount !== undefined &&\n item.baseUnitMeasure !== undefined\n ? {\n // Fixed Rate Taxation\n PerUnitAmount: [\n {\n _: item.taxPerUnitAmount,\n currencyID: invoice.invoiceCurrencyCode,\n },\n ],\n BaseUnitMeasure: [\n {\n _: item.baseUnitMeasure,\n unitCode: item.baseUnitMeasureCode || 'C62',\n },\n ],\n }\n : item.taxRate !== undefined\n ? {\n // Percentage Taxation\n Percent: [{ _: item.taxRate }],\n }\n : {}),\n TaxCategory: [\n {\n ID: [{ _: item.taxType }],\n TaxScheme: [\n {\n ID: [\n {\n _: 'OTH',\n schemeAgencyID: '6',\n schemeID: 'UN/ECE 5153',\n },\n ],\n },\n ],\n },\n ],\n },\n ],\n },\n ],\n })),\n\n // === TAX EXCHANGE RATE (mandatory where applicable) ===\n TaxExchangeRate: invoice.currencyExchangeRate\n ? [\n {\n SourceCurrencyCode: [\n {\n _: invoice.invoiceCurrencyCode,\n },\n ],\n TargetCurrencyCode: [\n {\n _: 'MYR',\n },\n ],\n CalculationRate: [\n {\n _: invoice.currencyExchangeRate,\n },\n ],\n },\n ]\n : undefined,\n }\n}\n\n/**\n * Generates the complete UBL document structure with namespace declarations\n */\nexport const generateCleanUBLDocument = (\n invoices: AllDocumentsV1_1[],\n): UBLDocument => {\n return {\n _D: 'urn:oasis:names:specification:ubl:schema:xsd:Invoice-2',\n _A: 'urn:oasis:names:specification:ubl:schema:xsd:CommonAggregateComponents-2',\n _B: 'urn:oasis:names:specification:ubl:schema:xsd:CommonBasicComponents-2',\n Invoice: invoices.map(generateCleanInvoiceObject),\n }\n}\n\n/**\n * Step 2: Calculate Document Digest\n * FIXED: Remove UBLExtensions and Signature before hashing (DS322)\n * Based on working implementation pattern\n */\nexport const calculateDocumentDigest = (\n invoices: AllDocumentsV1_1[],\n): string => {\n // Generate clean UBL document structure\n const cleanDocument = generateCleanUBLDocument(invoices)\n\n // CRITICAL FIX: Remove UBLExtensions and Signature from each invoice before hashing\n const documentForHashing = JSON.parse(JSON.stringify(cleanDocument))\n if (documentForHashing.Invoice && Array.isArray(documentForHashing.Invoice)) {\n documentForHashing.Invoice.forEach((invoice: any) => {\n delete invoice.UBLExtensions\n delete invoice.Signature\n })\n }\n\n // Convert to string for hashing (no canonicalization - use direct JSON.stringify)\n const documentString = JSON.stringify(documentForHashing)\n\n // Calculate SHA-256 hash\n const hash = crypto.createHash('sha256')\n hash.update(documentString, 'utf8')\n\n // Return as Base64 (DocDigest)\n return hash.digest('base64')\n}\n\n/**\n * Step 4: Calculate Certificate Digest\n * Enhanced to handle certificate content properly\n */\nexport const calculateCertificateDigest = (certificatePem: string): string => {\n // Extract certificate content (Base64 without PEM headers/footers)\n const certificateContent = certificatePem\n .replace(/-----BEGIN CERTIFICATE-----/g, '')\n .replace(/-----END CERTIFICATE-----/g, '')\n .replace(/\\s+/g, '') // Remove all whitespace\n\n // Convert Base64 to binary\n const certificateBinary = Buffer.from(certificateContent, 'base64')\n\n // Calculate SHA-256 hash of binary content\n const hash = crypto.createHash('sha256')\n hash.update(certificateBinary)\n\n // Return as Base64\n return hash.digest('base64')\n}\n\n/**\n * Enhanced certificate info extraction with better error handling\n * FIXED: Normalize issuer name format to match MyInvois expectations (DS326)\n */\nexport const extractCertificateInfo = (\n certificatePem: string,\n): {\n issuerName: string\n serialNumber: string\n subjectName: string\n} => {\n try {\n const cert = new X509Certificate(certificatePem)\n\n // Extract serial number and convert to decimal string\n const serialNumberHex = cert.serialNumber\n\n // ENHANCED: Normalize distinguished name format (DS326)\n // Ensures MyInvois portal compatibility for X509IssuerName matching\n const normalizeDistinguishedName = (dn: string): string => {\n // Convert multi-line format to comma-separated RFC2253 format\n const normalized = dn\n .split('\\n')\n .map(line => line.trim())\n .filter(line => line.length > 0)\n .join(', ')\n\n // MyInvois-specific normalization to prevent DS326 errors\n return normalized\n .replace(/\\s*=\\s*/g, '=') // Remove spaces around equals (CRITICAL for portal)\n .replace(/,\\s+/g, ', ') // Ensure single space after commas\n .replace(/\\r/g, '') // Remove any carriage returns\n .replace(/\\s{2,}/g, ' ') // Replace multiple spaces with single space\n .trim() // Remove leading/trailing whitespace\n }\n\n // Enhanced serial number formatting\n const formatSerialNumber = (serialHex: string): string => {\n // Convert hex to decimal and ensure it's a string\n const decimal = BigInt('0x' + serialHex).toString()\n return decimal\n }\n\n return {\n issuerName: normalizeDistinguishedName(cert.issuer),\n serialNumber: formatSerialNumber(serialNumberHex),\n subjectName: normalizeDistinguishedName(cert.subject),\n }\n } catch (error: unknown) {\n throw new Error(\n `Failed to extract certificate info: ${error instanceof Error ? error.message : String(error)}`,\n )\n }\n}\n\n/**\n * Step 5: Create SignedProperties with enhanced structure\n * FIXED: Simplified structure to match MyInvois expectations (DS320)\n * Following MyInvois JSON signature specification exactly\n */\nexport const createSignedProperties = (\n certificateDigest: string,\n signingTime: string,\n issuerName: string,\n serialNumber: string,\n): SignedPropertiesObject => {\n return {\n SignedProperties: [\n {\n Id: 'id-xades-signed-props',\n SignedSignatureProperties: [\n {\n SigningTime: [{ _: signingTime }],\n SigningCertificate: [\n {\n Cert: [\n {\n CertDigest: [\n {\n DigestMethod: [\n {\n Algorithm:\n 'http://www.w3.org/2001/04/xmlenc#sha256',\n },\n ],\n DigestValue: [{ _: certificateDigest }],\n },\n ],\n IssuerSerial: [\n {\n X509IssuerName: [{ _: issuerName }],\n X509SerialNumber: [{ _: serialNumber }],\n },\n ],\n },\n ],\n },\n ],\n },\n ],\n },\n ],\n }\n}\n\n/**\n * Step 6: Calculate SignedProperties Digest\n * FIXED: Add Target wrapper and use direct JSON stringify (DS320)\n * Based on working implementation pattern\n */\nexport const calculateSignedPropertiesDigest = (\n signedProperties: SignedPropertiesObject,\n): string => {\n // According to XAdES, the digest applies ONLY to the <SignedProperties> element\n // (the element whose Id equals 'id-xades-signed-props'). It must NOT include\n // the surrounding <QualifyingProperties> wrapper and its Target attribute.\n\n // Therefore, we hash the exact JSON representation that ends up inside\n // QualifyingProperties – i.e. the SignedProperties array structure itself.\n\n const signedPropertiesString = canonicalizeJSON(signedProperties)\n\n // Calculate SHA-256 hash\n const hash = crypto.createHash('sha256')\n hash.update(signedPropertiesString, 'utf8')\n\n // Return as Base64 (PropsDigest)\n return hash.digest('base64')\n}\n\n/**\n * Step 3: Create SignedInfo and calculate signature\n * Enhanced with better structure and signature generation\n */\nexport const createSignedInfoAndSign = (\n docDigest: string,\n propsDigest: string,\n privateKeyPem: string,\n): { signedInfo: SignedInfoObject; signatureValue: string } => {\n // Create SignedInfo structure following specification exactly\n const signedInfo = {\n CanonicalizationMethod: [\n {\n _: '',\n Algorithm: 'http://www.w3.org/2006/12/xml-c14n11',\n },\n ],\n SignatureMethod: [\n {\n _: '',\n Algorithm: 'http://www.w3.org/2001/04/xmldsig-more#rsa-sha256',\n },\n ],\n Reference: [\n {\n Id: 'id-doc-signed-data',\n Type: '',\n URI: '',\n DigestMethod: [\n {\n _: '',\n Algorithm: 'http://www.w3.org/2001/04/xmlenc#sha256',\n },\n ],\n DigestValue: [{ _: docDigest }],\n },\n {\n Id: 'id-xades-signed-props',\n Type: 'http://uri.etsi.org/01903/v1.3.2#SignedProperties',\n URI: '#id-xades-signed-props',\n DigestMethod: [\n {\n _: '',\n Algorithm: 'http://www.w3.org/2001/04/xmlenc#sha256',\n },\n ],\n DigestValue: [{ _: propsDigest }],\n },\n ],\n }\n\n // FIXED: Use direct JSON.stringify instead of canonicalization (DS333)\n // Based on working implementation pattern\n const signedInfoString = JSON.stringify(signedInfo)\n\n // Sign with RSA-SHA256 directly (RSA-SHA256 handles hashing internally)\n // FIXED: Removed double-hashing bug that was causing DS333 errors\n try {\n const signer = crypto.createSign('RSA-SHA256')\n signer.update(signedInfoString, 'utf8')\n const signatureValue = signer.sign(privateKeyPem, 'base64')\n\n return { signedInfo, signatureValue }\n } catch (error) {\n throw new Error(\n `Signature generation failed: ${error instanceof Error ? error.message : String(error)}`,\n )\n }\n}\n\n/**\n * Complete document generation with signatures\n * Follows the complete MyInvois JSON signature creation process (Steps 1-7)\n */\nexport const generateCompleteDocument = (\n invoices: AllDocumentsV1_1[],\n signingCredentials: SigningCredentials,\n): CompleteInvoice => {\n try {\n // Step 1: Generate clean document (done in calculateDocumentDigest)\n // Step 2: Calculate document digest\n const docDigest = calculateDocumentDigest(invoices)\n\n // Generate signing time in proper ISO format\n const signingTime = new Date().toISOString()\n\n // Extract certificate information (enhanced)\n const certInfo = extractCertificateInfo(signingCredentials.certificatePem)\n\n // Step 4: Calculate certificate digest\n const certificateDigest = calculateCertificateDigest(\n signingCredentials.certificatePem,\n )\n\n // Step 5: Create SignedProperties using extracted cert info\n const signedProperties = createSignedProperties(\n certificateDigest,\n signingTime,\n certInfo.issuerName,\n certInfo.serialNumber,\n )\n\n // Step 6: Calculate SignedProperties digest\n const propsDigest = calculateSignedPropertiesDigest(signedProperties)\n\n // Step 3: Create SignedInfo and generate signature\n const { signedInfo, signatureValue } = createSignedInfoAndSign(\n docDigest,\n propsDigest,\n signingCredentials.privateKeyPem,\n )\n\n // Extract certificate content (Base64 without PEM headers)\n const certificate = signingCredentials.certificatePem\n .replace(/-----BEGIN CERTIFICATE-----/g, '')\n .replace(/-----END CERTIFICATE-----/g, '')\n .replace(/\\s+/g, '')\n\n // Step 7: Create final signed document\n const signedInvoices = invoices.map(invoice => {\n const cleanInvoice = generateCleanInvoiceObject(invoice)\n\n return {\n ...cleanInvoice,\n\n // Add UBLExtensions with complete signature structure\n UBLExtensions: [\n {\n UBLExtension: [\n {\n ExtensionURI: [\n {\n _: 'urn:oasis:names:specification:ubl:dsig:enveloped:xades',\n },\n ],\n ExtensionContent: [\n {\n UBLDocumentSignatures: [\n {\n SignatureInformation: [\n {\n ID: [\n {\n _: 'urn:oasis:names:specification:ubl:signature:1',\n },\n ],\n ReferencedSignatureID: [\n {\n _: 'urn:oasis:names:specification:ubl:signature:Invoice',\n },\n ],\n Signature: [\n {\n Id: 'signature',\n Object: [\n {\n QualifyingProperties: [\n {\n Target: 'signature',\n SignedProperties:\n signedProperties.SignedProperties,\n },\n ],\n },\n ],\n KeyInfo: [\n {\n X509Data: [\n {\n X509Certificate: [{ _: certificate }],\n X509SubjectName: [\n { _: certInfo.subjectName },\n ],\n X509IssuerSerial: [\n {\n X509IssuerName: [\n {\n _: certInfo.issuerName,\n },\n ],\n X509SerialNumber: [\n {\n _: certInfo.serialNumber,\n },\n ],\n },\n ],\n },\n ],\n },\n ],\n SignatureValue: [{ _: signatureValue }],\n SignedInfo: [signedInfo],\n },\n ],\n },\n ],\n },\n ],\n },\n ],\n },\n ],\n },\n ],\n\n // Add simple Signature reference\n Signature: [\n {\n ID: [\n {\n _: 'urn:oasis:names:specification:ubl:signature:Invoice',\n },\n ],\n SignatureMethod: [\n {\n _: 'urn:oasis:names:specification:ubl:dsig:enveloped:xades',\n },\n ],\n },\n ],\n }\n })\n\n return {\n _D: 'urn:oasis:names:specification:ubl:schema:xsd:Invoice-2',\n _A: 'urn:oasis:names:specification:ubl:schema:xsd:CommonAggregateComponents-2',\n _B: 'urn:oasis:names:specification:ubl:schema:xsd:CommonBasicComponents-2',\n Invoice: signedInvoices,\n }\n } catch (error) {\n throw new Error(\n `Document generation failed: ${error instanceof Error ? error.message : String(error)}`,\n )\n }\n}\n\n/**\n * Creates a line item with percentage-based taxation (e.g., SST, GST)\n */\nexport const createPercentageTaxLineItem = (params: {\n itemClassificationCode: ClassificationCode\n itemDescription: string\n unitPrice: number\n quantity?: number\n taxType: TaxTypeCode\n taxRate: number\n totalTaxableAmountPerLine?: number\n}): InvoiceLineItem => {\n const quantity = params.quantity || 1\n const totalTaxableAmount =\n params.totalTaxableAmountPerLine || params.unitPrice * quantity\n const taxAmount = (totalTaxableAmount * params.taxRate) / 100\n\n return {\n itemClassificationCode: params.itemClassificationCode,\n itemDescription: params.itemDescription,\n unitPrice: params.unitPrice,\n taxType: params.taxType,\n taxRate: params.taxRate,\n taxAmount: Math.round(taxAmount * 100) / 100, // Round to 2 decimal places\n totalTaxableAmountPerLine: totalTaxableAmount,\n totalAmountPerLine: totalTaxableAmount + taxAmount,\n }\n}\n\n/**\n * Creates a line item with fixed rate taxation (e.g., Tourism Tax)\n */\nexport const createFixedRateTaxLineItem = (params: {\n itemClassificationCode: ClassificationCode\n itemDescription: string\n unitPrice: number\n quantity?: number\n taxType: TaxTypeCode\n taxPerUnitAmount: number\n baseUnitMeasure: number\n baseUnitMeasureCode: UnitTypeCode\n totalTaxableAmountPerLine?: number\n}): InvoiceLineItem => {\n const quantity = params.quantity || 1\n const totalTaxableAmount =\n params.totalTaxableAmountPerLine || params.unitPrice * quantity\n const taxAmount = params.taxPerUnitAmount * params.baseUnitMeasure\n\n return {\n itemClassificationCode: params.itemClassificationCode,\n itemDescription: params.itemDescription,\n unitPrice: params.unitPrice,\n taxType: params.taxType,\n taxPerUnitAmount: params.taxPerUnitAmount,\n baseUnitMeasure: params.baseUnitMeasure,\n baseUnitMeasureCode: params.baseUnitMeasureCode,\n taxAmount: Math.round(taxAmount * 100) / 100, // Round to 2 decimal places\n totalTaxableAmountPerLine: totalTaxableAmount,\n totalAmountPerLine: totalTaxableAmount + taxAmount,\n }\n}\n\n/**\n * Calculates invoice totals from line items\n */\nexport const calculateInvoiceTotals = (\n lineItems: InvoiceLineItem[],\n): {\n legalMonetaryTotal: {\n taxExclusiveAmount: number\n taxInclusiveAmount: number\n payableAmount: number\n }\n taxTotal: {\n taxAmount: number\n }\n} => {\n const taxExclusiveAmount = lineItems.reduce(\n (sum, item) => sum + item.totalTaxableAmountPerLine,\n 0,\n )\n const totalTaxAmount = lineItems.reduce(\n (sum, item) => sum + item.taxAmount,\n 0,\n )\n const taxInclusiveAmount = taxExclusiveAmount + totalTaxAmount\n\n return {\n legalMonetaryTotal: {\n taxExclusiveAmount: Math.round(taxExclusiveAmount * 100) / 100,\n taxInclusiveAmount: Math.round(taxInclusiveAmount * 100) / 100,\n payableAmount: Math.round(taxInclusiveAmount * 100) / 100,\n },\n taxTotal: {\n taxAmount: Math.round(totalTaxAmount * 100) / 100,\n },\n }\n}\n"],"mappings":"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"}
@@ -1 +0,0 @@
1
- {"version":3,"file":"documentSubmission-CWscnXdN.cjs","names":["context: SubmissionContext","documents: AllDocumentsV1_1[]","context: Pick<SubmissionContext, 'fetch' | 'debug'>","submissionUid: string","pollInterval: number","maxRetries: number","documentUid: string","status: 'rejected' | 'cancelled'","reason: string"],"sources":["../src/api/documentSubmission.ts"],"sourcesContent":["import type {\n AllDocumentsV1_1,\n SubmissionResponse,\n SigningCredentials,\n SubmissionStatus,\n DocumentSummary,\n Fetch,\n StandardError,\n} from '../types'\nimport { generateCompleteDocument } from '../utils/document'\n\ninterface SubmissionContext {\n fetch: Fetch\n debug: boolean\n signingCredentials: SigningCredentials\n}\n\nexport async function submitDocument(\n context: SubmissionContext,\n documents: AllDocumentsV1_1[],\n): Promise<{\n data: SubmissionResponse\n status: number\n}> {\n const { fetch, debug, signingCredentials } = context\n\n // 🔒 Hard enforcement of platform submission limits\n if (documents.length > 100) {\n throw new Error(\n 'Submission rejected: Cannot submit more than 100 documents at once',\n )\n }\n\n if (debug) {\n console.log(`📦 Preparing to submit ${documents.length} document(s)...`)\n }\n\n // For batch submission, each document must be signed and encoded separately\n // Build the submission payload according to MyInvois API format\n const crypto = await import('crypto')\n\n const submissionPayload = {\n documents: await Promise.all(\n documents.map(async doc => {\n // 1️⃣ Sign the single document (generateCompleteDocument expects an array)\n const signedDocument = generateCompleteDocument(\n [doc],\n signingCredentials,\n )\n\n // 2️⃣ Serialize\n const docJson = JSON.stringify(signedDocument)\n\n // 3️⃣ Hash\n const docHash = crypto\n .createHash('sha256')\n .update(docJson, 'utf8')\n .digest('hex')\n\n // 4️⃣ Base64 encode\n const docBase64 = Buffer.from(docJson, 'utf8').toString('base64')\n\n // 🚨 Enforce 300 KB per-document limit\n const rawSize = Buffer.byteLength(docBase64, 'base64')\n if (rawSize > 300 * 1024) {\n throw new Error(\n `Submission rejected: Document ${doc.eInvoiceCodeOrNumber} is ${rawSize} bytes – exceeds 300KB limit`,\n )\n }\n\n if (debug) {\n console.log('—'.repeat(60))\n console.log(`📄 Prepared document: ${doc.eInvoiceCodeOrNumber}`)\n console.log(` • JSON size : ${docJson.length} bytes`)\n console.log(` • Base64 size: ${docBase64.length} bytes`)\n }\n\n return {\n format: 'JSON',\n document: docBase64,\n documentHash: docHash,\n codeNumber: doc.eInvoiceCodeOrNumber,\n }\n }),\n ),\n }\n\n const payloadSize = Buffer.byteLength(JSON.stringify(submissionPayload))\n\n if (payloadSize > 5 * 1024 * 1024) {\n throw new Error(\n `Submission rejected: Payload is ${payloadSize} bytes – exceeds 5MB limit`,\n )\n }\n\n if (debug) {\n console.log('🚀 Submission payload structure:')\n console.log('- Format: JSON')\n console.log('- Documents count:', submissionPayload.documents.length)\n console.log('- Total payload size:', payloadSize, 'bytes')\n }\n\n // Submit to MyInvois API with proper headers\n const response = await fetch('/api/v1.0/documentsubmissions', {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n },\n body: JSON.stringify(submissionPayload),\n })\n\n const responseData = (await response.json()) as SubmissionResponse\n\n if (debug) {\n console.log(`📡 API Response status: ${response.status}`)\n\n if (responseData.rejectedDocuments?.length > 0) {\n responseData.rejectedDocuments.forEach((doc, index) => {\n console.log(` Document ${index + 1}:`, doc.invoiceCodeNumber)\n if (doc.error) {\n console.log(` Error:`, doc.error.message)\n if (doc.error.details) {\n doc.error.details.forEach((detail, detailIndex) => {\n console.log(` Detail ${detailIndex + 1}:`, detail.message)\n })\n }\n }\n })\n }\n }\n\n const data = responseData as SubmissionResponse\n\n if (debug) {\n if (response.status !== 202) {\n console.error('❌ Submission failed with status:', response.status)\n console.error('❌ Response data:', JSON.stringify(data, null, 2))\n } else {\n console.log('✅ Submission successful!')\n console.log(`📋 Submission UID: ${data.submissionUid}`)\n console.log(\n `✅ Accepted documents: ${data.acceptedDocuments?.length || 0}`,\n )\n console.log(\n `❌ Rejected documents: ${data.rejectedDocuments?.length || 0}`,\n )\n }\n }\n\n return {\n data,\n status: response.status,\n }\n}\n\nexport async function getSubmissionStatus(\n context: Pick<SubmissionContext, 'fetch' | 'debug'>,\n submissionUid: string,\n pollInterval: number = 1000,\n maxRetries: number = 10,\n): Promise<{\n status: SubmissionStatus\n documentSummary?: DocumentSummary[]\n error?: {\n code: string\n message: string | null\n target: string\n details: {\n code: string\n message: string\n target: string\n }[]\n }\n}> {\n const { fetch, debug } = context\n\n try {\n const response = await fetch(\n `/api/v1.0/documentsubmissions/${submissionUid}`,\n )\n\n const data = await response.json()\n\n if (debug) {\n console.log('Submission:', data)\n if (data.error) {\n console.log('Submission error details:', data.error.details)\n }\n }\n\n // If we have a successful response and status is completed, return success\n if (data.overallStatus === 'Valid') {\n return {\n status: data.overallStatus,\n documentSummary: data.documentSummary,\n }\n }\n if (data.overallStatus === 'Invalid') {\n return {\n status: 'Invalid',\n documentSummary: data.documentSummary,\n }\n }\n\n // If we have retries left, continue polling for any non-completed status or errors\n if (maxRetries > 0) {\n await new Promise(resolve => setTimeout(resolve, pollInterval))\n return await getSubmissionStatus(\n context,\n submissionUid,\n pollInterval,\n maxRetries - 1,\n )\n }\n\n // No retries left - return timeout\n return {\n status: 'TimedOut',\n documentSummary: data.documentSummary,\n error: {\n code: 'Timeout',\n message: 'Submission timed out',\n target: 'submission',\n details: [],\n },\n }\n } catch (error) {\n // Handle any request errors by retrying if we have retries left\n if (maxRetries > 0) {\n if (debug) {\n console.log('Request error, retrying...', error)\n }\n await new Promise(resolve => setTimeout(resolve, pollInterval))\n return await getSubmissionStatus(\n context,\n submissionUid,\n pollInterval,\n maxRetries - 1,\n )\n }\n\n // No retries left - return timeout\n return {\n status: 'TimedOut',\n documentSummary: [],\n error: {\n code: 'Timeout',\n message: 'Submission timed out after request errors',\n target: 'submission',\n details: [],\n },\n }\n }\n}\n\nexport async function performDocumentAction(\n documentUid: string,\n status: 'rejected' | 'cancelled',\n reason: string,\n): Promise<{\n uuid: string\n status: string\n error: StandardError\n}> {\n const response = await fetch(\n `/api/v1.0/documents/state/${documentUid}/state`,\n {\n method: 'POST',\n body: JSON.stringify({\n status,\n reason,\n }),\n },\n )\n\n const data = (await response.json()) as {\n uuid: string\n status: string\n error: StandardError\n }\n\n return data\n}\n"],"mappings":"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"}