@ripwords/myinvois-client 0.3.6 → 0.4.0
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/README.md +27 -0
- package/dist/{apiQueue-DgKWaQDS.cjs → apiQueue-C45w-ymm.cjs} +1 -1
- package/dist/{apiQueue-DgKWaQDS.cjs.map → apiQueue-C45w-ymm.cjs.map} +1 -1
- package/dist/{certificate-CWmfCPdt.cjs → certificate-Dw46fkYv.cjs} +5 -31
- package/dist/certificate-Dw46fkYv.cjs.map +1 -0
- package/dist/{document-B6ab06s1.cjs → document-B11B5lqd.cjs} +75 -21
- package/dist/document-B11B5lqd.cjs.map +1 -0
- package/dist/{documentManagement-BviJDtf2.cjs → documentManagement-qJnd0l1z.cjs} +3 -2
- package/dist/documentManagement-qJnd0l1z.cjs.map +1 -0
- package/dist/{documentSubmission-B6hkivRK.cjs → documentSubmission-BaPECvYU.cjs} +6 -5
- package/dist/{documentSubmission-B6hkivRK.cjs.map → documentSubmission-BaPECvYU.cjs.map} +1 -1
- package/dist/{documentTypeManagement-D_-LiQVg.cjs → documentTypeManagement-DXRLfTsW.cjs} +1 -1
- package/dist/{documentTypeManagement-D_-LiQVg.cjs.map → documentTypeManagement-DXRLfTsW.cjs.map} +1 -1
- package/dist/{getBaseUrl-D0G4GZmp.cjs → getBaseUrl-D2iJdUGL.cjs} +1 -1
- package/dist/{getBaseUrl-D0G4GZmp.cjs.map → getBaseUrl-D2iJdUGL.cjs.map} +1 -1
- package/dist/index.cjs +14 -12
- package/dist/index.cjs.map +1 -1
- package/dist/index10.cjs +6 -195
- package/dist/index11.cjs +22 -0
- package/dist/index12.cjs +2 -24
- package/dist/index13.cjs +3 -0
- package/dist/index14.cjs +3 -0
- package/dist/index15.cjs +325 -25
- package/dist/index15.cjs.map +1 -1
- package/dist/index16.cjs +187 -19
- package/dist/index16.cjs.map +1 -1
- package/dist/index17.cjs +62 -0
- package/dist/index17.cjs.map +1 -0
- package/dist/index18.cjs +523 -25
- package/dist/index18.cjs.map +1 -1
- package/dist/index19.cjs +187 -15
- package/dist/index19.cjs.map +1 -1
- package/dist/index2.cjs +1 -1
- package/dist/index21.cjs +25 -0
- package/dist/{index12.cjs.map → index21.cjs.map} +1 -1
- package/dist/index23.cjs +0 -5
- package/dist/index24.cjs +25 -9
- package/dist/index24.cjs.map +1 -1
- package/dist/index25.cjs +24 -4
- package/dist/index25.cjs.map +1 -0
- package/dist/index26.cjs +0 -22
- package/dist/index27.cjs +33 -2
- package/dist/index27.cjs.map +1 -0
- package/dist/index28.cjs +23 -2
- package/dist/index28.cjs.map +1 -0
- package/dist/index29.cjs +0 -330
- package/dist/index3.cjs +2 -2
- package/dist/index30.cjs +0 -193
- package/dist/index31.cjs +0 -208
- package/dist/index32.cjs +203 -104
- package/dist/index32.cjs.map +1 -1
- package/dist/index33.cjs +104 -132
- package/dist/index33.cjs.map +1 -1
- package/dist/index34.cjs +132 -59
- package/dist/index34.cjs.map +1 -1
- package/dist/index35.cjs +59 -262
- package/dist/index35.cjs.map +1 -1
- package/dist/index36.cjs +262 -74
- package/dist/index36.cjs.map +1 -1
- package/dist/index37.cjs +74 -102
- package/dist/index37.cjs.map +1 -1
- package/dist/index38.cjs +102 -68
- package/dist/index38.cjs.map +1 -1
- package/dist/index39.cjs +68 -107
- package/dist/index39.cjs.map +1 -1
- package/dist/index4.cjs +1 -1
- package/dist/index40.cjs +107 -95
- package/dist/index40.cjs.map +1 -1
- package/dist/index41.cjs +95 -13
- package/dist/index41.cjs.map +1 -1
- package/dist/index42.cjs +11 -31
- package/dist/index42.cjs.map +1 -1
- package/dist/index43.cjs +31 -12
- package/dist/index43.cjs.map +1 -1
- package/dist/index44.cjs +12 -22
- package/dist/index44.cjs.map +1 -1
- package/dist/index45.cjs +22 -13
- package/dist/index45.cjs.map +1 -1
- package/dist/index46.cjs +13 -7
- package/dist/index46.cjs.map +1 -1
- package/dist/index47.cjs +7 -7
- package/dist/index47.cjs.map +1 -1
- package/dist/index48.cjs +7 -10
- package/dist/index48.cjs.map +1 -1
- package/dist/index49.cjs +10 -6
- package/dist/index49.cjs.map +1 -1
- package/dist/index5.cjs +1 -1
- package/dist/index50.cjs +6 -101
- package/dist/index50.cjs.map +1 -1
- package/dist/index51.cjs +101 -106
- package/dist/index51.cjs.map +1 -1
- package/dist/index52.cjs +106 -119
- package/dist/index52.cjs.map +1 -1
- package/dist/index53.cjs +119 -106
- package/dist/index53.cjs.map +1 -1
- package/dist/index54.cjs +106 -98
- package/dist/index54.cjs.map +1 -1
- package/dist/index55.cjs +98 -118
- package/dist/index55.cjs.map +1 -1
- package/dist/index56.cjs +118 -127
- package/dist/index56.cjs.map +1 -1
- package/dist/index57.cjs +127 -117
- package/dist/index57.cjs.map +1 -1
- package/dist/index58.cjs +117 -14
- package/dist/index58.cjs.map +1 -1
- package/dist/index59.cjs +14 -95
- package/dist/index59.cjs.map +1 -1
- package/dist/index6.cjs +1 -1
- package/dist/index60.cjs +95 -142
- package/dist/index60.cjs.map +1 -1
- package/dist/index61.cjs +142 -114
- package/dist/index61.cjs.map +1 -1
- package/dist/index62.cjs +114 -144
- package/dist/index62.cjs.map +1 -1
- package/dist/index63.cjs +144 -113
- package/dist/index63.cjs.map +1 -1
- package/dist/index64.cjs +113 -17
- package/dist/index64.cjs.map +1 -1
- package/dist/index65.cjs +17 -112
- package/dist/index65.cjs.map +1 -1
- package/dist/index66.cjs +112 -47
- package/dist/index66.cjs.map +1 -1
- package/dist/index67.cjs +47 -14
- package/dist/index67.cjs.map +1 -1
- package/dist/index68.cjs +14 -28
- package/dist/index68.cjs.map +1 -1
- package/dist/index69.cjs +28 -22
- package/dist/index69.cjs.map +1 -1
- package/dist/index7.cjs +1 -1
- package/dist/index70.cjs +22 -9
- package/dist/index70.cjs.map +1 -1
- package/dist/index71.cjs +9 -8
- package/dist/index71.cjs.map +1 -1
- package/dist/index72.cjs +8 -17
- package/dist/index72.cjs.map +1 -1
- package/dist/index73.cjs +17 -412
- package/dist/index73.cjs.map +1 -1
- package/dist/index74.cjs +412 -8
- package/dist/index74.cjs.map +1 -1
- package/dist/index75.cjs +8 -9
- package/dist/index75.cjs.map +1 -1
- package/dist/index76.cjs +16 -0
- package/dist/index76.cjs.map +1 -0
- package/dist/index8.cjs +4 -61
- package/dist/index9.cjs +9 -528
- package/dist/index9.cjs.map +1 -1
- package/dist/lineItems-C8iT0OL1.cjs +64 -0
- package/dist/lineItems-C8iT0OL1.cjs.map +1 -0
- package/dist/{notificationManagement-DLBDn77E.cjs → notificationManagement-C_qrFwWL.cjs} +1 -1
- package/dist/{notificationManagement-DLBDn77E.cjs.map → notificationManagement-C_qrFwWL.cjs.map} +1 -1
- package/dist/{platformLogin-Ch6hFKoU.cjs → platformLogin-DC4s6Qpv.cjs} +1 -1
- package/dist/{platformLogin-Ch6hFKoU.cjs.map → platformLogin-DC4s6Qpv.cjs.map} +1 -1
- package/dist/{taxpayerValidation-j8s6YGED.cjs → taxpayerValidation-DzCJROgP.cjs} +1 -1
- package/dist/{taxpayerValidation-j8s6YGED.cjs.map → taxpayerValidation-DzCJROgP.cjs.map} +1 -1
- package/package.json +1 -1
- package/dist/0X-Cr3M7hci.d.cts +0 -211
- package/dist/0X-qj3c_vAo.d.ts +0 -210
- package/dist/1X-B-lyAVC4.d.ts +0 -111
- package/dist/1X-C72Wa4pk.d.ts +0 -19
- package/dist/1X-CWwmfCzo.d.cts +0 -112
- package/dist/1X-DYw2tURz.d.cts +0 -20
- package/dist/2X-BfCK614i.d.ts +0 -139
- package/dist/2X-CH89y3Af.d.ts +0 -39
- package/dist/2X-CXtqiwRg.d.cts +0 -40
- package/dist/2X-DNtkJ0tj.d.cts +0 -140
- package/dist/3X-BOxfatu3.d.cts +0 -62
- package/dist/3X-Bk_fUIAk.d.cts +0 -21
- package/dist/3X-DryrwYRf.d.ts +0 -20
- package/dist/3X-Du10Ix_V.d.ts +0 -61
- package/dist/4X-BPMLRoDv.d.ts +0 -30
- package/dist/4X-C7fzDWJ_.d.cts +0 -265
- package/dist/4X-CK9rj5qd.d.cts +0 -31
- package/dist/4X-x3cRFuSM.d.ts +0 -264
- package/dist/5X-B52AKeTz.d.ts +0 -76
- package/dist/5X-Bv7M6KyG.d.ts +0 -21
- package/dist/5X-CNAFsDm2.d.cts +0 -77
- package/dist/5X-CpAvNxvd.d.cts +0 -22
- package/dist/6X-DBwpWPQ6.d.ts +0 -104
- package/dist/6X-ZeZ8OB57.d.ts +0 -15
- package/dist/6X-bXWr_7xY.d.cts +0 -16
- package/dist/6X-uObUP4VG.d.cts +0 -105
- package/dist/7X-BJBOlWhV.d.cts +0 -16
- package/dist/7X-BifJnY24.d.cts +0 -71
- package/dist/7X-C_XGDSpE.d.ts +0 -70
- package/dist/7X-Dvw2Z2VN.d.ts +0 -15
- package/dist/8X-BkgoX8dg.d.ts +0 -18
- package/dist/8X-Di_0wuRX.d.ts +0 -109
- package/dist/8X-DioBXCJ0.d.cts +0 -110
- package/dist/8X-dshwiOxU.d.cts +0 -19
- package/dist/9X-8ZnvvViY.d.ts +0 -97
- package/dist/9X-9j2Djy4W.d.cts +0 -15
- package/dist/9X-BjffnXuq.d.cts +0 -98
- package/dist/9X-Du0e44cq.d.ts +0 -14
- package/dist/AX-BSPLpkVT.d.ts +0 -110
- package/dist/AX-CWCopjCC.d.cts +0 -111
- package/dist/BX-Cq7WdhD3.d.cts +0 -115
- package/dist/BX-u4yMaIkz.d.ts +0 -114
- package/dist/CX-BKZG0pVE.d.ts +0 -127
- package/dist/CX-DlcVZfxx.d.cts +0 -128
- package/dist/DX-CwA9WJAf.d.ts +0 -114
- package/dist/DX-DLdv-Nxj.d.cts +0 -115
- package/dist/EX-BRNu1Ooi.d.ts +0 -106
- package/dist/EX-CIJv8-Lw.d.cts +0 -107
- package/dist/FX-DBxjq6xY.d.ts +0 -126
- package/dist/FX-DlDM_Zea.d.cts +0 -127
- package/dist/GX-B3CvWNrP.d.ts +0 -135
- package/dist/GX-BFGNX6Jy.d.cts +0 -136
- package/dist/HX-D7FABgTv.d.ts +0 -125
- package/dist/HX-D7Y9qRSm.d.cts +0 -126
- package/dist/IX-D3ZknsAB.d.ts +0 -22
- package/dist/IX-DPTrbhgX.d.cts +0 -23
- package/dist/JX-BRaZM3Gc.d.ts +0 -103
- package/dist/JX-D8JmtNpo.d.cts +0 -104
- package/dist/KX-B1K0OWoi.d.ts +0 -150
- package/dist/KX-B46KJZrH.d.cts +0 -151
- package/dist/LX-CKRefinL.d.ts +0 -122
- package/dist/LX-CfTKy8Sx.d.cts +0 -123
- package/dist/MX-BKJ4gNvV.d.cts +0 -153
- package/dist/MX-DKmaeO9r.d.ts +0 -152
- package/dist/NX-CJL2Gn1U.d.ts +0 -121
- package/dist/NX-CSdLIFqU.d.cts +0 -122
- package/dist/OX-BwqOsulT.d.cts +0 -26
- package/dist/OX-Do2ap5v_.d.ts +0 -25
- package/dist/PX-BTx0TXsJ.d.cts +0 -121
- package/dist/PX-_DOr_rm9.d.ts +0 -120
- package/dist/QX-BUXR8RJL.d.cts +0 -59
- package/dist/QX-HlXLkIki.d.ts +0 -58
- package/dist/RX-CZ8HiH37.d.cts +0 -23
- package/dist/RX-DVPGOVES.d.ts +0 -22
- package/dist/SX-CLxsqiQk.d.cts +0 -40
- package/dist/SX-OGT3qwlS.d.ts +0 -39
- package/dist/TX-DKc8hLc4.d.cts +0 -31
- package/dist/TX-cPDINmZ5.d.ts +0 -30
- package/dist/UX-Be3sqvPV.d.ts +0 -17
- package/dist/UX-bc7xDnZI.d.cts +0 -18
- package/dist/VX-CIHTFyoC.d.cts +0 -17
- package/dist/VX-CoEuBNDK.d.ts +0 -16
- package/dist/WX-BHvlq7Jx.d.cts +0 -26
- package/dist/WX-DMvw__jH.d.ts +0 -25
- package/dist/XX-DOA-10JW.d.ts +0 -827
- package/dist/XX-DztQxYHJ.d.cts +0 -828
- package/dist/YX-CkxLeFb4.d.cts +0 -20
- package/dist/YX-F34sJ7Ik.d.ts +0 -19
- package/dist/ZX-CDQOfsHh.d.ts +0 -21
- package/dist/ZX-DVb_buNI.d.cts +0 -22
- package/dist/api/documentManagement.d.ts +0 -89
- package/dist/api/documentManagement.js +0 -3
- package/dist/api/documentSubmission.d.ts +0 -92
- package/dist/api/documentSubmission.js +0 -5
- package/dist/api/documentTypeManagement.d.ts +0 -70
- package/dist/api/documentTypeManagement.js +0 -3
- package/dist/api/notificationManagement.d.ts +0 -68
- package/dist/api/notificationManagement.js +0 -3
- package/dist/api/platformLogin.d.ts +0 -68
- package/dist/api/platformLogin.js +0 -3
- package/dist/api/taxpayerValidation.d.ts +0 -71
- package/dist/api/taxpayerValidation.js +0 -4
- package/dist/apiQueue-B6Q644Bz.js +0 -201
- package/dist/certificate-COwqszxD.js +0 -71
- package/dist/certificate-CWmfCPdt.cjs.map +0 -1
- package/dist/chunk-CUT6urMc.cjs +0 -30
- package/dist/classification-codes-B15PbWxz.d.cts +0 -118
- package/dist/classification-codes-C2X4xW5-.d.ts +0 -117
- package/dist/country-code-DPeNFMMi.d.cts +0 -543
- package/dist/country-code-DsI8Mbzx.d.ts +0 -542
- package/dist/currencies-DKuDflOO.d.ts +0 -207
- package/dist/currencies-S5g1gzBU.d.cts +0 -208
- package/dist/document-B6ab06s1.cjs.map +0 -1
- package/dist/document-Bb5WkhRe.js +0 -626
- package/dist/documentManagement-BviJDtf2.cjs.map +0 -1
- package/dist/documentManagement-lPm8Gziu.js +0 -49
- package/dist/documentSubmission-CMMzKvVl.js +0 -134
- package/dist/documentTypeManagement-cBtVCOY3.js +0 -22
- package/dist/documents-BYV12TVt.d.ts +0 -1196
- package/dist/documents-D_V38Q_U.d.cts +0 -1197
- package/dist/e-invoice-BuwtFnlI.d.cts +0 -44
- package/dist/e-invoice-C2TxhyrK.d.ts +0 -43
- package/dist/formatIdValue-qTxJqj9o.js +0 -7
- package/dist/getBaseUrl-D7nUmoYb.js +0 -7
- package/dist/index-CygwSf0x.d.ts +0 -15
- package/dist/index-Yr1QAbIF.d.cts +0 -16
- package/dist/index.d.ts +0 -614
- package/dist/index.js +0 -627
- package/dist/index10.cjs.map +0 -1
- package/dist/index29.cjs.map +0 -1
- package/dist/index30.cjs.map +0 -1
- package/dist/index31.cjs.map +0 -1
- package/dist/index61.cts.map +0 -1
- package/dist/index62.cts.map +0 -1
- package/dist/index63.cts.map +0 -1
- package/dist/index64.cts.map +0 -1
- package/dist/index65.cts.map +0 -1
- package/dist/index66.cts.map +0 -1
- package/dist/index67.cts.map +0 -1
- package/dist/index68.cts.map +0 -1
- package/dist/index69.cts.map +0 -1
- package/dist/index70.cts.map +0 -1
- package/dist/index71.cts.map +0 -1
- package/dist/index72.cts.map +0 -1
- package/dist/index73.cts.map +0 -1
- package/dist/index74.cts.map +0 -1
- package/dist/index75.cts.map +0 -1
- package/dist/index8.cjs.map +0 -1
- package/dist/msic-codes-CIKdPqag.d.cts +0 -26
- package/dist/msic-codes-CjrrJmzb.d.ts +0 -25
- package/dist/notificationManagement-n4Z5e-My.js +0 -19
- package/dist/notifications-DdlEkprb.d.cts +0 -63
- package/dist/notifications-sFhgh3rJ.d.ts +0 -62
- package/dist/payment-modes-7c1lWlIr.d.cts +0 -44
- package/dist/payment-modes-g3DzLmWb.d.ts +0 -43
- package/dist/platformLogin-CqI9OLYP.js +0 -39
- package/dist/signatures-CerHUrj3.d.ts +0 -172
- package/dist/signatures-CyDSZr_e.d.cts +0 -173
- package/dist/state-codes-Ow4oehYT.d.ts +0 -61
- package/dist/state-codes-u3ppueWo.d.cts +0 -62
- package/dist/tax-types-ClyLgydM.d.ts +0 -41
- package/dist/tax-types-fu5Q6dic.d.cts +0 -42
- package/dist/taxpayer-BAoT73gg.d.ts +0 -35
- package/dist/taxpayer-DwGzY1IL.d.cts +0 -36
- package/dist/taxpayerValidation-Xd_EHDvk.js +0 -47
- package/dist/types/classification-codes.d.ts +0 -2
- package/dist/types/country-code.d.ts +0 -2
- package/dist/types/currencies.d.ts +0 -2
- package/dist/types/documents.d.ts +0 -55
- package/dist/types/e-invoice.d.ts +0 -2
- package/dist/types/index.d.ts +0 -61
- package/dist/types/msic/0X.d.ts +0 -2
- package/dist/types/msic/1X.d.ts +0 -2
- package/dist/types/msic/2X.d.ts +0 -2
- package/dist/types/msic/3X.d.ts +0 -2
- package/dist/types/msic/4X.d.ts +0 -2
- package/dist/types/msic/5X.d.ts +0 -2
- package/dist/types/msic/6X.d.ts +0 -2
- package/dist/types/msic/7X.d.ts +0 -2
- package/dist/types/msic/8X.d.ts +0 -2
- package/dist/types/msic/9X.d.ts +0 -2
- package/dist/types/msic-codes.d.ts +0 -12
- package/dist/types/notifications.d.ts +0 -2
- package/dist/types/payment-modes.d.ts +0 -2
- package/dist/types/signatures.d.ts +0 -2
- package/dist/types/state-codes.d.ts +0 -2
- package/dist/types/tax-types.d.ts +0 -2
- package/dist/types/taxpayer.d.ts +0 -56
- package/dist/types/unit/1X.d.ts +0 -2
- package/dist/types/unit/2X.d.ts +0 -2
- package/dist/types/unit/3X.d.ts +0 -2
- package/dist/types/unit/4X.d.ts +0 -2
- package/dist/types/unit/5X.d.ts +0 -2
- package/dist/types/unit/6X.d.ts +0 -2
- package/dist/types/unit/7X.d.ts +0 -2
- package/dist/types/unit/8X.d.ts +0 -2
- package/dist/types/unit/9X.d.ts +0 -2
- package/dist/types/unit/AX.d.ts +0 -2
- package/dist/types/unit/BX.d.ts +0 -2
- package/dist/types/unit/CX.d.ts +0 -2
- package/dist/types/unit/DX.d.ts +0 -2
- package/dist/types/unit/EX.d.ts +0 -2
- package/dist/types/unit/FX.d.ts +0 -2
- package/dist/types/unit/GX.d.ts +0 -2
- package/dist/types/unit/HX.d.ts +0 -2
- package/dist/types/unit/IX.d.ts +0 -2
- package/dist/types/unit/JX.d.ts +0 -2
- package/dist/types/unit/KX.d.ts +0 -2
- package/dist/types/unit/LX.d.ts +0 -2
- package/dist/types/unit/MX.d.ts +0 -2
- package/dist/types/unit/NX.d.ts +0 -2
- package/dist/types/unit/OX.d.ts +0 -2
- package/dist/types/unit/PX.d.ts +0 -2
- package/dist/types/unit/QX.d.ts +0 -2
- package/dist/types/unit/RX.d.ts +0 -2
- package/dist/types/unit/SX.d.ts +0 -2
- package/dist/types/unit/TX.d.ts +0 -2
- package/dist/types/unit/UX.d.ts +0 -2
- package/dist/types/unit/VX.d.ts +0 -2
- package/dist/types/unit/WX.d.ts +0 -2
- package/dist/types/unit/XX.d.ts +0 -2
- package/dist/types/unit/YX.d.ts +0 -2
- package/dist/types/unit/ZX.d.ts +0 -2
- package/dist/types/unit-types.d.ts +0 -37
- package/dist/types/utils.d.ts +0 -2
- package/dist/unit-types-BXdufGWm.d.cts +0 -56
- package/dist/unit-types-VgYXIwTT.d.ts +0 -55
- package/dist/utils/apiQueue.d.ts +0 -45
- package/dist/utils/apiQueue.js +0 -3
- package/dist/utils/base64.d.ts +0 -5
- package/dist/utils/base64.js +0 -10
- package/dist/utils/certificate.d.ts +0 -26
- package/dist/utils/certificate.js +0 -3
- package/dist/utils/document.d.ts +0 -211
- package/dist/utils/document.js +0 -4
- package/dist/utils/formatIdValue.d.ts +0 -4
- package/dist/utils/formatIdValue.js +0 -3
- package/dist/utils/getBaseUrl.d.ts +0 -4
- package/dist/utils/getBaseUrl.js +0 -3
- package/dist/utils/signature-diagnostics.d.ts +0 -96
- package/dist/utils/signature-diagnostics.js +0 -327
- package/dist/utils/validation.d.ts +0 -108
- package/dist/utils/validation.js +0 -186
- package/dist/utils-C4FoVKLq.d.ts +0 -4
- package/dist/utils-Cdqbbzca.d.cts +0 -5
package/README.md
CHANGED
|
@@ -123,6 +123,33 @@ Replace the placeholders with your actual values:
|
|
|
123
123
|
- `YOUR_TIN`: `IG12345678901` (MyInvois TIN)
|
|
124
124
|
- `YOUR_DOMAIN`: `mycompany.com`
|
|
125
125
|
|
|
126
|
+
## Rate Limiting
|
|
127
|
+
|
|
128
|
+
The client includes a built-in, per-`clientId` request queue that proactively
|
|
129
|
+
keeps each API category within the MyInvois-documented limits (e.g. submit
|
|
130
|
+
documents 100/min, get submission 300/min, cancel/reject 12/min). Requests that
|
|
131
|
+
would exceed a category's limit are queued and released as the sliding window
|
|
132
|
+
frees up — you can fire calls freely and the queue paces them.
|
|
133
|
+
|
|
134
|
+
### Known limitations
|
|
135
|
+
|
|
136
|
+
These are intentional trade-offs for a dependency-free, in-process limiter.
|
|
137
|
+
Plan around them for production deployments:
|
|
138
|
+
|
|
139
|
+
- **In-process only.** Limits are tracked in memory per Node.js process. If you
|
|
140
|
+
run multiple instances/workers (clustered NestJS, PM2 cluster, serverless,
|
|
141
|
+
multiple pods), each process tracks its own counts, so the *combined* traffic
|
|
142
|
+
can exceed the MyInvois limit. For horizontally-scaled deployments, throttle
|
|
143
|
+
upstream or route MyInvois calls through a single worker.
|
|
144
|
+
- **Bursts within the cap.** The limiter allows up to `max` requests in the same
|
|
145
|
+
instant (as long as the 60s window allows it). This is within the documented
|
|
146
|
+
limit, but if you observe `429`s under heavy concurrency, reduce upstream
|
|
147
|
+
concurrency.
|
|
148
|
+
- **No automatic `429` / `Retry-After` back-off.** The limiter is proactive
|
|
149
|
+
only; it does not currently inspect rate-limit response headers and retry.
|
|
150
|
+
- **Login is not queued.** Token requests (`/connect/token`) bypass the queue.
|
|
151
|
+
Cache and reuse the client so token refreshes stay infrequent.
|
|
152
|
+
|
|
126
153
|
## Production Certificates
|
|
127
154
|
|
|
128
155
|
⚠️ **For production use, obtain official business certificates from MyInvois-approved Certificate Authorities:**
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"apiQueue-DgKWaQDS.cjs","names":["queues: Record<string, Queue>","LIMITS: Record<ApiCategory, { max: number; perMs: number }>","timestamps: number[]","windowMs: number","now: number","max: number","clientId: string","category: ApiCategory","task: Task<T>","now","key: string","debug: boolean","category?: ApiCategory","path: string","method: string"],"sources":["../src/utils/apiQueue.ts"],"sourcesContent":["// A very small utility that provides per-endpoint request queuing with rate-limits.\n// The goal is to make sure that we never exceed the vendor-defined limits while also ensuring\n// that every request is eventually executed.\n//\n// NOTE: This is intentionally minimal – no external dependencies are introduced.\n// If you need more advanced features (persistence, jitter, etc.) consider a library such as `bottleneck`.\n\n/*\nRate-limit specification (per 60-second window)\n----------------------------------------------\nLogin as Taxpayer System : 12\nLogin as Intermediary System : 12\nSubmit Documents : 100\nGet Submission : 300\nCancel Document : 12\nReject Document : 12\nGet Document : 60\nGet Document Details : 125\nGet Recent Documents : 12\nSearch Documents : 12\nSearch Taxpayer's TIN : 60\nTaxpayer's QR Code : 60\n*/\n\nexport type ApiCategory =\n | 'loginTaxpayer'\n | 'loginIntermediary'\n | 'submitDocuments'\n | 'getSubmission'\n | 'cancelDocument'\n | 'rejectDocument'\n | 'getDocument'\n | 'getDocumentDetails'\n | 'getRecentDocuments'\n | 'searchDocuments'\n | 'searchTin'\n | 'taxpayerQr'\n | 'default'\n\ntype Task<T> = () => Promise<T>\n\ninterface Queue {\n running: number\n queue: Array<{ run: () => void; addedAt: number }>\n requestTimestamps: number[] // Track all request timestamps in the sliding window\n nextTimer: NodeJS.Timeout | null // Track scheduled timer to avoid pile-up\n}\n\nconst queues: Record<string, Queue> = {}\n\n// Rate limits: max requests per time window (in ms)\nconst LIMITS: Record<ApiCategory, { max: number; perMs: number }> = {\n loginTaxpayer: { max: 12, perMs: 60000 }, // 12 req/60s\n loginIntermediary: { max: 12, perMs: 60000 }, // 12 req/60s\n submitDocuments: { max: 100, perMs: 60000 }, // 100 req/60s\n getSubmission: { max: 300, perMs: 60000 }, // 300 req/60s\n cancelDocument: { max: 12, perMs: 60000 }, // 12 req/60s\n rejectDocument: { max: 12, perMs: 60000 }, // 12 req/60s\n getDocument: { max: 60, perMs: 60000 }, // 60 req/60s\n getDocumentDetails: { max: 125, perMs: 60000 }, // 125 req/60s\n getRecentDocuments: { max: 12, perMs: 60000 }, // 12 req/60s\n searchDocuments: { max: 12, perMs: 60000 }, // 12 req/60s\n searchTin: { max: 60, perMs: 60000 }, // 60 req/60s\n taxpayerQr: { max: 60, perMs: 60000 }, // 60 req/60s\n default: { max: 12, perMs: 60000 }, // 12 req/60s (minimum limit)\n}\n\n/**\n * Clean up old timestamps outside the sliding window\n */\nfunction cleanupTimestamps(\n timestamps: number[],\n windowMs: number,\n now: number,\n): number[] {\n return timestamps.filter(ts => now - ts < windowMs)\n}\n\n/**\n * Calculate when we can make the next request without exceeding the rate limit\n */\nfunction getNextAvailableTime(\n timestamps: number[],\n max: number,\n windowMs: number,\n now: number,\n): number {\n if (timestamps.length < max) {\n return now // Can execute immediately\n }\n\n // We're at the limit. Find when the oldest request will expire\n const oldestTimestamp = timestamps[0]\n if (!oldestTimestamp) {\n return now // Shouldn't happen, but fallback to now\n }\n return oldestTimestamp + windowMs\n}\n\n/**\n * Public helper to schedule a request according to the category's limits.\n * Rate limits are enforced per clientId, so multiple instances with the same\n * clientId will share rate limiters, while different clientIds get separate limiters.\n *\n * This implementation uses a sliding window to track all requests within the time window.\n */\nexport function queueRequest<T>(\n clientId: string,\n category: ApiCategory,\n task: Task<T>,\n debug = false,\n): Promise<T> {\n const key = `${clientId}:${category}`\n if (!queues[key]) {\n queues[key] = {\n running: 0,\n queue: [],\n requestTimestamps: [],\n nextTimer: null,\n }\n }\n\n const queue = queues[key]!\n const { max, perMs } = LIMITS[category] ?? LIMITS.default\n\n return new Promise<T>((resolve, reject) => {\n const run = () => {\n const now = Date.now()\n\n // Clean up old timestamps before checking\n queue.requestTimestamps = cleanupTimestamps(\n queue.requestTimestamps,\n perMs,\n now,\n )\n\n // Check if we can execute now\n if (queue.requestTimestamps.length >= max) {\n // We've hit the rate limit, need to wait\n const nextAvailable = getNextAvailableTime(\n queue.requestTimestamps,\n max,\n perMs,\n now,\n )\n const waitTime = nextAvailable - now\n\n if (debug) {\n console.log(\n `[apiQueue] 🚫 Rate limit reached (${queue.requestTimestamps.length}/${max} in last ${perMs}ms). Queuing request. Need to wait ${waitTime.toFixed(0)}ms. Queue size: ${queue.queue.length + 1}`,\n )\n }\n\n // Re-queue this request to try again later\n queue.queue.push({ run, addedAt: now })\n // Don't call processQueue here - let the scheduled timer handle it\n // to avoid potential requeue loops\n return\n }\n\n // Record this request timestamp\n queue.requestTimestamps.push(now)\n queue.running++\n\n if (debug) {\n console.log(\n `[apiQueue] ▶️ Executing request (${category}). Requests in window: ${queue.requestTimestamps.length}/${max}. Running: ${queue.running}. Queue size: ${queue.queue.length}`,\n )\n }\n\n task()\n .then(resolve)\n .catch(reject)\n .finally(() => {\n queue.running--\n if (debug) {\n console.log(\n `[apiQueue] ✅ Request completed (${category}). Requests in window: ${queue.requestTimestamps.length}/${max}. Running: ${queue.running}. Queue size: ${queue.queue.length}`,\n )\n }\n processQueue(key, debug, category)\n })\n }\n\n // Add to queue or run immediately\n const now = Date.now()\n queue.requestTimestamps = cleanupTimestamps(\n queue.requestTimestamps,\n perMs,\n now,\n )\n\n if (queue.queue.length > 0 || queue.requestTimestamps.length >= max) {\n // Either there's already a queue, or we're at the rate limit\n queue.queue.push({ run, addedAt: now })\n if (debug) {\n console.log(\n `[apiQueue] ⏳ Queued request (${category}). Requests in window: ${queue.requestTimestamps.length}/${max}. Queue size: ${queue.queue.length}`,\n )\n }\n processQueue(key, debug, category)\n } else {\n // Can run immediately\n run()\n }\n })\n}\n\nfunction processQueue(key: string, debug: boolean, category: ApiCategory) {\n const queue = queues[key]\n if (!queue || queue.queue.length === 0) return\n\n const { max, perMs } = LIMITS[category] ?? LIMITS.default\n const now = Date.now()\n\n // Clean up old timestamps\n queue.requestTimestamps = cleanupTimestamps(\n queue.requestTimestamps,\n perMs,\n now,\n )\n\n // Check if we can process the next request\n if (queue.requestTimestamps.length < max) {\n // We have capacity, process immediately\n const next = queue.queue.shift()\n if (next) {\n if (debug) {\n const waitTime = Date.now() - next.addedAt\n console.log(\n `[apiQueue] 🚀 Processing queued request (${category}). Waited: ${waitTime.toFixed(0)}ms. Queue size: ${queue.queue.length}`,\n )\n }\n next.run()\n }\n // After one runs, immediately try again (in case there's more space)\n if (queue.queue.length > 0) {\n processQueue(key, debug, category)\n }\n } else {\n // We're at capacity, schedule for when the oldest request expires\n const nextAvailable = getNextAvailableTime(\n queue.requestTimestamps,\n max,\n perMs,\n now,\n )\n const delay = Math.max(0, nextAvailable - now + 50) // +50ms buffer for timer precision\n\n if (debug) {\n console.log(\n `[apiQueue] ⏸️ Delaying queue processing (${category}). Will retry in ${delay.toFixed(0)}ms. Queue size: ${queue.queue.length}`,\n )\n }\n\n // Only schedule a timer if one isn't already scheduled\n // This prevents timer pile-up during bursts\n if (!queue.nextTimer) {\n queue.nextTimer = setTimeout(() => {\n queue.nextTimer = null\n processQueue(key, debug, category)\n }, delay)\n }\n }\n}\n\n/**\n * Cleanup function to clear all queues and timers for a specific client.\n * Useful for testing or cleanup on application shutdown.\n */\nexport function clearQueue(clientId: string, category?: ApiCategory): void {\n if (category) {\n const key = `${clientId}:${category}`\n const queue = queues[key]\n if (queue) {\n if (queue.nextTimer) {\n clearTimeout(queue.nextTimer)\n queue.nextTimer = null\n }\n queue.queue = []\n queue.requestTimestamps = []\n queue.running = 0\n }\n } else {\n // Clear all queues for this client\n Object.keys(queues).forEach(key => {\n if (key.startsWith(`${clientId}:`)) {\n const queue = queues[key]\n if (queue?.nextTimer) {\n clearTimeout(queue.nextTimer)\n queue.nextTimer = null\n }\n delete queues[key]\n }\n })\n }\n}\n\n/**\n * Very naive path-based category detection. If no matcher fits, the `default` category\n * (effectively unlimited) is returned. Adjust these heuristics as your API surface evolves.\n */\nexport function categorizeRequest(\n path: string,\n method: string = 'GET',\n): ApiCategory {\n const cleanPath = path.toLowerCase()\n const isPost = method?.toUpperCase() === 'POST'\n\n if (cleanPath.includes('/documentsubmissions')) {\n return isPost ? 'submitDocuments' : 'getSubmission'\n }\n\n // -----------------------------\n // v1.0 API endpoint matchers\n // -----------------------------\n\n // Get Recent Documents - /api/v1.0/documents/recent\n if (cleanPath.includes('/documents/recent')) {\n return 'getRecentDocuments'\n }\n\n // Search Documents - /api/v1.0/documents/search\n if (cleanPath.includes('/documents/search')) {\n return 'searchDocuments'\n }\n\n // Document state actions (cancel/reject) - PUT /api/v1.0/documents/state/{uuid}/state\n // Both cancel and reject use the same endpoint, differentiated only by request body\n if (cleanPath.includes('/documents/state/') && cleanPath.endsWith('/state')) {\n // Both cancelDocument and rejectDocument share the same rate limit (12 RPM)\n // Use cancelDocument category for both since they share the same bucket\n return 'cancelDocument'\n }\n\n // Document raw content - /api/v1.0/documents/{uuid}/raw\n if (/\\/documents\\/[^/]+\\/raw$/.test(cleanPath)) {\n return 'getDocument'\n }\n\n // Document details - /api/v1.0/documents/{uuid}/details\n if (/\\/documents\\/[^/]+\\/details$/.test(cleanPath)) {\n return 'getDocumentDetails'\n }\n\n // Taxpayer TIN search & validation share same limit bucket\n if (cleanPath.includes('/taxpayer/search/tin')) return 'searchTin'\n if (cleanPath.includes('/taxpayer/validate/')) return 'searchTin'\n\n // Taxpayer QR code info\n if (cleanPath.includes('/taxpayer/qrcode')) return 'taxpayerQr'\n\n // Legacy matchers (kept for backward compatibility)\n if (cleanPath.includes('/searchtin')) return 'searchTin'\n if (cleanPath.includes('/qrcode')) return 'taxpayerQr'\n if (cleanPath.includes('/connect/token')) {\n return 'loginTaxpayer'\n }\n\n return 'default'\n}\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"apiQueue-C45w-ymm.cjs","names":["queues: Record<string, Queue>","LIMITS: Record<ApiCategory, { max: number; perMs: number }>","timestamps: number[]","windowMs: number","now: number","max: number","clientId: string","category: ApiCategory","task: Task<T>","now","key: string","debug: boolean","category?: ApiCategory","path: string","method: string"],"sources":["../src/utils/apiQueue.ts"],"sourcesContent":["// A very small utility that provides per-endpoint request queuing with rate-limits.\n// The goal is to make sure that we never exceed the vendor-defined limits while also ensuring\n// that every request is eventually executed.\n//\n// NOTE: This is intentionally minimal – no external dependencies are introduced.\n// If you need more advanced features (persistence, jitter, etc.) consider a library such as `bottleneck`.\n\n/*\nRate-limit specification (per 60-second window)\n----------------------------------------------\nLogin as Taxpayer System : 12\nLogin as Intermediary System : 12\nSubmit Documents : 100\nGet Submission : 300\nCancel Document : 12\nReject Document : 12\nGet Document : 60\nGet Document Details : 125\nGet Recent Documents : 12\nSearch Documents : 12\nSearch Taxpayer's TIN : 60\nTaxpayer's QR Code : 60\n*/\n\nexport type ApiCategory =\n | 'loginTaxpayer'\n | 'loginIntermediary'\n | 'submitDocuments'\n | 'getSubmission'\n | 'cancelDocument'\n | 'rejectDocument'\n | 'getDocument'\n | 'getDocumentDetails'\n | 'getRecentDocuments'\n | 'searchDocuments'\n | 'searchTin'\n | 'taxpayerQr'\n | 'default'\n\ntype Task<T> = () => Promise<T>\n\ninterface Queue {\n running: number\n queue: Array<{ run: () => void; addedAt: number }>\n requestTimestamps: number[] // Track all request timestamps in the sliding window\n nextTimer: NodeJS.Timeout | null // Track scheduled timer to avoid pile-up\n}\n\nconst queues: Record<string, Queue> = {}\n\n// Rate limits: max requests per time window (in ms)\nconst LIMITS: Record<ApiCategory, { max: number; perMs: number }> = {\n loginTaxpayer: { max: 12, perMs: 60000 }, // 12 req/60s\n loginIntermediary: { max: 12, perMs: 60000 }, // 12 req/60s\n submitDocuments: { max: 100, perMs: 60000 }, // 100 req/60s\n getSubmission: { max: 300, perMs: 60000 }, // 300 req/60s\n cancelDocument: { max: 12, perMs: 60000 }, // 12 req/60s\n rejectDocument: { max: 12, perMs: 60000 }, // 12 req/60s\n getDocument: { max: 60, perMs: 60000 }, // 60 req/60s\n getDocumentDetails: { max: 125, perMs: 60000 }, // 125 req/60s\n getRecentDocuments: { max: 12, perMs: 60000 }, // 12 req/60s\n searchDocuments: { max: 12, perMs: 60000 }, // 12 req/60s\n searchTin: { max: 60, perMs: 60000 }, // 60 req/60s\n taxpayerQr: { max: 60, perMs: 60000 }, // 60 req/60s\n default: { max: 12, perMs: 60000 }, // 12 req/60s (minimum limit)\n}\n\n/**\n * Clean up old timestamps outside the sliding window\n */\nfunction cleanupTimestamps(\n timestamps: number[],\n windowMs: number,\n now: number,\n): number[] {\n return timestamps.filter(ts => now - ts < windowMs)\n}\n\n/**\n * Calculate when we can make the next request without exceeding the rate limit\n */\nfunction getNextAvailableTime(\n timestamps: number[],\n max: number,\n windowMs: number,\n now: number,\n): number {\n if (timestamps.length < max) {\n return now // Can execute immediately\n }\n\n // We're at the limit. Find when the oldest request will expire\n const oldestTimestamp = timestamps[0]\n if (!oldestTimestamp) {\n return now // Shouldn't happen, but fallback to now\n }\n return oldestTimestamp + windowMs\n}\n\n/**\n * Public helper to schedule a request according to the category's limits.\n * Rate limits are enforced per clientId, so multiple instances with the same\n * clientId will share rate limiters, while different clientIds get separate limiters.\n *\n * This implementation uses a sliding window to track all requests within the time window.\n */\nexport function queueRequest<T>(\n clientId: string,\n category: ApiCategory,\n task: Task<T>,\n debug = false,\n): Promise<T> {\n const key = `${clientId}:${category}`\n if (!queues[key]) {\n queues[key] = {\n running: 0,\n queue: [],\n requestTimestamps: [],\n nextTimer: null,\n }\n }\n\n const queue = queues[key]!\n const { max, perMs } = LIMITS[category] ?? LIMITS.default\n\n return new Promise<T>((resolve, reject) => {\n const run = () => {\n const now = Date.now()\n\n // Clean up old timestamps before checking\n queue.requestTimestamps = cleanupTimestamps(\n queue.requestTimestamps,\n perMs,\n now,\n )\n\n // Check if we can execute now\n if (queue.requestTimestamps.length >= max) {\n // We've hit the rate limit, need to wait\n const nextAvailable = getNextAvailableTime(\n queue.requestTimestamps,\n max,\n perMs,\n now,\n )\n const waitTime = nextAvailable - now\n\n if (debug) {\n console.log(\n `[apiQueue] 🚫 Rate limit reached (${queue.requestTimestamps.length}/${max} in last ${perMs}ms). Queuing request. Need to wait ${waitTime.toFixed(0)}ms. Queue size: ${queue.queue.length + 1}`,\n )\n }\n\n // Re-queue this request to try again later\n queue.queue.push({ run, addedAt: now })\n // Don't call processQueue here - let the scheduled timer handle it\n // to avoid potential requeue loops\n return\n }\n\n // Record this request timestamp\n queue.requestTimestamps.push(now)\n queue.running++\n\n if (debug) {\n console.log(\n `[apiQueue] ▶️ Executing request (${category}). Requests in window: ${queue.requestTimestamps.length}/${max}. Running: ${queue.running}. Queue size: ${queue.queue.length}`,\n )\n }\n\n task()\n .then(resolve)\n .catch(reject)\n .finally(() => {\n queue.running--\n if (debug) {\n console.log(\n `[apiQueue] ✅ Request completed (${category}). Requests in window: ${queue.requestTimestamps.length}/${max}. Running: ${queue.running}. Queue size: ${queue.queue.length}`,\n )\n }\n processQueue(key, debug, category)\n })\n }\n\n // Add to queue or run immediately\n const now = Date.now()\n queue.requestTimestamps = cleanupTimestamps(\n queue.requestTimestamps,\n perMs,\n now,\n )\n\n if (queue.queue.length > 0 || queue.requestTimestamps.length >= max) {\n // Either there's already a queue, or we're at the rate limit\n queue.queue.push({ run, addedAt: now })\n if (debug) {\n console.log(\n `[apiQueue] ⏳ Queued request (${category}). Requests in window: ${queue.requestTimestamps.length}/${max}. Queue size: ${queue.queue.length}`,\n )\n }\n processQueue(key, debug, category)\n } else {\n // Can run immediately\n run()\n }\n })\n}\n\nfunction processQueue(key: string, debug: boolean, category: ApiCategory) {\n const queue = queues[key]\n if (!queue || queue.queue.length === 0) return\n\n const { max, perMs } = LIMITS[category] ?? LIMITS.default\n const now = Date.now()\n\n // Clean up old timestamps\n queue.requestTimestamps = cleanupTimestamps(\n queue.requestTimestamps,\n perMs,\n now,\n )\n\n // Check if we can process the next request\n if (queue.requestTimestamps.length < max) {\n // We have capacity, process immediately\n const next = queue.queue.shift()\n if (next) {\n if (debug) {\n const waitTime = Date.now() - next.addedAt\n console.log(\n `[apiQueue] 🚀 Processing queued request (${category}). Waited: ${waitTime.toFixed(0)}ms. Queue size: ${queue.queue.length}`,\n )\n }\n next.run()\n }\n // After one runs, immediately try again (in case there's more space)\n if (queue.queue.length > 0) {\n processQueue(key, debug, category)\n }\n } else {\n // We're at capacity, schedule for when the oldest request expires\n const nextAvailable = getNextAvailableTime(\n queue.requestTimestamps,\n max,\n perMs,\n now,\n )\n const delay = Math.max(0, nextAvailable - now + 50) // +50ms buffer for timer precision\n\n if (debug) {\n console.log(\n `[apiQueue] ⏸️ Delaying queue processing (${category}). Will retry in ${delay.toFixed(0)}ms. Queue size: ${queue.queue.length}`,\n )\n }\n\n // Only schedule a timer if one isn't already scheduled\n // This prevents timer pile-up during bursts\n if (!queue.nextTimer) {\n queue.nextTimer = setTimeout(() => {\n queue.nextTimer = null\n processQueue(key, debug, category)\n }, delay)\n }\n }\n}\n\n/**\n * Cleanup function to clear all queues and timers for a specific client.\n * Useful for testing or cleanup on application shutdown.\n */\nexport function clearQueue(clientId: string, category?: ApiCategory): void {\n if (category) {\n const key = `${clientId}:${category}`\n const queue = queues[key]\n if (queue) {\n if (queue.nextTimer) {\n clearTimeout(queue.nextTimer)\n queue.nextTimer = null\n }\n queue.queue = []\n queue.requestTimestamps = []\n queue.running = 0\n }\n } else {\n // Clear all queues for this client\n Object.keys(queues).forEach(key => {\n if (key.startsWith(`${clientId}:`)) {\n const queue = queues[key]\n if (queue?.nextTimer) {\n clearTimeout(queue.nextTimer)\n queue.nextTimer = null\n }\n delete queues[key]\n }\n })\n }\n}\n\n/**\n * Very naive path-based category detection. If no matcher fits, the `default` category\n * (effectively unlimited) is returned. Adjust these heuristics as your API surface evolves.\n */\nexport function categorizeRequest(\n path: string,\n method: string = 'GET',\n): ApiCategory {\n const cleanPath = path.toLowerCase()\n const isPost = method?.toUpperCase() === 'POST'\n\n if (cleanPath.includes('/documentsubmissions')) {\n return isPost ? 'submitDocuments' : 'getSubmission'\n }\n\n // -----------------------------\n // v1.0 API endpoint matchers\n // -----------------------------\n\n // Get Recent Documents - /api/v1.0/documents/recent\n if (cleanPath.includes('/documents/recent')) {\n return 'getRecentDocuments'\n }\n\n // Search Documents - /api/v1.0/documents/search\n if (cleanPath.includes('/documents/search')) {\n return 'searchDocuments'\n }\n\n // Document state actions (cancel/reject) - PUT /api/v1.0/documents/state/{uuid}/state\n // Both cancel and reject use the same endpoint, differentiated only by request body\n if (cleanPath.includes('/documents/state/') && cleanPath.endsWith('/state')) {\n // Both cancelDocument and rejectDocument share the same rate limit (12 RPM)\n // Use cancelDocument category for both since they share the same bucket\n return 'cancelDocument'\n }\n\n // Document raw content - /api/v1.0/documents/{uuid}/raw\n if (/\\/documents\\/[^/]+\\/raw$/.test(cleanPath)) {\n return 'getDocument'\n }\n\n // Document details - /api/v1.0/documents/{uuid}/details\n if (/\\/documents\\/[^/]+\\/details$/.test(cleanPath)) {\n return 'getDocumentDetails'\n }\n\n // Taxpayer TIN search & validation share same limit bucket\n if (cleanPath.includes('/taxpayer/search/tin')) return 'searchTin'\n if (cleanPath.includes('/taxpayer/validate/')) return 'searchTin'\n\n // Taxpayer QR code info\n if (cleanPath.includes('/taxpayer/qrcode')) return 'taxpayerQr'\n\n // Legacy matchers (kept for backward compatibility)\n if (cleanPath.includes('/searchtin')) return 'searchTin'\n if (cleanPath.includes('/qrcode')) return 'taxpayerQr'\n if (cleanPath.includes('/connect/token')) {\n return 'loginTaxpayer'\n }\n\n return 'default'\n}\n"],"mappings":"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"}
|
|
@@ -1,30 +1,10 @@
|
|
|
1
|
-
const
|
|
2
|
-
const crypto =
|
|
3
|
-
const node_forge =
|
|
4
|
-
const fs =
|
|
1
|
+
const require_document = require('./document-B11B5lqd.cjs');
|
|
2
|
+
const crypto = require_document.__toESM(require("crypto"));
|
|
3
|
+
const node_forge = require_document.__toESM(require("node-forge"));
|
|
4
|
+
const fs = require_document.__toESM(require("fs"));
|
|
5
5
|
|
|
6
6
|
//#region src/utils/certificate.ts
|
|
7
7
|
/**
|
|
8
|
-
* Extracts certificate information from PEM certificate
|
|
9
|
-
* Converts hex serial number to decimal as required by MyInvois
|
|
10
|
-
*
|
|
11
|
-
* @param certificatePem - Certificate in PEM format
|
|
12
|
-
* @returns Object with issuer name and decimal serial number
|
|
13
|
-
*/
|
|
14
|
-
const extractCertificateInfo = (certificatePem) => {
|
|
15
|
-
try {
|
|
16
|
-
const cert = new crypto.default.X509Certificate(certificatePem);
|
|
17
|
-
const serialNumberHex = cert.serialNumber;
|
|
18
|
-
const serialNumberDecimal = BigInt("0x" + serialNumberHex).toString();
|
|
19
|
-
return {
|
|
20
|
-
issuerName: cert.issuer,
|
|
21
|
-
serialNumber: serialNumberDecimal
|
|
22
|
-
};
|
|
23
|
-
} catch (error) {
|
|
24
|
-
throw new Error(`Failed to extract certificate info: ${error}`);
|
|
25
|
-
}
|
|
26
|
-
};
|
|
27
|
-
/**
|
|
28
8
|
* Validates that certificate and private key form a valid pair
|
|
29
9
|
*
|
|
30
10
|
* @param certificatePem - Certificate in PEM format
|
|
@@ -69,12 +49,6 @@ const getPemFromP12 = (p12Input, passphrase) => {
|
|
|
69
49
|
};
|
|
70
50
|
|
|
71
51
|
//#endregion
|
|
72
|
-
Object.defineProperty(exports, 'extractCertificateInfo', {
|
|
73
|
-
enumerable: true,
|
|
74
|
-
get: function () {
|
|
75
|
-
return extractCertificateInfo;
|
|
76
|
-
}
|
|
77
|
-
});
|
|
78
52
|
Object.defineProperty(exports, 'getPemFromP12', {
|
|
79
53
|
enumerable: true,
|
|
80
54
|
get: function () {
|
|
@@ -87,4 +61,4 @@ Object.defineProperty(exports, 'validateKeyPair', {
|
|
|
87
61
|
return validateKeyPair;
|
|
88
62
|
}
|
|
89
63
|
});
|
|
90
|
-
//# sourceMappingURL=certificate-
|
|
64
|
+
//# sourceMappingURL=certificate-Dw46fkYv.cjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"certificate-Dw46fkYv.cjs","names":["certificatePem: string","privateKeyPem: string","p12Input: Buffer | string","passphrase: string","p12Buffer: Buffer","forge"],"sources":["../src/utils/certificate.ts"],"sourcesContent":["import crypto from 'crypto'\nimport * as forge from 'node-forge'\nimport fs from 'fs'\n\n// Single source of truth for issuer name / serial number extraction. The signer\n// (utils/document) owns the canonical normalization (RDN order + XML escaping)\n// that MyInvois validates against (DS326), so re-export it here rather than\n// maintaining a second, divergent implementation.\nexport { extractCertificateInfo } from './document'\n\n/**\n * Validates that certificate and private key form a valid pair\n *\n * @param certificatePem - Certificate in PEM format\n * @param privateKeyPem - Private key in PEM format\n * @returns Boolean indicating if key pair is valid\n */\nexport const validateKeyPair = (\n certificatePem: string,\n privateKeyPem: string,\n): boolean => {\n try {\n const cert = new crypto.X509Certificate(certificatePem)\n const publicKey = cert.publicKey\n\n // Test signature/verification cycle\n const testData = 'validation-test-data'\n const signer = crypto.createSign('RSA-SHA256')\n signer.update(testData)\n const signature = signer.sign(privateKeyPem)\n\n const verifier = crypto.createVerify('RSA-SHA256')\n verifier.update(testData)\n return verifier.verify(publicKey, signature)\n } catch (error) {\n console.error('Key pair validation failed:', error)\n return false\n }\n}\n\nexport const getPemFromP12 = (\n p12Input: Buffer | string,\n passphrase: string,\n): {\n certificatePem: string\n privateKeyPem: string\n} => {\n try {\n // Load binary DER contents of the p12 file\n const p12Buffer: Buffer = Buffer.isBuffer(p12Input)\n ? p12Input\n : fs.readFileSync(p12Input)\n\n // Convert DER -> ASN.1\n const p12Der = forge.util.createBuffer(p12Buffer.toString('binary'))\n const p12Asn1 = forge.asn1.fromDer(p12Der.getBytes())\n\n // Parse the PKCS#12 using the provided passphrase\n const p12 = forge.pkcs12.pkcs12FromAsn1(p12Asn1, passphrase)\n\n // Extract the first certificate (assumes it is the signing cert)\n const certBags = p12.getBags({ bagType: forge.pki.oids.certBag })[\n forge.pki.oids.certBag\n ]\n if (!certBags || certBags.length === 0)\n throw new Error('No certificate found in PKCS#12 bundle')\n const certificatePem = forge.pki.certificateToPem(certBags[0].cert)\n\n // Extract the first private key\n const keyBags = p12.getBags({\n bagType: forge.pki.oids.pkcs8ShroudedKeyBag,\n })[forge.pki.oids.pkcs8ShroudedKeyBag]\n if (!keyBags || keyBags.length === 0)\n throw new Error('No private key found in PKCS#12 bundle')\n const privateKeyPem = forge.pki.privateKeyToPem(keyBags[0].key)\n\n return { certificatePem, privateKeyPem }\n } catch (error) {\n throw new Error(`Failed to extract PEM from PKCS#12 file: ${error}`)\n }\n}\n"],"mappings":";;;;;;;;;;;;;AAiBA,MAAa,kBAAkB,CAC7BA,gBACAC,kBACY;AACZ,KAAI;EACF,MAAM,OAAO,IAAI,eAAO,gBAAgB;EACxC,MAAM,YAAY,KAAK;EAGvB,MAAM,WAAW;EACjB,MAAM,SAAS,eAAO,WAAW,aAAa;AAC9C,SAAO,OAAO,SAAS;EACvB,MAAM,YAAY,OAAO,KAAK,cAAc;EAE5C,MAAM,WAAW,eAAO,aAAa,aAAa;AAClD,WAAS,OAAO,SAAS;AACzB,SAAO,SAAS,OAAO,WAAW,UAAU;CAC7C,SAAQ,OAAO;AACd,UAAQ,MAAM,+BAA+B,MAAM;AACnD,SAAO;CACR;AACF;AAED,MAAa,gBAAgB,CAC3BC,UACAC,eAIG;AACH,KAAI;EAEF,MAAMC,YAAoB,OAAO,SAAS,SAAS,GAC/C,WACA,WAAG,aAAa,SAAS;EAG7B,MAAM,SAAS,WAAM,KAAK,aAAa,UAAU,SAAS,SAAS,CAAC;EACpE,MAAM,UAAU,WAAM,KAAK,QAAQ,OAAO,UAAU,CAAC;EAGrD,MAAM,MAAM,WAAM,OAAO,eAAe,SAAS,WAAW;EAG5D,MAAM,WAAW,IAAI,QAAQ,EAAE,SAASC,WAAM,IAAI,KAAK,QAAS,EAAC,CAC/DA,WAAM,IAAI,KAAK;AAEjB,OAAK,YAAY,SAAS,WAAW,EACnC,OAAM,IAAI,MAAM;EAClB,MAAM,iBAAiB,WAAM,IAAI,iBAAiB,SAAS,GAAG,KAAK;EAGnE,MAAM,UAAU,IAAI,QAAQ,EAC1B,SAASA,WAAM,IAAI,KAAK,oBACzB,EAAC,CAACA,WAAM,IAAI,KAAK;AAClB,OAAK,WAAW,QAAQ,WAAW,EACjC,OAAM,IAAI,MAAM;EAClB,MAAM,gBAAgB,WAAM,IAAI,gBAAgB,QAAQ,GAAG,IAAI;AAE/D,SAAO;GAAE;GAAgB;EAAe;CACzC,SAAQ,OAAO;AACd,QAAM,IAAI,OAAO,2CAA2C,MAAM;CACnE;AACF"}
|
|
@@ -1,6 +1,28 @@
|
|
|
1
|
-
|
|
1
|
+
//#region rolldown:runtime
|
|
2
|
+
var __create = Object.create;
|
|
3
|
+
var __defProp = Object.defineProperty;
|
|
4
|
+
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
5
|
+
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
6
|
+
var __getProtoOf = Object.getPrototypeOf;
|
|
7
|
+
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
8
|
+
var __copyProps = (to, from, except, desc) => {
|
|
9
|
+
if (from && typeof from === "object" || typeof from === "function") for (var keys = __getOwnPropNames(from), i = 0, n = keys.length, key; i < n; i++) {
|
|
10
|
+
key = keys[i];
|
|
11
|
+
if (!__hasOwnProp.call(to, key) && key !== except) __defProp(to, key, {
|
|
12
|
+
get: ((k) => from[k]).bind(null, key),
|
|
13
|
+
enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable
|
|
14
|
+
});
|
|
15
|
+
}
|
|
16
|
+
return to;
|
|
17
|
+
};
|
|
18
|
+
var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", {
|
|
19
|
+
value: mod,
|
|
20
|
+
enumerable: true
|
|
21
|
+
}) : target, mod));
|
|
22
|
+
|
|
23
|
+
//#endregion
|
|
2
24
|
const require_formatIdValue = require('./formatIdValue-i67o4kyD.cjs');
|
|
3
|
-
const crypto =
|
|
25
|
+
const crypto = __toESM(require("crypto"));
|
|
4
26
|
|
|
5
27
|
//#region src/utils/document.ts
|
|
6
28
|
/**
|
|
@@ -71,6 +93,29 @@ const canonicalizeJSON = (obj) => {
|
|
|
71
93
|
*/
|
|
72
94
|
const generateCleanInvoiceObject = (invoice) => {
|
|
73
95
|
const lineItems = getLineItems(invoice);
|
|
96
|
+
const round2 = (n) => Math.round(n * 100) / 100;
|
|
97
|
+
const taxCategoryOrder = [];
|
|
98
|
+
const taxByCategory = /* @__PURE__ */ new Map();
|
|
99
|
+
for (const item of lineItems) {
|
|
100
|
+
const category = item.taxType;
|
|
101
|
+
if (!taxByCategory.has(category)) {
|
|
102
|
+
taxByCategory.set(category, {
|
|
103
|
+
taxable: 0,
|
|
104
|
+
tax: 0
|
|
105
|
+
});
|
|
106
|
+
taxCategoryOrder.push(category);
|
|
107
|
+
}
|
|
108
|
+
const bucket = taxByCategory.get(category);
|
|
109
|
+
bucket.taxable += item.totalTaxableAmountPerLine ?? 0;
|
|
110
|
+
bucket.tax += item.taxAmount ?? 0;
|
|
111
|
+
}
|
|
112
|
+
if (taxCategoryOrder.length === 0) {
|
|
113
|
+
taxCategoryOrder.push(lineItems[0]?.taxType || "01");
|
|
114
|
+
taxByCategory.set(taxCategoryOrder[0], {
|
|
115
|
+
taxable: invoice.legalMonetaryTotal.taxExclusiveAmount,
|
|
116
|
+
tax: invoice.taxTotal.taxAmount
|
|
117
|
+
});
|
|
118
|
+
}
|
|
74
119
|
return {
|
|
75
120
|
ID: [{ _: invoice.eInvoiceCodeOrNumber }],
|
|
76
121
|
IssueDate: [{ _: invoice.eInvoiceDate }],
|
|
@@ -142,24 +187,27 @@ const generateCleanInvoiceObject = (invoice) => {
|
|
|
142
187
|
_: invoice.taxTotal.taxAmount,
|
|
143
188
|
currencyID: invoice.invoiceCurrencyCode
|
|
144
189
|
}],
|
|
145
|
-
TaxSubtotal:
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
_:
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
190
|
+
TaxSubtotal: taxCategoryOrder.map((category) => {
|
|
191
|
+
const { taxable, tax } = taxByCategory.get(category);
|
|
192
|
+
return {
|
|
193
|
+
TaxableAmount: [{
|
|
194
|
+
_: round2(taxable),
|
|
195
|
+
currencyID: invoice.invoiceCurrencyCode
|
|
196
|
+
}],
|
|
197
|
+
TaxAmount: [{
|
|
198
|
+
_: round2(tax),
|
|
199
|
+
currencyID: invoice.invoiceCurrencyCode
|
|
200
|
+
}],
|
|
201
|
+
TaxCategory: [{
|
|
202
|
+
ID: [{ _: category }],
|
|
203
|
+
TaxScheme: [{ ID: [{
|
|
204
|
+
_: "OTH",
|
|
205
|
+
schemeAgencyID: "6",
|
|
206
|
+
schemeID: "UN/ECE 5153"
|
|
207
|
+
}] }]
|
|
208
|
+
}]
|
|
209
|
+
};
|
|
210
|
+
})
|
|
163
211
|
}],
|
|
164
212
|
LegalMonetaryTotal: [{
|
|
165
213
|
LineExtensionAmount: [{
|
|
@@ -624,6 +672,12 @@ const calculateInvoiceTotals = (lineItems, payableRoundingAmount = 0) => {
|
|
|
624
672
|
};
|
|
625
673
|
|
|
626
674
|
//#endregion
|
|
675
|
+
Object.defineProperty(exports, '__toESM', {
|
|
676
|
+
enumerable: true,
|
|
677
|
+
get: function () {
|
|
678
|
+
return __toESM;
|
|
679
|
+
}
|
|
680
|
+
});
|
|
627
681
|
Object.defineProperty(exports, 'calculateCertificateDigest', {
|
|
628
682
|
enumerable: true,
|
|
629
683
|
get: function () {
|
|
@@ -738,4 +792,4 @@ Object.defineProperty(exports, 'transformDocumentForHashing', {
|
|
|
738
792
|
return transformDocumentForHashing;
|
|
739
793
|
}
|
|
740
794
|
});
|
|
741
|
-
//# sourceMappingURL=document-
|
|
795
|
+
//# sourceMappingURL=document-B11B5lqd.cjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"document-B11B5lqd.cjs","names":["item: InvoiceLineItem","doc: AllDocumentsV1_1","obj: unknown","sortedObj: Record<string, unknown>","invoice: AllDocumentsV1_1","n: number","taxCategoryOrder: string[]","invoices: AllDocumentsV1_1[]","invoice: InvoiceSubmission","certificatePem: string","X509Certificate","str: string","issuer: string","serialHex: string","error: unknown","certificateDigest: string","signingTime: string","issuerName: string","serialNumber: string","signedProperties: SignedPropertiesObject","useTargetWrapper: boolean","digestObj: unknown","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 measurement?: string\n taxType: TaxTypeCode\n taxRate: number\n totalTaxableAmountPerLine?: number\n discountAmount?: number\n discountRate?: number\n}","params: {\n itemClassificationCode: ClassificationCode\n itemDescription: string\n unitPrice: number\n quantity?: number\n measurement?: string\n taxType: TaxTypeCode\n taxPerUnitAmount: number\n baseUnitMeasure: number\n baseUnitMeasureCode: UnitTypeCode\n totalTaxableAmountPerLine?: number\n discountAmount?: number\n discountRate?: number\n}","lineItems: InvoiceLineItem[]","payableRoundingAmount: number"],"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\n // Group line items by tax category so the document-level TaxTotal carries one\n // TaxSubtotal per distinct tax type (derived entirely from the line items).\n const round2 = (n: number) => Math.round(n * 100) / 100\n const taxCategoryOrder: string[] = []\n const taxByCategory = new Map<string, { taxable: number; tax: number }>()\n for (const item of lineItems) {\n const category = item.taxType\n if (!taxByCategory.has(category)) {\n taxByCategory.set(category, { taxable: 0, tax: 0 })\n taxCategoryOrder.push(category)\n }\n const bucket = taxByCategory.get(category)!\n bucket.taxable += item.totalTaxableAmountPerLine ?? 0\n bucket.tax += item.taxAmount ?? 0\n }\n // Fallback for the (invalid) no-line-items case: keep prior single-subtotal shape.\n if (taxCategoryOrder.length === 0) {\n taxCategoryOrder.push(lineItems[0]?.taxType || '01')\n taxByCategory.set(taxCategoryOrder[0]!, {\n taxable: invoice.legalMonetaryTotal.taxExclusiveAmount,\n tax: invoice.taxTotal.taxAmount,\n })\n }\n\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 // One TaxSubtotal per distinct tax category, derived from line items.\n TaxSubtotal: taxCategoryOrder.map(category => {\n const { taxable, tax } = taxByCategory.get(category)!\n return {\n TaxableAmount: [\n {\n _: round2(taxable),\n currencyID: invoice.invoiceCurrencyCode,\n },\n ],\n TaxAmount: [\n {\n _: round2(tax),\n currencyID: invoice.invoiceCurrencyCode,\n },\n ],\n TaxCategory: [\n {\n ID: [{ _: category }],\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 PayableRoundingAmount: [\n {\n _: invoice.legalMonetaryTotal.payableRoundingAmount,\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 // Quantity and Measurement (optional)\n ...(item.quantity !== undefined || item.measurement !== undefined\n ? {\n InvoicedQuantity: [\n {\n _: item.quantity ?? 1,\n ...(item.measurement !== undefined\n ? { unitCode: item.measurement }\n : {}),\n },\n ],\n }\n : {}),\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 _:\n (item.totalTaxableAmountPerLine ?? 0) +\n (item.discountAmount ?? 0),\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 // Line-level discount (AllowanceCharge with ChargeIndicator=false) – include only if a discount exists\n ...(item.discountAmount !== undefined || item.discountRate !== undefined\n ? {\n AllowanceCharge: [\n {\n ChargeIndicator: [{ _: false }],\n Amount: [\n {\n _: item.discountAmount ?? 0,\n currencyID: invoice.invoiceCurrencyCode,\n },\n ],\n ...(item.discountRate !== undefined\n ? {\n MultiplierFactorNumeric: [{ _: item.discountRate }],\n BaseAmount: [\n {\n _:\n (item.totalTaxableAmountPerLine ?? 0) +\n (item.discountAmount ?? 0),\n currencyID: invoice.invoiceCurrencyCode,\n },\n ],\n }\n : {}),\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 *\n * FIXED: Use regex-based minification to match PowerShell script exactly\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 // Convert to JSON string first\n const jsonString = JSON.stringify(documentForTransform)\n\n // Apply the exact same regex-based minification as PowerShell script\n // This regex preserves whitespace within quoted strings but removes all other whitespace\n const minifiedJson = jsonString.replace(\n /(\"(?:\\\\.|[^\"\\\\])*\")|\\s+/g,\n (match, quotedString) => {\n if (quotedString) {\n return quotedString // Keep string content exactly as-is\n } else {\n return '' // Remove all other whitespace\n }\n },\n )\n\n return minifiedJson\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 * FIXED: Match PowerShell script exactly - use raw certificate data like $cert.RawData\n */\nexport const calculateCertificateDigest = (certificatePem: string): string => {\n try {\n // Create X509Certificate object to get raw data (like PowerShell $cert.RawData)\n const cert = new X509Certificate(certificatePem)\n\n // Get the raw certificate data (DER-encoded, like PowerShell $cert.RawData)\n const rawCertificateData = cert.raw\n\n // Calculate SHA-256 hash of raw certificate data (like PowerShell $sha256.ComputeHash($cert.RawData))\n const hash = crypto.createHash('sha256')\n hash.update(rawCertificateData)\n\n // Return as Base64 (like PowerShell [Convert]::ToBase64String($certHash))\n return hash.digest('base64')\n } catch {\n // Fallback to the previous method if X509Certificate fails\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/**\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 // FIXED: Use raw issuer name like PowerShell $cert.IssuerName.Name\n // Apply XML escaping to match PowerShell [System.Security.SecurityElement]::Escape()\n const escapeXmlSpecialChars = (str: string): string => {\n return str\n .replace(/&/g, '&')\n .replace(/</g, '<')\n .replace(/>/g, '>')\n .replace(/\"/g, '"')\n .replace(/'/g, ''')\n }\n\n // FIXED: Normalize issuer name to match MyInvois expectations\n // The issuer name should be in the format: \"CN=Trial LHDNM Sub CA V1, OU=Terms of use at http://www.testcertcomp.com.my, O=LHDNM, C=MY\"\n const normalizeIssuerName = (issuer: 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 = issuer\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 // FIXED: Use raw subject name without normalization to match PowerShell $cert.SubjectName.Name\n // The subject name should be in the format shown in the specification\n const rawSubjectName = cert.subject\n\n return {\n issuerName: escapeXmlSpecialChars(normalizeIssuerName(cert.issuer)),\n serialNumber: formatSerialNumber(serialNumberHex),\n subjectName: rawSubjectName, // Use raw subject name like PowerShell\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 * FIXED: Calculate digest from SignedProperties only (without Target wrapper)\n */\nexport const calculateSignedPropertiesDigest = (\n signedProperties: SignedPropertiesObject,\n useTargetWrapper: boolean = true, // Changed to true - validator calculates digest from embedded structure with Target wrapper\n): string => {\n let digestObj: unknown\n if (useTargetWrapper) {\n digestObj = {\n Target: 'signature',\n SignedProperties: signedProperties.SignedProperties,\n }\n } else {\n digestObj = signedProperties.SignedProperties\n }\n\n // FIXED: Don't sort object keys - use exact structure as embedded\n // The validator calculates digest from the exact embedded structure\n const signedPropertiesString = JSON.stringify(digestObj)\n\n // Apply the same regex-based minification as the document transformation\n const minifiedSignedProperties = signedPropertiesString.replace(\n /(\"(?:\\\\.|[^\"\\\\])*\")|\\s+/g,\n (match, quotedString) => {\n if (quotedString) {\n return quotedString // Keep string content exactly as-is\n } else {\n return '' // Remove all other whitespace\n }\n },\n )\n\n const hash = crypto.createHash('sha256')\n hash.update(minifiedSignedProperties, '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 * FIXED: Match PowerShell script exactly - first compute hash, then sign the hash\n */\nexport const signDocumentString = (\n documentString: string,\n privateKeyPem: string,\n): string => {\n // Create signer with RSA-SHA256 (matches PowerShell RSAPKCS1SignatureFormatter)\n const signer = crypto.createSign('RSA-SHA256')\n signer.update(documentString, 'utf8')\n return signer.sign(privateKeyPem, '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 matching PowerShell exactly\n // PowerShell: Get-Date -Format \"yyyy-MM-ddTHH:mm:ssZ\"\n const now = new Date()\n const signingTime =\n now.getFullYear() +\n '-' +\n String(now.getMonth() + 1).padStart(2, '0') +\n '-' +\n String(now.getDate()).padStart(2, '0') +\n 'T' +\n String(now.getHours()).padStart(2, '0') +\n ':' +\n String(now.getMinutes()).padStart(2, '0') +\n ':' +\n String(now.getSeconds()).padStart(2, '0') +\n 'Z'\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 the dedicated function\n const propsDigest = calculateSignedPropertiesDigest(signedProperties)\n\n // Create simple SignedInfo structure (matching PowerShell approach)\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 // Extract certificate content using raw data (like PowerShell $cert.RawData)\n const cert = new X509Certificate(signingCredentials.certificatePem)\n const certificate = cert.raw.toString('base64')\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 measurement?: string\n taxType: TaxTypeCode\n taxRate: number\n totalTaxableAmountPerLine?: number\n discountAmount?: number\n discountRate?: number\n}): InvoiceLineItem => {\n const quantity = params.quantity\n ? params.quantity <= 0\n ? 1\n : (params.quantity ?? 1)\n : 1\n // Compute pre-discount from unit price and quantity only to avoid double-discounting\n const preDiscountAmount = params.unitPrice * quantity\n const hasDiscount =\n params.discountAmount !== undefined || params.discountRate !== undefined\n const computedDiscountByRate =\n hasDiscount && params.discountRate !== undefined\n ? preDiscountAmount * params.discountRate\n : 0\n const discountAmount =\n hasDiscount && params.discountAmount !== undefined\n ? params.discountAmount\n : computedDiscountByRate\n const totalTaxableAmount = Math.max(\n 0,\n preDiscountAmount - (hasDiscount ? discountAmount : 0),\n )\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 ...(params.quantity !== undefined ? { quantity: params.quantity } : {}),\n ...(params.measurement !== undefined\n ? { measurement: params.measurement }\n : {}),\n ...(hasDiscount && discountAmount !== undefined ? { discountAmount } : {}),\n ...(hasDiscount && params.discountRate !== undefined\n ? { discountRate: params.discountRate }\n : {}),\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 measurement?: string\n taxType: TaxTypeCode\n taxPerUnitAmount: number\n baseUnitMeasure: number\n baseUnitMeasureCode: UnitTypeCode\n totalTaxableAmountPerLine?: number\n discountAmount?: number\n discountRate?: number\n}): InvoiceLineItem => {\n const quantity = params.quantity || 1\n // Compute pre-discount from unit price and quantity only to avoid double-discounting\n const preDiscountAmount = params.unitPrice * quantity\n const hasDiscount =\n params.discountAmount !== undefined || params.discountRate !== undefined\n const computedDiscountByRate =\n hasDiscount && params.discountRate !== undefined\n ? preDiscountAmount * params.discountRate\n : 0\n const discountAmount =\n hasDiscount && params.discountAmount !== undefined\n ? params.discountAmount\n : computedDiscountByRate\n const totalTaxableAmount = Math.max(\n 0,\n preDiscountAmount - (hasDiscount ? discountAmount : 0),\n )\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 ...(params.quantity !== undefined ? { quantity: params.quantity } : {}),\n ...(params.measurement !== undefined\n ? { measurement: params.measurement }\n : {}),\n ...(hasDiscount && discountAmount !== undefined ? { discountAmount } : {}),\n ...(hasDiscount && params.discountRate !== undefined\n ? { discountRate: params.discountRate }\n : {}),\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 payableRoundingAmount: number = 0,\n): {\n legalMonetaryTotal: {\n taxExclusiveAmount: number\n taxInclusiveAmount: number\n payableRoundingAmount: 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 const payableAmount = taxInclusiveAmount + payableRoundingAmount\n\n return {\n legalMonetaryTotal: {\n taxExclusiveAmount: Math.round(taxExclusiveAmount * 100) / 100,\n taxInclusiveAmount: Math.round(taxInclusiveAmount * 100) / 100,\n payableRoundingAmount: Math.round(payableRoundingAmount * 100) / 100,\n payableAmount: Math.round(payableAmount * 100) / 100,\n },\n taxTotal: {\n taxAmount: Math.round(totalTaxAmount * 100) / 100,\n },\n }\n}\n"],"mappings":"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"}
|
|
@@ -40,7 +40,8 @@ async function searchDocuments(context, params) {
|
|
|
40
40
|
const response = await fetch(`/api/v1.0/documents/search?${queryParams.toString()}`);
|
|
41
41
|
const data = await response.json();
|
|
42
42
|
if (rawLogs) console.log("Raw search documents:", data);
|
|
43
|
-
|
|
43
|
+
const documents = Array.isArray(data) ? data : data?.result ?? [];
|
|
44
|
+
return documents.map((doc) => ({
|
|
44
45
|
...doc,
|
|
45
46
|
longId: doc.longID ?? doc.longId
|
|
46
47
|
}));
|
|
@@ -65,4 +66,4 @@ Object.defineProperty(exports, 'searchDocuments', {
|
|
|
65
66
|
return searchDocuments;
|
|
66
67
|
}
|
|
67
68
|
});
|
|
68
|
-
//# sourceMappingURL=documentManagement-
|
|
69
|
+
//# sourceMappingURL=documentManagement-qJnd0l1z.cjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"documentManagement-qJnd0l1z.cjs","names":["context: DocumentContext","documentUid: string","params: {\n uuid?: string\n submissionDateFrom: string\n submissionDateTo?: string\n pageSize?: number\n pageNo?: number\n issueDateFrom?: string\n issueDateTo?: string\n invoiceDirection?: 'Sent' | 'Received'\n status?: DocumentStatus\n documentType?: EInvoiceTypeCode\n searchQuery?: string\n }","documents: (DocumentSummary & { longID?: string })[]"],"sources":["../src/api/documentManagement.ts"],"sourcesContent":["import type {\n DocumentSummary,\n DocumentStatus,\n EInvoiceTypeCode,\n DocumentValidationResult,\n DocumentValidationStepResult,\n Fetch,\n} from '../types'\n\ninterface DocumentContext {\n fetch: Fetch\n}\n\nconst rawLogs = process.env.E_INV_RAW_LOGS === 'true'\n\nexport async function getDocument(\n context: DocumentContext,\n documentUid: string,\n): Promise<DocumentSummary & { document: string }> {\n const { fetch } = context\n\n const response = await fetch(`/api/v1.0/documents/${documentUid}/raw`)\n const data = await response.json()\n\n if (rawLogs) {\n console.log('Raw document:', data)\n }\n\n return { ...data, longId: data.longID ?? data.longId } as DocumentSummary & {\n document: string\n }\n}\n\nexport async function getDocumentDetails(\n context: DocumentContext,\n documentUid: string,\n): Promise<\n DocumentSummary & {\n validationResults: {\n status: DocumentValidationResult\n validationSteps: DocumentValidationStepResult[]\n }\n }\n> {\n const { fetch } = context\n\n const response = await fetch(`/api/v1.0/documents/${documentUid}/details`)\n const data = await response.json()\n const resp = {\n ...data,\n longId: data.longID ?? data.longId,\n } as DocumentSummary & {\n validationResults: {\n status: DocumentValidationResult\n validationSteps: DocumentValidationStepResult[]\n }\n }\n\n if (rawLogs) {\n console.log('Raw document details:', resp)\n }\n\n return resp\n}\n\nexport async function searchDocuments(\n context: DocumentContext,\n params: {\n uuid?: string\n submissionDateFrom: string\n submissionDateTo?: string\n pageSize?: number\n pageNo?: number\n issueDateFrom?: string\n issueDateTo?: string\n invoiceDirection?: 'Sent' | 'Received'\n status?: DocumentStatus\n documentType?: EInvoiceTypeCode\n searchQuery?: string\n },\n): Promise<DocumentSummary[]> {\n const { fetch } = context\n const {\n uuid,\n submissionDateFrom,\n submissionDateTo,\n pageSize,\n pageNo,\n issueDateFrom,\n issueDateTo,\n invoiceDirection,\n status,\n documentType,\n searchQuery,\n } = params\n\n const queryParams = new URLSearchParams()\n\n if (uuid) queryParams.set('uuid', uuid)\n if (submissionDateFrom)\n queryParams.set('submissionDateFrom', submissionDateFrom)\n if (submissionDateTo) queryParams.set('submissionDateTo', submissionDateTo)\n if (pageSize) queryParams.set('pageSize', pageSize.toString())\n if (pageNo) queryParams.set('pageNo', pageNo.toString())\n if (issueDateFrom) queryParams.set('issueDateFrom', issueDateFrom)\n if (issueDateTo) queryParams.set('issueDateTo', issueDateTo)\n if (invoiceDirection) queryParams.set('invoiceDirection', invoiceDirection)\n if (status) queryParams.set('status', status)\n if (documentType) queryParams.set('documentType', documentType)\n if (searchQuery) queryParams.set('searchQuery', searchQuery)\n\n const response = await fetch(\n `/api/v1.0/documents/search?${queryParams.toString()}`,\n )\n\n const data = await response.json()\n\n if (rawLogs) {\n console.log('Raw search documents:', data)\n }\n\n // The MyInvois API wraps results in `{ result: [...], metadata: {...} }`.\n // Fall back to a bare array for forward/backward compatibility.\n const documents: (DocumentSummary & { longID?: string })[] = Array.isArray(\n data,\n )\n ? data\n : (data?.result ?? [])\n\n return documents.map(doc => ({\n ...doc,\n longId: doc.longID ?? doc.longId,\n })) as DocumentSummary[]\n}\n"],"mappings":";;AAaA,MAAM,UAAU,QAAQ,IAAI,mBAAmB;AAE/C,eAAsB,YACpBA,SACAC,aACiD;CACjD,MAAM,EAAE,OAAO,GAAG;CAElB,MAAM,WAAW,MAAM,OAAO,sBAAsB,YAAY,MAAM;CACtE,MAAM,OAAO,MAAM,SAAS,MAAM;AAElC,KAAI,QACF,SAAQ,IAAI,iBAAiB,KAAK;AAGpC,QAAO;EAAE,GAAG;EAAM,QAAQ,KAAK,UAAU,KAAK;CAAQ;AAGvD;AAED,eAAsB,mBACpBD,SACAC,aAQA;CACA,MAAM,EAAE,OAAO,GAAG;CAElB,MAAM,WAAW,MAAM,OAAO,sBAAsB,YAAY,UAAU;CAC1E,MAAM,OAAO,MAAM,SAAS,MAAM;CAClC,MAAM,OAAO;EACX,GAAG;EACH,QAAQ,KAAK,UAAU,KAAK;CAC7B;AAOD,KAAI,QACF,SAAQ,IAAI,yBAAyB,KAAK;AAG5C,QAAO;AACR;AAED,eAAsB,gBACpBD,SACAE,QAa4B;CAC5B,MAAM,EAAE,OAAO,GAAG;CAClB,MAAM,EACJ,MACA,oBACA,kBACA,UACA,QACA,eACA,aACA,kBACA,QACA,cACA,aACD,GAAG;CAEJ,MAAM,cAAc,IAAI;AAExB,KAAI,KAAM,aAAY,IAAI,QAAQ,KAAK;AACvC,KAAI,mBACF,aAAY,IAAI,sBAAsB,mBAAmB;AAC3D,KAAI,iBAAkB,aAAY,IAAI,oBAAoB,iBAAiB;AAC3E,KAAI,SAAU,aAAY,IAAI,YAAY,SAAS,UAAU,CAAC;AAC9D,KAAI,OAAQ,aAAY,IAAI,UAAU,OAAO,UAAU,CAAC;AACxD,KAAI,cAAe,aAAY,IAAI,iBAAiB,cAAc;AAClE,KAAI,YAAa,aAAY,IAAI,eAAe,YAAY;AAC5D,KAAI,iBAAkB,aAAY,IAAI,oBAAoB,iBAAiB;AAC3E,KAAI,OAAQ,aAAY,IAAI,UAAU,OAAO;AAC7C,KAAI,aAAc,aAAY,IAAI,gBAAgB,aAAa;AAC/D,KAAI,YAAa,aAAY,IAAI,eAAe,YAAY;CAE5D,MAAM,WAAW,MAAM,OACpB,6BAA6B,YAAY,UAAU,CAAC,EACtD;CAED,MAAM,OAAO,MAAM,SAAS,MAAM;AAElC,KAAI,QACF,SAAQ,IAAI,yBAAyB,KAAK;CAK5C,MAAMC,YAAuD,MAAM,QACjE,KACD,GACG,OACC,MAAM,UAAU,CAAE;AAEvB,QAAO,UAAU,IAAI,UAAQ;EAC3B,GAAG;EACH,QAAQ,IAAI,UAAU,IAAI;CAC3B,GAAE;AACJ"}
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
const require_document = require('./document-
|
|
1
|
+
const require_document = require('./document-B11B5lqd.cjs');
|
|
2
2
|
|
|
3
3
|
//#region src/api/documentSubmission.ts
|
|
4
4
|
async function submitDocument(context, documents) {
|
|
@@ -67,7 +67,7 @@ async function submitDocument(context, documents) {
|
|
|
67
67
|
status: response.status
|
|
68
68
|
};
|
|
69
69
|
}
|
|
70
|
-
async function getSubmissionStatus(context, submissionUid, pollInterval = 1e3, maxRetries = 10, pageNo =
|
|
70
|
+
async function getSubmissionStatus(context, submissionUid, pollInterval = 1e3, maxRetries = 10, pageNo = 1, pageSize = 10) {
|
|
71
71
|
const { fetch, debug } = context;
|
|
72
72
|
try {
|
|
73
73
|
const response = await fetch(`/api/v1.0/documentsubmissions/${submissionUid}?pageNo=${pageNo}&pageSize=${pageSize}`);
|
|
@@ -86,7 +86,7 @@ async function getSubmissionStatus(context, submissionUid, pollInterval = 1e3, m
|
|
|
86
86
|
};
|
|
87
87
|
if (maxRetries > 0) {
|
|
88
88
|
await new Promise((resolve) => setTimeout(resolve, pollInterval));
|
|
89
|
-
return await getSubmissionStatus(context, submissionUid, pollInterval, maxRetries - 1);
|
|
89
|
+
return await getSubmissionStatus(context, submissionUid, pollInterval, maxRetries - 1, pageNo, pageSize);
|
|
90
90
|
}
|
|
91
91
|
return {
|
|
92
92
|
status: data.overallStatus || "TimedOut",
|
|
@@ -102,7 +102,7 @@ async function getSubmissionStatus(context, submissionUid, pollInterval = 1e3, m
|
|
|
102
102
|
if (maxRetries > 0) {
|
|
103
103
|
if (debug) console.log("Request error, retrying...", error);
|
|
104
104
|
await new Promise((resolve) => setTimeout(resolve, pollInterval));
|
|
105
|
-
return await getSubmissionStatus(context, submissionUid, pollInterval, maxRetries - 1);
|
|
105
|
+
return await getSubmissionStatus(context, submissionUid, pollInterval, maxRetries - 1, pageNo, pageSize);
|
|
106
106
|
}
|
|
107
107
|
return {
|
|
108
108
|
status: "TimedOut",
|
|
@@ -121,6 +121,7 @@ async function performDocumentAction(context, documentUid, status, reason) {
|
|
|
121
121
|
const { fetch } = context;
|
|
122
122
|
const response = await fetch(`/api/v1.0/documents/state/${documentUid}/state`, {
|
|
123
123
|
method: "PUT",
|
|
124
|
+
headers: { "Content-Type": "application/json" },
|
|
124
125
|
body: JSON.stringify({
|
|
125
126
|
status,
|
|
126
127
|
reason
|
|
@@ -149,4 +150,4 @@ Object.defineProperty(exports, 'submitDocument', {
|
|
|
149
150
|
return submitDocument;
|
|
150
151
|
}
|
|
151
152
|
});
|
|
152
|
-
//# sourceMappingURL=documentSubmission-
|
|
153
|
+
//# sourceMappingURL=documentSubmission-BaPECvYU.cjs.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"documentSubmission-B6hkivRK.cjs","names":["context: SubmissionContext","documents: AllDocumentsV1_1[]","context: Pick<SubmissionContext, 'fetch' | 'debug'>","submissionUid: string","pollInterval: number","maxRetries: number","pageNo: number","pageSize: number","context: Pick<SubmissionContext, 'fetch'>","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 pageNo: number = 0,\n pageSize: 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}?pageNo=${pageNo}&pageSize=${pageSize}`,\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 context: Pick<SubmissionContext, 'fetch'>,\n documentUid: string,\n status: 'rejected' | 'cancelled',\n reason: string,\n): Promise<{\n uuid: string\n status: string\n error: StandardError\n}> {\n if (reason.length > 300) {\n throw new Error('Reason must not exceed 300 characters')\n }\n\n const { fetch } = context\n const response = await fetch(\n `/api/v1.0/documents/state/${documentUid}/state`,\n {\n method: 'PUT',\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
|
+
{"version":3,"file":"documentSubmission-BaPECvYU.cjs","names":["context: SubmissionContext","documents: AllDocumentsV1_1[]","context: Pick<SubmissionContext, 'fetch' | 'debug'>","submissionUid: string","pollInterval: number","maxRetries: number","pageNo: number","pageSize: number","context: Pick<SubmissionContext, 'fetch'>","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 pageNo: number = 1,\n pageSize: 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}?pageNo=${pageNo}&pageSize=${pageSize}`,\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 pageNo,\n pageSize,\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 pageNo,\n pageSize,\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 context: Pick<SubmissionContext, 'fetch'>,\n documentUid: string,\n status: 'rejected' | 'cancelled',\n reason: string,\n): Promise<{\n uuid: string\n status: string\n error: StandardError\n}> {\n if (reason.length > 300) {\n throw new Error('Reason must not exceed 300 characters')\n }\n\n const { fetch } = context\n const response = await fetch(\n `/api/v1.0/documents/state/${documentUid}/state`,\n {\n method: 'PUT',\n headers: {\n 'Content-Type': 'application/json',\n },\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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"}
|
package/dist/{documentTypeManagement-D_-LiQVg.cjs.map → documentTypeManagement-DXRLfTsW.cjs.map}
RENAMED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"documentTypeManagement-
|
|
1
|
+
{"version":3,"file":"documentTypeManagement-DXRLfTsW.cjs","names":["context: DocumentTypeContext","id: number","versionId: number"],"sources":["../src/api/documentTypeManagement.ts"],"sourcesContent":["import type {\n DocumentTypeResponse,\n DocumentTypesResponse,\n DocumentTypeVersionResponse,\n Fetch,\n} from '../types'\n\ninterface DocumentTypeContext {\n fetch: Fetch\n}\n\nexport async function getDocumentTypes(\n context: DocumentTypeContext,\n): Promise<DocumentTypesResponse> {\n const { fetch } = context\n\n const response = await fetch('/api/v1.0/documenttypes')\n const data = (await response.json()) as DocumentTypesResponse\n\n return data\n}\n\nexport async function getDocumentType(\n context: DocumentTypeContext,\n id: number,\n): Promise<DocumentTypeResponse> {\n const { fetch } = context\n\n const response = await fetch(`/api/v1.0/documenttypes/${id}`)\n const data = (await response.json()) as DocumentTypeResponse\n\n return data\n}\n\nexport async function getDocumentTypeVersion(\n context: DocumentTypeContext,\n id: number,\n versionId: number,\n): Promise<DocumentTypeVersionResponse> {\n const { fetch } = context\n\n const response = await fetch(\n `/api/v1.0/documenttypes/${id}/versions/${versionId}`,\n )\n\n const data = (await response.json()) as DocumentTypeVersionResponse\n\n return data\n}\n"],"mappings":";;AAWA,eAAsB,iBACpBA,SACgC;CAChC,MAAM,EAAE,OAAO,GAAG;CAElB,MAAM,WAAW,MAAM,MAAM,0BAA0B;CACvD,MAAM,OAAQ,MAAM,SAAS,MAAM;AAEnC,QAAO;AACR;AAED,eAAsB,gBACpBA,SACAC,IAC+B;CAC/B,MAAM,EAAE,OAAO,GAAG;CAElB,MAAM,WAAW,MAAM,OAAO,0BAA0B,GAAG,EAAE;CAC7D,MAAM,OAAQ,MAAM,SAAS,MAAM;AAEnC,QAAO;AACR;AAED,eAAsB,uBACpBD,SACAC,IACAC,WACsC;CACtC,MAAM,EAAE,OAAO,GAAG;CAElB,MAAM,WAAW,MAAM,OACpB,0BAA0B,GAAG,YAAY,UAAU,EACrD;CAED,MAAM,OAAQ,MAAM,SAAS,MAAM;AAEnC,QAAO;AACR"}
|