@devlas/dte-sii 2.5.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.
Files changed (60) hide show
  1. package/BoletaService.js +109 -0
  2. package/CAF.js +173 -0
  3. package/CafSolicitor.js +380 -0
  4. package/Certificado.js +123 -0
  5. package/ConsumoFolio.js +376 -0
  6. package/DTE.js +399 -0
  7. package/EnviadorSII.js +1304 -0
  8. package/Envio.js +196 -0
  9. package/FolioRegistry.js +553 -0
  10. package/FolioService.js +703 -0
  11. package/LICENSE +27 -0
  12. package/LibroBase.js +134 -0
  13. package/LibroCompraVenta.js +205 -0
  14. package/LibroGuia.js +225 -0
  15. package/README.md +239 -0
  16. package/Signer.js +94 -0
  17. package/SiiCertificacion.js +1189 -0
  18. package/SiiPortalAuth.js +460 -0
  19. package/SiiSession.js +499 -0
  20. package/cert/BoletaCert.js +731 -0
  21. package/cert/CertFolioHelper.js +185 -0
  22. package/cert/CertRunner.js +2658 -0
  23. package/cert/ConfigLoader.js +133 -0
  24. package/cert/IntercambioCert.js +429 -0
  25. package/cert/LibroCompras.js +359 -0
  26. package/cert/LibroGuias.js +171 -0
  27. package/cert/LibroVentas.js +153 -0
  28. package/cert/MuestrasImpresas.js +676 -0
  29. package/cert/SetBase.js +321 -0
  30. package/cert/SetBasico.js +413 -0
  31. package/cert/SetCompra.js +472 -0
  32. package/cert/SetExenta.js +490 -0
  33. package/cert/SetGuia.js +283 -0
  34. package/cert/SetParser.js +1184 -0
  35. package/cert/SetsProvider.js +499 -0
  36. package/cert/Simulacion.js +521 -0
  37. package/cert/comunaOficina.js +460 -0
  38. package/cert/index.js +124 -0
  39. package/cert/types.js +330 -0
  40. package/dte-sii.d.ts +458 -0
  41. package/index.js +428 -0
  42. package/package.json +48 -0
  43. package/utils/c14n.js +275 -0
  44. package/utils/calculo.js +396 -0
  45. package/utils/config.js +276 -0
  46. package/utils/constants.js +302 -0
  47. package/utils/emisor.js +174 -0
  48. package/utils/endpoints.js +225 -0
  49. package/utils/error.js +235 -0
  50. package/utils/index.js +339 -0
  51. package/utils/logger.js +239 -0
  52. package/utils/pfx.js +203 -0
  53. package/utils/receptor.js +218 -0
  54. package/utils/referencia.js +169 -0
  55. package/utils/resolucion.js +119 -0
  56. package/utils/rut.js +169 -0
  57. package/utils/sanitize.js +124 -0
  58. package/utils/tokenCache.js +214 -0
  59. package/utils/xml.js +358 -0
  60. package/utils.js +4 -0
