@ripwords/myinvois-client 0.1.4 → 0.1.5
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.
- package/CHANGELOG.md +13 -0
- package/bun.lock +460 -0
- package/dist/0X-CTIq3y3a.d.ts +210 -0
- package/dist/0X-Cr3M7hci.d.cts +211 -0
- package/dist/1X-Bu4oiv8D.d.ts +111 -0
- package/dist/1X-BywXCqtn.d.ts +19 -0
- package/dist/1X-CWwmfCzo.d.cts +112 -0
- package/dist/1X-ClnJ79HH.d.cts +20 -0
- package/dist/2X-3fSEGIuE.d.ts +39 -0
- package/dist/2X-CV9eYhog.d.cts +40 -0
- package/dist/2X-DNtkJ0tj.d.cts +140 -0
- package/dist/2X-DnG3FY1f.d.ts +139 -0
- package/dist/3X--VWVsC84.d.cts +21 -0
- package/dist/3X-BOxfatu3.d.cts +62 -0
- package/dist/3X-CLdmoel1.d.ts +61 -0
- package/dist/3X-Ynjfgoll.d.ts +20 -0
- package/dist/4X-B5ePdMky.d.ts +30 -0
- package/dist/4X-BTG6o1Gn.d.ts +264 -0
- package/dist/4X-By9PzHaY.d.cts +31 -0
- package/dist/4X-C7fzDWJ_.d.cts +265 -0
- package/dist/5X-B5M0Cv_K.d.cts +22 -0
- package/dist/5X-CNAFsDm2.d.cts +77 -0
- package/dist/5X-CjSz1zxJ.d.ts +76 -0
- package/dist/5X-dhP6fyou.d.ts +21 -0
- package/dist/6X-B9KP_vEn.d.ts +104 -0
- package/dist/6X-BHaY0TCf.d.ts +15 -0
- package/dist/6X-C3elgd-n.d.cts +16 -0
- package/dist/6X-uObUP4VG.d.cts +105 -0
- package/dist/7X-BifJnY24.d.cts +71 -0
- package/dist/7X-C4eX_tAk.d.ts +15 -0
- package/dist/7X-D3idQBl9.d.cts +16 -0
- package/dist/7X-DtkQutx2.d.ts +70 -0
- package/dist/8X-5n9seY3z.d.ts +109 -0
- package/dist/8X-C6UMFvQW.d.cts +19 -0
- package/dist/8X-D6HTLShY.d.ts +18 -0
- package/dist/8X-DioBXCJ0.d.cts +110 -0
- package/dist/9X-BjffnXuq.d.cts +98 -0
- package/dist/9X-CHZKsIdD.d.ts +14 -0
- package/dist/9X-CJjPJgIG.d.cts +15 -0
- package/dist/9X-Cpb6V4JC.d.ts +97 -0
- package/dist/AX-C0w_r3PA.d.ts +110 -0
- package/dist/AX-DrEd2Ov6.d.cts +111 -0
- package/dist/BX-C5jc6myN.d.cts +115 -0
- package/dist/BX-RadlZ-6m.d.ts +114 -0
- package/dist/CX-CT1Zzb9D.d.ts +127 -0
- package/dist/CX-oSgvmn3h.d.cts +128 -0
- package/dist/DX-CtUeTKMM.d.ts +114 -0
- package/dist/DX-ZIG0enmK.d.cts +115 -0
- package/dist/EX-084Yu7Wt.d.ts +106 -0
- package/dist/EX-BpWZ5ADq.d.cts +107 -0
- package/dist/FX-BrbQM0CW.d.ts +126 -0
- package/dist/FX-CvYBEc51.d.cts +127 -0
- package/dist/GX-CU8PZy0G.d.cts +136 -0
- package/dist/GX-gKj8iAAh.d.ts +135 -0
- package/dist/HX-9D-ZXTy5.d.ts +125 -0
- package/dist/HX-DoEErDeE.d.cts +126 -0
- package/dist/IX-DR4Fc92A.d.ts +22 -0
- package/dist/IX-jLAUHe8i.d.cts +23 -0
- package/dist/JX-DIKa9ma-.d.ts +103 -0
- package/dist/JX-DX8BjYQC.d.cts +104 -0
- package/dist/KX-DpsJ_unT.d.ts +150 -0
- package/dist/KX-fZb4_vhw.d.cts +151 -0
- package/dist/LX-D_mI1bk-.d.cts +123 -0
- package/dist/LX-DeXPPWUX.d.ts +122 -0
- package/dist/MX-CKMjg_zK.d.ts +152 -0
- package/dist/MX-DlYZhpkT.d.cts +153 -0
- package/dist/NX-DELEEbad.d.cts +122 -0
- package/dist/NX-oYdb4NDo.d.ts +121 -0
- package/dist/OX-Bid9es3D.d.ts +25 -0
- package/dist/OX-CKPyrIFf.d.cts +26 -0
- package/dist/PX-D7jy9CFk.d.ts +120 -0
- package/dist/PX-SP6NxBtG.d.cts +121 -0
- package/dist/QX-BFWOR4g3.d.ts +58 -0
- package/dist/QX-QZA7E2O-.d.cts +59 -0
- package/dist/RX-DBcgbdeE.d.cts +23 -0
- package/dist/RX-XHgCyMV1.d.ts +22 -0
- package/dist/SX-Cl8RN8oH.d.ts +39 -0
- package/dist/SX-DmiJeCMN.d.cts +40 -0
- package/dist/TX-C1SoDobK.d.cts +31 -0
- package/dist/TX-ZymoIbtr.d.ts +30 -0
- package/dist/UX-D-Ndd1ov.d.ts +17 -0
- package/dist/UX-DFVynFhQ.d.cts +18 -0
- package/dist/VX-B-1b34r7.d.ts +16 -0
- package/dist/VX-DsRum5k3.d.cts +17 -0
- package/dist/WX-0GuMjPNZ.d.ts +25 -0
- package/dist/WX-VReU5R5M.d.cts +26 -0
- package/dist/XX-CF2NANsy.d.ts +827 -0
- package/dist/XX-ChAviUMj.d.cts +828 -0
- package/dist/YX-B51BbNSg.d.ts +19 -0
- package/dist/YX-QfzeMEKx.d.cts +20 -0
- package/dist/ZX-BMJS1iAv.d.ts +21 -0
- package/dist/ZX-Ca2E726a.d.cts +22 -0
- package/dist/api/platform/platformLogin.d.ts +65 -0
- package/dist/api/platform/platformLogin.js +3 -0
- package/dist/certificate-DFK-788s.cjs +62 -0
- package/dist/certificate-DFK-788s.cjs.map +1 -0
- package/dist/certificate-aooIRf9A.js +49 -0
- package/dist/chunk-CUT6urMc.cjs +30 -0
- package/dist/classification-codes-B15PbWxz.d.cts +118 -0
- package/dist/classification-codes-BKxV-rO9.d.ts +117 -0
- package/dist/country-code-CQuaiQm2.d.ts +542 -0
- package/dist/country-code-DPeNFMMi.d.cts +543 -0
- package/dist/currencies-BYJK-m6v.d.ts +207 -0
- package/dist/currencies-S5g1gzBU.d.cts +208 -0
- package/dist/document-CARHiGdp.cjs +472 -0
- package/dist/document-CARHiGdp.cjs.map +1 -0
- package/dist/document-D6VKMAtx.js +405 -0
- package/dist/documents-CQy_uB6C.d.cts +858 -0
- package/dist/documents-i5EV868Y.d.ts +857 -0
- package/dist/e-invoice-BuwtFnlI.d.cts +44 -0
- package/dist/e-invoice-CmbLQkHw.d.ts +43 -0
- package/dist/getBaseUrl-CO7Jp27d.cjs +14 -0
- package/dist/getBaseUrl-CO7Jp27d.cjs.map +1 -0
- package/dist/getBaseUrl-R3IdgCu3.js +7 -0
- package/dist/index-0-EvC6Nv.d.ts +15 -0
- package/dist/index-D2_HVwCz.d.cts +16 -0
- package/dist/index.cjs +309 -0
- package/dist/index.cjs.map +1 -0
- package/dist/index.d.ts +233 -0
- package/dist/index.js +308 -0
- package/dist/index10.cjs +0 -0
- package/dist/index11.cjs +34 -0
- package/dist/index11.cjs.map +1 -0
- package/dist/index12.cjs +24 -0
- package/dist/index12.cjs.map +1 -0
- package/dist/index13.cjs +0 -0
- package/dist/index14.cjs +13 -0
- package/dist/index14.cjs.map +1 -0
- package/dist/index15.cjs +4 -0
- package/dist/index16.cjs +13 -0
- package/dist/index17.cjs +3 -0
- package/dist/index18.cjs +340 -0
- package/dist/index18.cjs.map +1 -0
- package/dist/index19.cjs +329 -0
- package/dist/index19.cjs.map +1 -0
- package/dist/index2.cjs +62 -0
- package/dist/index2.cjs.map +1 -0
- package/dist/index20.cjs +140 -0
- package/dist/index20.cjs.map +1 -0
- package/dist/index21.cjs +3 -0
- package/dist/index22.cjs +208 -0
- package/dist/index22.cjs.map +1 -0
- package/dist/index23.cjs +109 -0
- package/dist/index23.cjs.map +1 -0
- package/dist/index24.cjs +137 -0
- package/dist/index24.cjs.map +1 -0
- package/dist/index25.cjs +64 -0
- package/dist/index25.cjs.map +1 -0
- package/dist/index26.cjs +267 -0
- package/dist/index26.cjs.map +1 -0
- package/dist/index27.cjs +79 -0
- package/dist/index27.cjs.map +1 -0
- package/dist/index28.cjs +107 -0
- package/dist/index28.cjs.map +1 -0
- package/dist/index29.cjs +73 -0
- package/dist/index29.cjs.map +1 -0
- package/dist/index3.cjs +532 -0
- package/dist/index3.cjs.map +1 -0
- package/dist/index30.cjs +112 -0
- package/dist/index30.cjs.map +1 -0
- package/dist/index31.cjs +100 -0
- package/dist/index31.cjs.map +1 -0
- package/dist/index32.cjs +18 -0
- package/dist/index32.cjs.map +1 -0
- package/dist/index33.cjs +38 -0
- package/dist/index33.cjs.map +1 -0
- package/dist/index34.cjs +19 -0
- package/dist/index34.cjs.map +1 -0
- package/dist/index35.cjs +29 -0
- package/dist/index35.cjs.map +1 -0
- package/dist/index36.cjs +20 -0
- package/dist/index36.cjs.map +1 -0
- package/dist/index37.cjs +14 -0
- package/dist/index37.cjs.map +1 -0
- package/dist/index38.cjs +14 -0
- package/dist/index38.cjs.map +1 -0
- package/dist/index39.cjs +17 -0
- package/dist/index39.cjs.map +1 -0
- package/dist/index4.cjs +196 -0
- package/dist/index4.cjs.map +1 -0
- package/dist/index40.cjs +13 -0
- package/dist/index40.cjs.map +1 -0
- package/dist/index41.cjs +108 -0
- package/dist/index41.cjs.map +1 -0
- package/dist/index42.cjs +113 -0
- package/dist/index42.cjs.map +1 -0
- package/dist/index43.cjs +126 -0
- package/dist/index43.cjs.map +1 -0
- package/dist/index44.cjs +113 -0
- package/dist/index44.cjs.map +1 -0
- package/dist/index45.cjs +105 -0
- package/dist/index45.cjs.map +1 -0
- package/dist/index46.cjs +125 -0
- package/dist/index46.cjs.map +1 -0
- package/dist/index47.cjs +134 -0
- package/dist/index47.cjs.map +1 -0
- package/dist/index48.cjs +124 -0
- package/dist/index48.cjs.map +1 -0
- package/dist/index49.cjs +21 -0
- package/dist/index49.cjs.map +1 -0
- package/dist/index5.cjs +0 -0
- package/dist/index50.cjs +102 -0
- package/dist/index50.cjs.map +1 -0
- package/dist/index51.cjs +149 -0
- package/dist/index51.cjs.map +1 -0
- package/dist/index52.cjs +121 -0
- package/dist/index52.cjs.map +1 -0
- package/dist/index53.cjs +151 -0
- package/dist/index53.cjs.map +1 -0
- package/dist/index54.cjs +120 -0
- package/dist/index54.cjs.map +1 -0
- package/dist/index55.cjs +24 -0
- package/dist/index55.cjs.map +1 -0
- package/dist/index56.cjs +119 -0
- package/dist/index56.cjs.map +1 -0
- package/dist/index57.cjs +54 -0
- package/dist/index57.cjs.map +1 -0
- package/dist/index58.cjs +21 -0
- package/dist/index58.cjs.map +1 -0
- package/dist/index58.cts.map +1 -0
- package/dist/index59.cjs +35 -0
- package/dist/index59.cjs.map +1 -0
- package/dist/index59.cts.map +1 -0
- package/dist/index6.cjs +25 -0
- package/dist/index6.cjs.map +1 -0
- package/dist/index60.cjs +29 -0
- package/dist/index60.cjs.map +1 -0
- package/dist/index60.cts.map +1 -0
- package/dist/index61.cjs +16 -0
- package/dist/index61.cjs.map +1 -0
- package/dist/index61.cts.map +1 -0
- package/dist/index62.cjs +15 -0
- package/dist/index62.cjs.map +1 -0
- package/dist/index62.cts.map +1 -0
- package/dist/index63.cjs +24 -0
- package/dist/index63.cjs.map +1 -0
- package/dist/index63.cts.map +1 -0
- package/dist/index64.cjs +419 -0
- package/dist/index64.cjs.map +1 -0
- package/dist/index64.cts.map +1 -0
- package/dist/index65.cjs +15 -0
- package/dist/index65.cjs.map +1 -0
- package/dist/index65.cts.map +1 -0
- package/dist/index66.cjs +16 -0
- package/dist/index66.cjs.map +1 -0
- package/dist/index66.cts.map +1 -0
- package/dist/index7.cjs +0 -0
- package/dist/index8.cjs +0 -0
- package/dist/index9.cjs +25 -0
- package/dist/index9.cjs.map +1 -0
- package/dist/msic-codes-C8PJVOaA.d.ts +25 -0
- package/dist/msic-codes-CIKdPqag.d.cts +26 -0
- package/dist/payment-modes-BOTSRC_X.d.ts +43 -0
- package/dist/payment-modes-ghFEXnUl.d.cts +44 -0
- package/dist/platformLogin-PGzMhw1X.cjs +37 -0
- package/dist/platformLogin-PGzMhw1X.cjs.map +1 -0
- package/dist/platformLogin-f0bNAoZI.js +30 -0
- package/dist/signatures-W-_brwdb.d.ts +172 -0
- package/dist/signatures-hFbt_std.d.cts +173 -0
- package/dist/state-codes-BlILGZ9d.d.cts +62 -0
- package/dist/state-codes-oeFKlwXZ.d.ts +61 -0
- package/dist/tax-types-B5sQ8UyN.d.cts +42 -0
- package/dist/tax-types-CsQ76JMf.d.ts +41 -0
- package/dist/types/classification-codes.d.ts +2 -0
- package/dist/types/country-code.d.ts +2 -0
- package/dist/types/currencies.d.ts +2 -0
- package/dist/types/documents.d.ts +18 -0
- package/dist/types/e-invoice.d.ts +2 -0
- package/dist/types/index.d.ts +58 -0
- package/dist/types/msic/0X.d.ts +2 -0
- package/dist/types/msic/1X.d.ts +2 -0
- package/dist/types/msic/2X.d.ts +2 -0
- package/dist/types/msic/3X.d.ts +2 -0
- package/dist/types/msic/4X.d.ts +2 -0
- package/dist/types/msic/5X.d.ts +2 -0
- package/dist/types/msic/6X.d.ts +2 -0
- package/dist/types/msic/7X.d.ts +2 -0
- package/dist/types/msic/8X.d.ts +2 -0
- package/dist/types/msic/9X.d.ts +2 -0
- package/dist/types/msic-codes.d.ts +12 -0
- package/dist/types/payment-modes.d.ts +2 -0
- package/dist/types/signatures.d.ts +2 -0
- package/dist/types/state-codes.d.ts +2 -0
- package/dist/types/tax-types.d.ts +2 -0
- package/dist/types/unit/1X.d.ts +2 -0
- package/dist/types/unit/2X.d.ts +2 -0
- package/dist/types/unit/3X.d.ts +2 -0
- package/dist/types/unit/4X.d.ts +2 -0
- package/dist/types/unit/5X.d.ts +2 -0
- package/dist/types/unit/6X.d.ts +2 -0
- package/dist/types/unit/7X.d.ts +2 -0
- package/dist/types/unit/8X.d.ts +2 -0
- package/dist/types/unit/9X.d.ts +2 -0
- package/dist/types/unit/AX.d.ts +2 -0
- package/dist/types/unit/BX.d.ts +2 -0
- package/dist/types/unit/CX.d.ts +2 -0
- package/dist/types/unit/DX.d.ts +2 -0
- package/dist/types/unit/EX.d.ts +2 -0
- package/dist/types/unit/FX.d.ts +2 -0
- package/dist/types/unit/GX.d.ts +2 -0
- package/dist/types/unit/HX.d.ts +2 -0
- package/dist/types/unit/IX.d.ts +2 -0
- package/dist/types/unit/JX.d.ts +2 -0
- package/dist/types/unit/KX.d.ts +2 -0
- package/dist/types/unit/LX.d.ts +2 -0
- package/dist/types/unit/MX.d.ts +2 -0
- package/dist/types/unit/NX.d.ts +2 -0
- package/dist/types/unit/OX.d.ts +2 -0
- package/dist/types/unit/PX.d.ts +2 -0
- package/dist/types/unit/QX.d.ts +2 -0
- package/dist/types/unit/RX.d.ts +2 -0
- package/dist/types/unit/SX.d.ts +2 -0
- package/dist/types/unit/TX.d.ts +2 -0
- package/dist/types/unit/UX.d.ts +2 -0
- package/dist/types/unit/VX.d.ts +2 -0
- package/dist/types/unit/WX.d.ts +2 -0
- package/dist/types/unit/XX.d.ts +2 -0
- package/dist/types/unit/YX.d.ts +2 -0
- package/dist/types/unit/ZX.d.ts +2 -0
- package/dist/types/unit-types.d.ts +37 -0
- package/dist/unit-types-B41YFOLT.d.ts +55 -0
- package/dist/unit-types-CjWNk2wS.d.cts +56 -0
- package/dist/utils/base64.d.ts +5 -0
- package/dist/utils/base64.js +10 -0
- package/dist/utils/certificate.d.ts +22 -0
- package/dist/utils/certificate.js +3 -0
- package/dist/utils/document.d.ts +135 -0
- package/dist/utils/document.js +3 -0
- package/dist/utils/getBaseUrl.d.ts +4 -0
- package/dist/utils/getBaseUrl.js +3 -0
- package/dist/utils/helpers.d.ts +271 -0
- package/dist/utils/helpers.js +330 -0
- package/dist/utils/signature-diagnostics.d.ts +93 -0
- package/dist/utils/signature-diagnostics.js +326 -0
- package/dist/utils/validation.d.ts +46 -0
- package/dist/utils/validation.js +134 -0
- package/package.json +4 -4
|
@@ -0,0 +1,405 @@
|
|
|
1
|
+
import crypto, { X509Certificate } from "crypto";
|
|
2
|
+
|
|
3
|
+
//#region src/utils/document.ts
|
|
4
|
+
/**
|
|
5
|
+
* MyInvois v1.1 Document Generation and Signing Utilities
|
|
6
|
+
* Strictly follows: https://sdk.myinvois.hasil.gov.my/documents/invoice-v1-1
|
|
7
|
+
* JSON Signature Guide: https://sdk.myinvois.hasil.gov.my/signature-creation-json/
|
|
8
|
+
*/
|
|
9
|
+
/**
|
|
10
|
+
* Helper function to recursively sort object keys for JSON canonicalization
|
|
11
|
+
*/
|
|
12
|
+
function sortObjectKeys(obj) {
|
|
13
|
+
if (obj === null || typeof obj !== "object") return obj;
|
|
14
|
+
if (Array.isArray(obj)) return obj.map(sortObjectKeys);
|
|
15
|
+
const sortedObj = {};
|
|
16
|
+
const keys = Object.keys(obj).sort();
|
|
17
|
+
for (const key of keys) sortedObj[key] = sortObjectKeys(obj[key]);
|
|
18
|
+
return sortedObj;
|
|
19
|
+
}
|
|
20
|
+
/**
|
|
21
|
+
* Enhanced canonicalization following MyInvois specification exactly
|
|
22
|
+
* Key changes: ensure consistent ordering and formatting
|
|
23
|
+
*/
|
|
24
|
+
const canonicalizeJSON = (obj) => {
|
|
25
|
+
const sortedObj = sortObjectKeys(obj);
|
|
26
|
+
return JSON.stringify(sortedObj, null, 0);
|
|
27
|
+
};
|
|
28
|
+
/**
|
|
29
|
+
* Generates a clean invoice object following MyInvois v1.1 specification exactly
|
|
30
|
+
* This is the base invoice structure WITHOUT signature elements (for hash calculation)
|
|
31
|
+
*
|
|
32
|
+
* Key requirements from working documents:
|
|
33
|
+
* - All mandatory fields must be present
|
|
34
|
+
* - Many optional fields must be present even if empty
|
|
35
|
+
* - Specific field ordering and structure
|
|
36
|
+
* - Correct listID values (e.g., "3166-1" not "ISO3166-1")
|
|
37
|
+
*/
|
|
38
|
+
const generateCleanInvoiceObject = (invoice) => {
|
|
39
|
+
return {
|
|
40
|
+
ID: [{ _: invoice.eInvoiceCodeOrNumber }],
|
|
41
|
+
IssueDate: [{ _: invoice.eInvoiceDate }],
|
|
42
|
+
IssueTime: [{ _: invoice.eInvoiceTime }],
|
|
43
|
+
InvoiceTypeCode: [{
|
|
44
|
+
_: invoice.eInvoiceTypeCode,
|
|
45
|
+
listVersionID: invoice.eInvoiceVersion || "1.1"
|
|
46
|
+
}],
|
|
47
|
+
DocumentCurrencyCode: [{ _: invoice.invoiceCurrencyCode }],
|
|
48
|
+
AccountingSupplierParty: [{ Party: [{
|
|
49
|
+
IndustryClassificationCode: [{
|
|
50
|
+
_: invoice.supplier.industryClassificationCode,
|
|
51
|
+
name: invoice.supplier.industryClassificationDescription
|
|
52
|
+
}],
|
|
53
|
+
PartyIdentification: [{ ID: [{
|
|
54
|
+
_: invoice.supplier.tin,
|
|
55
|
+
schemeID: "TIN"
|
|
56
|
+
}] }, { ID: [{
|
|
57
|
+
_: invoice.supplier.registrationNumber,
|
|
58
|
+
schemeID: invoice.supplier.registrationType || "NRIC"
|
|
59
|
+
}] }],
|
|
60
|
+
PostalAddress: [{
|
|
61
|
+
CityName: [{ _: invoice.supplier.address.cityName }],
|
|
62
|
+
CountrySubentityCode: [{ _: invoice.supplier.address.state }],
|
|
63
|
+
AddressLine: [{ Line: [{ _: invoice.supplier.address.addressLine0 }] }],
|
|
64
|
+
Country: [{ IdentificationCode: [{
|
|
65
|
+
_: invoice.supplier.address.country || "MYS",
|
|
66
|
+
listID: "3166-1",
|
|
67
|
+
listAgencyID: "ISO"
|
|
68
|
+
}] }]
|
|
69
|
+
}],
|
|
70
|
+
PartyLegalEntity: [{ RegistrationName: [{ _: invoice.supplier.name }] }],
|
|
71
|
+
Contact: [{ Telephone: [{ _: invoice.supplier.contactNumber || "" }] }]
|
|
72
|
+
}] }],
|
|
73
|
+
AccountingCustomerParty: [{ Party: [{
|
|
74
|
+
PartyIdentification: [
|
|
75
|
+
{ ID: [{
|
|
76
|
+
_: invoice.buyer.tin,
|
|
77
|
+
schemeID: "TIN"
|
|
78
|
+
}] },
|
|
79
|
+
{ ID: [{
|
|
80
|
+
_: invoice.buyer.registrationNumber,
|
|
81
|
+
schemeID: invoice.buyer.registrationType || "NRIC"
|
|
82
|
+
}] },
|
|
83
|
+
{ ID: [{
|
|
84
|
+
_: invoice.buyer.sstRegistrationNumber || "NA",
|
|
85
|
+
schemeID: "SST"
|
|
86
|
+
}] }
|
|
87
|
+
],
|
|
88
|
+
PostalAddress: [{
|
|
89
|
+
CityName: [{ _: invoice.buyer.address.cityName }],
|
|
90
|
+
CountrySubentityCode: [{ _: invoice.buyer.address.state }],
|
|
91
|
+
AddressLine: [{ Line: [{ _: invoice.buyer.address.addressLine0 }] }],
|
|
92
|
+
Country: [{ IdentificationCode: [{
|
|
93
|
+
_: invoice.buyer.address.country || "MYS",
|
|
94
|
+
listID: "3166-1",
|
|
95
|
+
listAgencyID: "ISO"
|
|
96
|
+
}] }]
|
|
97
|
+
}],
|
|
98
|
+
PartyLegalEntity: [{ RegistrationName: [{ _: invoice.buyer.name }] }],
|
|
99
|
+
Contact: [{ Telephone: [{ _: invoice.buyer.contactNumber || "" }] }]
|
|
100
|
+
}] }],
|
|
101
|
+
TaxTotal: [{
|
|
102
|
+
TaxAmount: [{
|
|
103
|
+
_: invoice.taxTotal.taxAmount,
|
|
104
|
+
currencyID: invoice.invoiceCurrencyCode
|
|
105
|
+
}],
|
|
106
|
+
TaxSubtotal: [{
|
|
107
|
+
TaxableAmount: [{
|
|
108
|
+
_: invoice.legalMonetaryTotal.taxExclusiveAmount,
|
|
109
|
+
currencyID: invoice.invoiceCurrencyCode
|
|
110
|
+
}],
|
|
111
|
+
TaxAmount: [{
|
|
112
|
+
_: invoice.taxTotal.taxAmount,
|
|
113
|
+
currencyID: invoice.invoiceCurrencyCode
|
|
114
|
+
}],
|
|
115
|
+
TaxCategory: [{
|
|
116
|
+
ID: [{ _: invoice.invoiceLineItems[0]?.taxType || "01" }],
|
|
117
|
+
TaxScheme: [{ ID: [{
|
|
118
|
+
_: "OTH",
|
|
119
|
+
schemeAgencyID: "6",
|
|
120
|
+
schemeID: "UN/ECE 5153"
|
|
121
|
+
}] }]
|
|
122
|
+
}]
|
|
123
|
+
}]
|
|
124
|
+
}],
|
|
125
|
+
LegalMonetaryTotal: [{
|
|
126
|
+
LineExtensionAmount: [{
|
|
127
|
+
_: invoice.legalMonetaryTotal.taxExclusiveAmount,
|
|
128
|
+
currencyID: invoice.invoiceCurrencyCode
|
|
129
|
+
}],
|
|
130
|
+
TaxExclusiveAmount: [{
|
|
131
|
+
_: invoice.legalMonetaryTotal.taxExclusiveAmount,
|
|
132
|
+
currencyID: invoice.invoiceCurrencyCode
|
|
133
|
+
}],
|
|
134
|
+
TaxInclusiveAmount: [{
|
|
135
|
+
_: invoice.legalMonetaryTotal.taxInclusiveAmount,
|
|
136
|
+
currencyID: invoice.invoiceCurrencyCode
|
|
137
|
+
}],
|
|
138
|
+
PayableAmount: [{
|
|
139
|
+
_: invoice.legalMonetaryTotal.payableAmount,
|
|
140
|
+
currencyID: invoice.invoiceCurrencyCode
|
|
141
|
+
}]
|
|
142
|
+
}],
|
|
143
|
+
InvoiceLine: invoice.invoiceLineItems.map((item, index) => ({
|
|
144
|
+
ID: [{ _: (index + 1).toString() }],
|
|
145
|
+
Item: [{
|
|
146
|
+
CommodityClassification: [{ ItemClassificationCode: [{
|
|
147
|
+
_: item.itemClassificationCode,
|
|
148
|
+
listID: "CLASS"
|
|
149
|
+
}] }],
|
|
150
|
+
Description: [{ _: item.itemDescription }]
|
|
151
|
+
}],
|
|
152
|
+
ItemPriceExtension: [{ Amount: [{
|
|
153
|
+
_: item.totalTaxableAmountPerLine,
|
|
154
|
+
currencyID: invoice.invoiceCurrencyCode
|
|
155
|
+
}] }],
|
|
156
|
+
LineExtensionAmount: [{
|
|
157
|
+
_: item.totalTaxableAmountPerLine,
|
|
158
|
+
currencyID: invoice.invoiceCurrencyCode
|
|
159
|
+
}],
|
|
160
|
+
Price: [{ PriceAmount: [{
|
|
161
|
+
_: item.unitPrice,
|
|
162
|
+
currencyID: invoice.invoiceCurrencyCode
|
|
163
|
+
}] }],
|
|
164
|
+
TaxTotal: [{
|
|
165
|
+
TaxAmount: [{
|
|
166
|
+
_: item.taxAmount,
|
|
167
|
+
currencyID: invoice.invoiceCurrencyCode
|
|
168
|
+
}],
|
|
169
|
+
TaxSubtotal: [{
|
|
170
|
+
TaxableAmount: [{
|
|
171
|
+
_: item.totalTaxableAmountPerLine,
|
|
172
|
+
currencyID: invoice.invoiceCurrencyCode
|
|
173
|
+
}],
|
|
174
|
+
TaxAmount: [{
|
|
175
|
+
_: item.taxAmount,
|
|
176
|
+
currencyID: invoice.invoiceCurrencyCode
|
|
177
|
+
}],
|
|
178
|
+
TaxCategory: [{
|
|
179
|
+
ID: [{ _: item.taxType }],
|
|
180
|
+
TaxScheme: [{ ID: [{
|
|
181
|
+
_: "OTH",
|
|
182
|
+
schemeAgencyID: "6",
|
|
183
|
+
schemeID: "UN/ECE 5153"
|
|
184
|
+
}] }]
|
|
185
|
+
}]
|
|
186
|
+
}]
|
|
187
|
+
}]
|
|
188
|
+
})),
|
|
189
|
+
TaxExchangeRate: invoice.currencyExchangeRate ? [{
|
|
190
|
+
SourceCurrencyCode: [{ _: invoice.invoiceCurrencyCode }],
|
|
191
|
+
TargetCurrencyCode: [{ _: "MYR" }],
|
|
192
|
+
CalculationRate: [{ _: invoice.currencyExchangeRate }]
|
|
193
|
+
}] : void 0
|
|
194
|
+
};
|
|
195
|
+
};
|
|
196
|
+
/**
|
|
197
|
+
* Generates the complete UBL document structure with namespace declarations
|
|
198
|
+
*/
|
|
199
|
+
const generateCleanUBLDocument = (invoices) => {
|
|
200
|
+
return {
|
|
201
|
+
_D: "urn:oasis:names:specification:ubl:schema:xsd:Invoice-2",
|
|
202
|
+
_A: "urn:oasis:names:specification:ubl:schema:xsd:CommonAggregateComponents-2",
|
|
203
|
+
_B: "urn:oasis:names:specification:ubl:schema:xsd:CommonBasicComponents-2",
|
|
204
|
+
Invoice: invoices.map(generateCleanInvoiceObject)
|
|
205
|
+
};
|
|
206
|
+
};
|
|
207
|
+
/**
|
|
208
|
+
* Step 2: Calculate Document Digest
|
|
209
|
+
* FIXED: Remove UBLExtensions and Signature before hashing (DS322)
|
|
210
|
+
* Based on working implementation pattern
|
|
211
|
+
*/
|
|
212
|
+
const calculateDocumentDigest = (invoices) => {
|
|
213
|
+
const cleanDocument = generateCleanUBLDocument(invoices);
|
|
214
|
+
const documentForHashing = JSON.parse(JSON.stringify(cleanDocument));
|
|
215
|
+
if (documentForHashing.Invoice && Array.isArray(documentForHashing.Invoice)) documentForHashing.Invoice.forEach((invoice) => {
|
|
216
|
+
delete invoice.UBLExtensions;
|
|
217
|
+
delete invoice.Signature;
|
|
218
|
+
});
|
|
219
|
+
const documentString = JSON.stringify(documentForHashing);
|
|
220
|
+
const hash = crypto.createHash("sha256");
|
|
221
|
+
hash.update(documentString, "utf8");
|
|
222
|
+
return hash.digest("base64");
|
|
223
|
+
};
|
|
224
|
+
/**
|
|
225
|
+
* Step 4: Calculate Certificate Digest
|
|
226
|
+
* Enhanced to handle certificate content properly
|
|
227
|
+
*/
|
|
228
|
+
const calculateCertificateDigest = (certificatePem) => {
|
|
229
|
+
const certificateContent = certificatePem.replace(/-----BEGIN CERTIFICATE-----/g, "").replace(/-----END CERTIFICATE-----/g, "").replace(/\s+/g, "");
|
|
230
|
+
const certificateBinary = Buffer.from(certificateContent, "base64");
|
|
231
|
+
const hash = crypto.createHash("sha256");
|
|
232
|
+
hash.update(certificateBinary);
|
|
233
|
+
return hash.digest("base64");
|
|
234
|
+
};
|
|
235
|
+
/**
|
|
236
|
+
* Enhanced certificate info extraction with better error handling
|
|
237
|
+
* FIXED: Normalize issuer name format to match MyInvois expectations (DS326)
|
|
238
|
+
*/
|
|
239
|
+
const extractCertificateInfo = (certificatePem) => {
|
|
240
|
+
try {
|
|
241
|
+
const cert = new X509Certificate(certificatePem);
|
|
242
|
+
const serialNumberHex = cert.serialNumber;
|
|
243
|
+
const normalizeDistinguishedName = (dn) => {
|
|
244
|
+
const normalized = dn.split("\n").map((line) => line.trim()).filter((line) => line.length > 0).join(", ");
|
|
245
|
+
return normalized.replace(/\s*=\s*/g, "=").replace(/,\s+/g, ", ").replace(/\r/g, "").replace(/\s{2,}/g, " ").trim();
|
|
246
|
+
};
|
|
247
|
+
const formatSerialNumber = (serialHex) => {
|
|
248
|
+
const decimal = BigInt("0x" + serialHex).toString();
|
|
249
|
+
return decimal;
|
|
250
|
+
};
|
|
251
|
+
return {
|
|
252
|
+
issuerName: normalizeDistinguishedName(cert.issuer),
|
|
253
|
+
serialNumber: formatSerialNumber(serialNumberHex),
|
|
254
|
+
subjectName: normalizeDistinguishedName(cert.subject)
|
|
255
|
+
};
|
|
256
|
+
} catch (error) {
|
|
257
|
+
throw new Error(`Failed to extract certificate info: ${error instanceof Error ? error.message : String(error)}`);
|
|
258
|
+
}
|
|
259
|
+
};
|
|
260
|
+
/**
|
|
261
|
+
* Step 5: Create SignedProperties with enhanced structure
|
|
262
|
+
* FIXED: Simplified structure to match MyInvois expectations (DS320)
|
|
263
|
+
* Following MyInvois JSON signature specification exactly
|
|
264
|
+
*/
|
|
265
|
+
const createSignedProperties = (certificateDigest, signingTime, issuerName, serialNumber) => {
|
|
266
|
+
return { SignedProperties: [{
|
|
267
|
+
Id: "id-xades-signed-props",
|
|
268
|
+
SignedSignatureProperties: [{
|
|
269
|
+
SigningTime: [{ _: signingTime }],
|
|
270
|
+
SigningCertificate: [{ Cert: [{
|
|
271
|
+
CertDigest: [{
|
|
272
|
+
DigestMethod: [{ Algorithm: "http://www.w3.org/2001/04/xmlenc#sha256" }],
|
|
273
|
+
DigestValue: [{ _: certificateDigest }]
|
|
274
|
+
}],
|
|
275
|
+
IssuerSerial: [{
|
|
276
|
+
X509IssuerName: [{ _: issuerName }],
|
|
277
|
+
X509SerialNumber: [{ _: serialNumber }]
|
|
278
|
+
}]
|
|
279
|
+
}] }]
|
|
280
|
+
}]
|
|
281
|
+
}] };
|
|
282
|
+
};
|
|
283
|
+
/**
|
|
284
|
+
* Step 6: Calculate SignedProperties Digest
|
|
285
|
+
* FIXED: Add Target wrapper and use direct JSON stringify (DS320)
|
|
286
|
+
* Based on working implementation pattern
|
|
287
|
+
*/
|
|
288
|
+
const calculateSignedPropertiesDigest = (signedProperties) => {
|
|
289
|
+
const signedPropertiesWithTarget = {
|
|
290
|
+
Target: "signature",
|
|
291
|
+
SignedProperties: signedProperties.SignedProperties
|
|
292
|
+
};
|
|
293
|
+
const signedPropertiesString = JSON.stringify(signedPropertiesWithTarget);
|
|
294
|
+
const hash = crypto.createHash("sha256");
|
|
295
|
+
hash.update(signedPropertiesString, "utf8");
|
|
296
|
+
return hash.digest("base64");
|
|
297
|
+
};
|
|
298
|
+
/**
|
|
299
|
+
* Step 3: Create SignedInfo and calculate signature
|
|
300
|
+
* Enhanced with better structure and signature generation
|
|
301
|
+
*/
|
|
302
|
+
const createSignedInfoAndSign = (docDigest, propsDigest, privateKeyPem) => {
|
|
303
|
+
const signedInfo = {
|
|
304
|
+
CanonicalizationMethod: [{
|
|
305
|
+
_: "",
|
|
306
|
+
Algorithm: "http://www.w3.org/2006/12/xml-c14n11"
|
|
307
|
+
}],
|
|
308
|
+
SignatureMethod: [{
|
|
309
|
+
_: "",
|
|
310
|
+
Algorithm: "http://www.w3.org/2001/04/xmldsig-more#rsa-sha256"
|
|
311
|
+
}],
|
|
312
|
+
Reference: [{
|
|
313
|
+
Id: "id-doc-signed-data",
|
|
314
|
+
Type: "",
|
|
315
|
+
URI: "",
|
|
316
|
+
DigestMethod: [{
|
|
317
|
+
_: "",
|
|
318
|
+
Algorithm: "http://www.w3.org/2001/04/xmlenc#sha256"
|
|
319
|
+
}],
|
|
320
|
+
DigestValue: [{ _: docDigest }]
|
|
321
|
+
}, {
|
|
322
|
+
Id: "id-xades-signed-props",
|
|
323
|
+
Type: "http://uri.etsi.org/01903/v1.3.2#SignedProperties",
|
|
324
|
+
URI: "#id-xades-signed-props",
|
|
325
|
+
DigestMethod: [{
|
|
326
|
+
_: "",
|
|
327
|
+
Algorithm: "http://www.w3.org/2001/04/xmlenc#sha256"
|
|
328
|
+
}],
|
|
329
|
+
DigestValue: [{ _: propsDigest }]
|
|
330
|
+
}]
|
|
331
|
+
};
|
|
332
|
+
const signedInfoString = JSON.stringify(signedInfo);
|
|
333
|
+
try {
|
|
334
|
+
const signer = crypto.createSign("RSA-SHA256");
|
|
335
|
+
signer.update(signedInfoString, "utf8");
|
|
336
|
+
const signatureValue = signer.sign(privateKeyPem, "base64");
|
|
337
|
+
return {
|
|
338
|
+
signedInfo,
|
|
339
|
+
signatureValue
|
|
340
|
+
};
|
|
341
|
+
} catch (error) {
|
|
342
|
+
throw new Error(`Signature generation failed: ${error instanceof Error ? error.message : String(error)}`);
|
|
343
|
+
}
|
|
344
|
+
};
|
|
345
|
+
/**
|
|
346
|
+
* Complete document generation with signatures
|
|
347
|
+
* Follows the complete MyInvois JSON signature creation process (Steps 1-7)
|
|
348
|
+
*/
|
|
349
|
+
const generateCompleteDocument = (invoices, signingCredentials) => {
|
|
350
|
+
try {
|
|
351
|
+
const docDigest = calculateDocumentDigest(invoices);
|
|
352
|
+
const signingTime = (/* @__PURE__ */ new Date()).toISOString();
|
|
353
|
+
const certInfo = extractCertificateInfo(signingCredentials.certificatePem);
|
|
354
|
+
const certificateDigest = calculateCertificateDigest(signingCredentials.certificatePem);
|
|
355
|
+
const signedProperties = createSignedProperties(certificateDigest, signingTime, certInfo.issuerName, certInfo.serialNumber);
|
|
356
|
+
const propsDigest = calculateSignedPropertiesDigest(signedProperties);
|
|
357
|
+
const { signedInfo, signatureValue } = createSignedInfoAndSign(docDigest, propsDigest, signingCredentials.privateKeyPem);
|
|
358
|
+
const certificate = signingCredentials.certificatePem.replace(/-----BEGIN CERTIFICATE-----/g, "").replace(/-----END CERTIFICATE-----/g, "").replace(/\s+/g, "");
|
|
359
|
+
const signedInvoices = invoices.map((invoice) => {
|
|
360
|
+
const cleanInvoice = generateCleanInvoiceObject(invoice);
|
|
361
|
+
return {
|
|
362
|
+
...cleanInvoice,
|
|
363
|
+
UBLExtensions: [{ UBLExtension: [{
|
|
364
|
+
ExtensionURI: [{ _: "urn:oasis:names:specification:ubl:dsig:enveloped:xades" }],
|
|
365
|
+
ExtensionContent: [{ UBLDocumentSignatures: [{ SignatureInformation: [{
|
|
366
|
+
ID: [{ _: "urn:oasis:names:specification:ubl:signature:1" }],
|
|
367
|
+
ReferencedSignatureID: [{ _: "urn:oasis:names:specification:ubl:signature:Invoice" }],
|
|
368
|
+
Signature: [{
|
|
369
|
+
Id: "signature",
|
|
370
|
+
Object: [{ QualifyingProperties: [{
|
|
371
|
+
Target: "signature",
|
|
372
|
+
SignedProperties: signedProperties.SignedProperties
|
|
373
|
+
}] }],
|
|
374
|
+
KeyInfo: [{ X509Data: [{
|
|
375
|
+
X509Certificate: [{ _: certificate }],
|
|
376
|
+
X509SubjectName: [{ _: certInfo.subjectName }],
|
|
377
|
+
X509IssuerSerial: [{
|
|
378
|
+
X509IssuerName: [{ _: certInfo.issuerName }],
|
|
379
|
+
X509SerialNumber: [{ _: certInfo.serialNumber }]
|
|
380
|
+
}]
|
|
381
|
+
}] }],
|
|
382
|
+
SignatureValue: [{ _: signatureValue }],
|
|
383
|
+
SignedInfo: [signedInfo]
|
|
384
|
+
}]
|
|
385
|
+
}] }] }]
|
|
386
|
+
}] }],
|
|
387
|
+
Signature: [{
|
|
388
|
+
ID: [{ _: "urn:oasis:names:specification:ubl:signature:Invoice" }],
|
|
389
|
+
SignatureMethod: [{ _: "urn:oasis:names:specification:ubl:dsig:enveloped:xades" }]
|
|
390
|
+
}]
|
|
391
|
+
};
|
|
392
|
+
});
|
|
393
|
+
return {
|
|
394
|
+
_D: "urn:oasis:names:specification:ubl:schema:xsd:Invoice-2",
|
|
395
|
+
_A: "urn:oasis:names:specification:ubl:schema:xsd:CommonAggregateComponents-2",
|
|
396
|
+
_B: "urn:oasis:names:specification:ubl:schema:xsd:CommonBasicComponents-2",
|
|
397
|
+
Invoice: signedInvoices
|
|
398
|
+
};
|
|
399
|
+
} catch (error) {
|
|
400
|
+
throw new Error(`Document generation failed: ${error instanceof Error ? error.message : String(error)}`);
|
|
401
|
+
}
|
|
402
|
+
};
|
|
403
|
+
|
|
404
|
+
//#endregion
|
|
405
|
+
export { calculateCertificateDigest, calculateDocumentDigest, calculateSignedPropertiesDigest, canonicalizeJSON, createSignedInfoAndSign, createSignedProperties, extractCertificateInfo, generateCleanInvoiceObject, generateCleanUBLDocument, generateCompleteDocument, sortObjectKeys };
|