package/Envio.js ADDED
@@ -0,0 +1,196 @@
1
+ // Copyright (c) 2026 Devlas SpA — https://devlas.cl
2
+ // Licencia MIT. Ver archivo LICENSE para mas detalles.
3
+ /**
4
+ * Clases de Envío (EnvioBOLETA, EnvioDTE)
5
+ *
6
+ * Sobres para agrupar múltiples DTEs y enviar al SII
7
+ */
8
+
9
+ const Signer = require('./Signer');
10
+
11
+ // ============================================
12
+ // CLASE BASE ENVIO
13
+ // ============================================
14
+
15
+ class EnvioBase {
16
+ constructor(config) {
17
+ this.certificado = config.certificado;
18
+ this.config = config;
19
+ this.dtes = [];
20
+ this.caratula = null;
21
+ this.setId = null;
22
+ this.xml = null;
23
+ this.xmlSinFirma = null;
24
+ }
25
+
26
+ /**
27
+ * Agregar un DTE al sobre
28
+ */
29
+ agregar(dte) {
30
+ // Auto-generar si tiene certificado/caf guardados
31
+ if (dte._certificado && dte._caf && !dte.xml) {
32
+ dte.generarXML();
33
+ dte.timbrar(dte._caf);
34
+ dte.firmar(dte._certificado);
35
+ }
36
+ this.dtes.push(dte);
37
+ return this;
38
+ }
39
+
40
+ /**
41
+ * Calcular subtotales por tipo de DTE
42
+ */
43
+ _getSubTotDTE() {
44
+ const tipos = {};
45
+ for (const dte of this.dtes) {
46
+ const tipo = dte.datos.Encabezado.IdDoc.TipoDTE;
47
+ if (!tipos[tipo]) tipos[tipo] = 0;
48
+ tipos[tipo]++;
49
+ }
50
+ return Object.entries(tipos).map(([tipo, cantidad]) => ({
51
+ TpoDTE: parseInt(tipo, 10),
52
+ NroDTE: cantidad,
53
+ }));
54
+ }
55
+
56
+ /**
57
+ * Generar timestamp para firma
58
+ */
59
+ _generateTimestamp() {
60
+ return new Date().toISOString().replace(/\.\d{3}Z$/, '');
61
+ }
62
+
63
+ /**
64
+ * Generar SetId con formato específico
65
+ */
66
+ _generateSetId(prefix) {
67
+ const now = new Date();
68
+ const dd = String(now.getDate()).padStart(2, '0');
69
+ const mm = String(now.getMonth() + 1).padStart(2, '0');
70
+ const yyyy = now.getFullYear();
71
+ const hh = String(now.getHours()).padStart(2, '0');
72
+ const min = String(now.getMinutes()).padStart(2, '0');
73
+ return `${prefix}_${dd}_${mm}_${yyyy}_${hh}_${min}`;
74
+ }
75
+
76
+ /**
77
+ * Extraer XML de DTEs sin declaration
78
+ */
79
+ _extractDTEsXml() {
80
+ return this.dtes.map(dte => {
81
+ return dte.getXML()
82
+ .replace('<?xml version="1.0" encoding="ISO-8859-1"?>', '')
83
+ .replace('<?xml version="1.0"?>', '')
84
+ .trim();
85
+ }).join('\n');
86
+ }
87
+
88
+ getXML() { return this.xml; }
89
+ getXMLSinFirma() { return this.xmlSinFirma; }
90
+ }
91
+
92
+ // ============================================
93
+ // CLASE ENVIOBOLETA
94
+ // ============================================
95
+
96
+ // DD-MM-YYYY → YYYY-MM-DD (xs:date requerido por EnvioBOLETA_v11.xsd y ConsumoFolios.xsd)
97
+ function _normDateXsd(d) {
98
+ if (!d) return d;
99
+ const m = String(d).match(/^(\d{2})-(\d{2})-(\d{4})$/);
100
+ return m ? `${m[3]}-${m[2]}-${m[1]}` : d;
101
+ }
102
+
103
+ class EnvioBOLETA extends EnvioBase {
104
+ setCaratula(caratula) {
105
+ this.setId = this._generateSetId('ENVIOBOLETA');
106
+ const timestamp = caratula.TmstFirmaEnv || this._generateTimestamp();
107
+
108
+ this.caratula = {
109
+ RutEmisor: caratula.RutEmisor,
110
+ RutEnvia: caratula.RutEnvia,
111
+ RutReceptor: '60803000-K', // Siempre SII para boletas
112
+ FchResol: _normDateXsd(caratula.FchResol),
113
+ NroResol: caratula.NroResol,
114
+ TmstFirmaEnv: timestamp,
115
+ SubTotDTE: this._getSubTotDTE(),
116
+ };
117
+
118
+ return this;
119
+ }
120
+
121
+ generar() {
122
+ if (!this.dtes.length) throw new Error('No hay DTEs para enviar');
123
+ if (!this.caratula) throw new Error('Falta carátula');
124
+
125
+ const subTotDTEs = this.caratula.SubTotDTE
126
+ .map(s => `<SubTotDTE><TpoDTE>${s.TpoDTE}</TpoDTE><NroDTE>${s.NroDTE}</NroDTE></SubTotDTE>`)
127
+ .join('');
128
+
129
+ const xmlBase = `<?xml version="1.0" encoding="ISO-8859-1"?>
130
+ <EnvioBOLETA xmlns="http://www.sii.cl/SiiDte" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.sii.cl/SiiDte EnvioBOLETA_v11.xsd" version="1.0"><SetDTE ID="${this.setId}"><Caratula version="1.0"><RutEmisor>${this.caratula.RutEmisor}</RutEmisor><RutEnvia>${this.caratula.RutEnvia}</RutEnvia><RutReceptor>${this.caratula.RutReceptor}</RutReceptor><FchResol>${this.caratula.FchResol}</FchResol><NroResol>${this.caratula.NroResol}</NroResol><TmstFirmaEnv>${this.caratula.TmstFirmaEnv}</TmstFirmaEnv>${subTotDTEs}</Caratula><!-- DTES_PLACEHOLDER --></SetDTE></EnvioBOLETA>`;
131
+
132
+ const dtesXml = this._extractDTEsXml();
133
+ this.xmlSinFirma = xmlBase.replace('<!-- DTES_PLACEHOLDER -->', dtesXml);
134
+
135
+ const signer = new Signer(this.certificado);
136
+ this.xml = signer.firmarSetDTE(this.xmlSinFirma, this.setId, 'EnvioBOLETA');
137
+
138
+ return this.xml;
139
+ }
140
+ }
141
+
142
+ // ============================================
143
+ // CLASE ENVIODTE
144
+ // ============================================
145
+
146
+ class EnvioDTE extends EnvioBase {
147
+ setCaratula(caratula) {
148
+ this.setId = caratula.SetDTEId || caratula.SetId || this._generateSetId('DTE_SetDoc');
149
+ const timestamp = this._generateTimestamp();
150
+
151
+ // Ordenar subtotales (33, 61, 56 primero)
152
+ const subTotDTE = this._getSubTotDTE().sort((a, b) => {
153
+ const orden = [33, 61, 56];
154
+ const idxA = orden.indexOf(a.TpoDTE);
155
+ const idxB = orden.indexOf(b.TpoDTE);
156
+ if (idxA === -1 && idxB === -1) return a.TpoDTE - b.TpoDTE;
157
+ if (idxA === -1) return 1;
158
+ if (idxB === -1) return -1;
159
+ return idxA - idxB;
160
+ });
161
+
162
+ this.caratula = {
163
+ RutEmisor: caratula.RutEmisor,
164
+ RutEnvia: caratula.RutEnvia,
165
+ RutReceptor: caratula.RutReceptor || '',
166
+ FchResol: caratula.FchResol,
167
+ NroResol: caratula.NroResol,
168
+ TmstFirmaEnv: timestamp,
169
+ SubTotDTE: subTotDTE,
170
+ };
171
+
172
+ return this;
173
+ }
174
+
175
+ generar() {
176
+ if (!this.dtes.length) throw new Error('No hay DTEs para enviar');
177
+ if (!this.caratula) throw new Error('Falta carátula');
178
+
179
+ const subTotDTEs = this.caratula.SubTotDTE
180
+ .map(s => `<SubTotDTE><TpoDTE>${s.TpoDTE}</TpoDTE><NroDTE>${s.NroDTE}</NroDTE></SubTotDTE>`)
181
+ .join('');
182
+
183
+ const xmlBase = `<?xml version="1.0" encoding="ISO-8859-1"?>
184
+ <EnvioDTE xmlns="http://www.sii.cl/SiiDte" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.sii.cl/SiiDte EnvioDTE_v10.xsd" version="1.0"><SetDTE ID="${this.setId}"><Caratula version="1.0"><RutEmisor>${this.caratula.RutEmisor}</RutEmisor><RutEnvia>${this.caratula.RutEnvia}</RutEnvia><RutReceptor>${this.caratula.RutReceptor}</RutReceptor><FchResol>${this.caratula.FchResol}</FchResol><NroResol>${this.caratula.NroResol}</NroResol><TmstFirmaEnv>${this.caratula.TmstFirmaEnv}</TmstFirmaEnv>${subTotDTEs}</Caratula><!-- DTES_PLACEHOLDER --></SetDTE></EnvioDTE>`;
185
+
186
+ const dtesXml = this._extractDTEsXml();
187
+ this.xmlSinFirma = xmlBase.replace('<!-- DTES_PLACEHOLDER -->', dtesXml);
188
+
189
+ const signer = new Signer(this.certificado);
190
+ this.xml = signer.firmarSetDTE(this.xmlSinFirma, this.setId, 'EnvioDTE');
191
+
192
+ return this.xml;
193
+ }
194
+ }
195
+
196
+ module.exports = { EnvioBase, EnvioBOLETA, EnvioDTE };