docxmlater 10.4.0 → 11.0.4

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 (699) hide show
  1. package/README.md +3 -3
  2. package/dist/constants/legacyCompatFlags.d.ts +1 -1
  3. package/dist/constants/legacyCompatFlags.d.ts.map +1 -1
  4. package/dist/constants/legacyCompatFlags.js.map +1 -1
  5. package/dist/core/Document.d.ts +75 -67
  6. package/dist/core/Document.d.ts.map +1 -1
  7. package/dist/core/Document.js +618 -414
  8. package/dist/core/Document.js.map +1 -1
  9. package/dist/core/DocumentContent.d.ts +11 -10
  10. package/dist/core/DocumentContent.d.ts.map +1 -1
  11. package/dist/core/DocumentContent.js +19 -19
  12. package/dist/core/DocumentContent.js.map +1 -1
  13. package/dist/core/DocumentEvents.d.ts +39 -0
  14. package/dist/core/DocumentEvents.d.ts.map +1 -0
  15. package/dist/core/DocumentEvents.js +51 -0
  16. package/dist/core/DocumentEvents.js.map +1 -0
  17. package/dist/core/DocumentGenerator.d.ts +11 -11
  18. package/dist/core/DocumentGenerator.d.ts.map +1 -1
  19. package/dist/core/DocumentGenerator.js +72 -52
  20. package/dist/core/DocumentGenerator.js.map +1 -1
  21. package/dist/core/DocumentParser.d.ts +15 -15
  22. package/dist/core/DocumentParser.d.ts.map +1 -1
  23. package/dist/core/DocumentParser.js +2100 -1076
  24. package/dist/core/DocumentParser.js.map +1 -1
  25. package/dist/core/DocumentValidator.d.ts +3 -3
  26. package/dist/core/DocumentValidator.d.ts.map +1 -1
  27. package/dist/core/DocumentValidator.js +31 -31
  28. package/dist/core/DocumentValidator.js.map +1 -1
  29. package/dist/core/ElementRegistry.d.ts +22 -0
  30. package/dist/core/ElementRegistry.d.ts.map +1 -0
  31. package/dist/core/ElementRegistry.js +27 -0
  32. package/dist/core/ElementRegistry.js.map +1 -0
  33. package/dist/core/Relationship.js +4 -4
  34. package/dist/core/Relationship.js.map +1 -1
  35. package/dist/core/RelationshipManager.d.ts +1 -1
  36. package/dist/core/RelationshipManager.d.ts.map +1 -1
  37. package/dist/core/RelationshipManager.js +32 -32
  38. package/dist/core/RelationshipManager.js.map +1 -1
  39. package/dist/elements/AlternateContent.d.ts +1 -1
  40. package/dist/elements/AlternateContent.d.ts.map +1 -1
  41. package/dist/elements/AlternateContent.js.map +1 -1
  42. package/dist/elements/Bookmark.d.ts +6 -1
  43. package/dist/elements/Bookmark.d.ts.map +1 -1
  44. package/dist/elements/Bookmark.js +19 -3
  45. package/dist/elements/Bookmark.js.map +1 -1
  46. package/dist/elements/BookmarkManager.d.ts +1 -1
  47. package/dist/elements/BookmarkManager.d.ts.map +1 -1
  48. package/dist/elements/BookmarkManager.js +7 -7
  49. package/dist/elements/BookmarkManager.js.map +1 -1
  50. package/dist/elements/Comment.d.ts +2 -2
  51. package/dist/elements/Comment.d.ts.map +1 -1
  52. package/dist/elements/Comment.js +4 -4
  53. package/dist/elements/Comment.js.map +1 -1
  54. package/dist/elements/CommentManager.d.ts +2 -2
  55. package/dist/elements/CommentManager.d.ts.map +1 -1
  56. package/dist/elements/CommentManager.js +9 -9
  57. package/dist/elements/CommentManager.js.map +1 -1
  58. package/dist/elements/CommonTypes.d.ts +9 -4
  59. package/dist/elements/CommonTypes.d.ts.map +1 -1
  60. package/dist/elements/CommonTypes.js +1 -0
  61. package/dist/elements/CommonTypes.js.map +1 -1
  62. package/dist/elements/CustomXml.d.ts +1 -1
  63. package/dist/elements/CustomXml.d.ts.map +1 -1
  64. package/dist/elements/CustomXml.js.map +1 -1
  65. package/dist/elements/Endnote.d.ts +2 -2
  66. package/dist/elements/Endnote.d.ts.map +1 -1
  67. package/dist/elements/Endnote.js +9 -9
  68. package/dist/elements/Endnote.js.map +1 -1
  69. package/dist/elements/EndnoteManager.d.ts +1 -1
  70. package/dist/elements/EndnoteManager.d.ts.map +1 -1
  71. package/dist/elements/EndnoteManager.js +11 -11
  72. package/dist/elements/EndnoteManager.js.map +1 -1
  73. package/dist/elements/Field.d.ts +9 -5
  74. package/dist/elements/Field.d.ts.map +1 -1
  75. package/dist/elements/Field.js +21 -9
  76. package/dist/elements/Field.js.map +1 -1
  77. package/dist/elements/FieldHelpers.d.ts +1 -1
  78. package/dist/elements/FieldHelpers.d.ts.map +1 -1
  79. package/dist/elements/FieldHelpers.js +10 -10
  80. package/dist/elements/FieldHelpers.js.map +1 -1
  81. package/dist/elements/Footer.d.ts +3 -3
  82. package/dist/elements/Footer.d.ts.map +1 -1
  83. package/dist/elements/Footer.js +5 -5
  84. package/dist/elements/Footer.js.map +1 -1
  85. package/dist/elements/Footnote.d.ts +2 -2
  86. package/dist/elements/Footnote.d.ts.map +1 -1
  87. package/dist/elements/Footnote.js +9 -9
  88. package/dist/elements/Footnote.js.map +1 -1
  89. package/dist/elements/FootnoteManager.d.ts +1 -1
  90. package/dist/elements/FootnoteManager.d.ts.map +1 -1
  91. package/dist/elements/FootnoteManager.js +11 -11
  92. package/dist/elements/FootnoteManager.js.map +1 -1
  93. package/dist/elements/Header.d.ts +3 -3
  94. package/dist/elements/Header.d.ts.map +1 -1
  95. package/dist/elements/Header.js +5 -5
  96. package/dist/elements/Header.js.map +1 -1
  97. package/dist/elements/HeaderFooterManager.d.ts +2 -2
  98. package/dist/elements/HeaderFooterManager.d.ts.map +1 -1
  99. package/dist/elements/HeaderFooterManager.js.map +1 -1
  100. package/dist/elements/Hyperlink.d.ts +5 -5
  101. package/dist/elements/Hyperlink.d.ts.map +1 -1
  102. package/dist/elements/Hyperlink.js +29 -29
  103. package/dist/elements/Hyperlink.js.map +1 -1
  104. package/dist/elements/Image.d.ts +1 -1
  105. package/dist/elements/Image.d.ts.map +1 -1
  106. package/dist/elements/Image.js +67 -67
  107. package/dist/elements/Image.js.map +1 -1
  108. package/dist/elements/ImageManager.d.ts +1 -1
  109. package/dist/elements/ImageManager.d.ts.map +1 -1
  110. package/dist/elements/ImageManager.js +4 -4
  111. package/dist/elements/ImageManager.js.map +1 -1
  112. package/dist/elements/ImageRun.d.ts +3 -3
  113. package/dist/elements/ImageRun.d.ts.map +1 -1
  114. package/dist/elements/ImageRun.js +2 -2
  115. package/dist/elements/ImageRun.js.map +1 -1
  116. package/dist/elements/MathElement.d.ts +1 -1
  117. package/dist/elements/MathElement.d.ts.map +1 -1
  118. package/dist/elements/MathElement.js.map +1 -1
  119. package/dist/elements/Paragraph.d.ts +34 -19
  120. package/dist/elements/Paragraph.d.ts.map +1 -1
  121. package/dist/elements/Paragraph.js +286 -231
  122. package/dist/elements/Paragraph.js.map +1 -1
  123. package/dist/elements/PreservedElement.d.ts +1 -1
  124. package/dist/elements/PreservedElement.d.ts.map +1 -1
  125. package/dist/elements/PreservedElement.js.map +1 -1
  126. package/dist/elements/PropertyChangeTypes.d.ts +2 -2
  127. package/dist/elements/PropertyChangeTypes.d.ts.map +1 -1
  128. package/dist/elements/PropertyChangeTypes.js.map +1 -1
  129. package/dist/elements/RangeMarker.d.ts +14 -1
  130. package/dist/elements/RangeMarker.d.ts.map +1 -1
  131. package/dist/elements/RangeMarker.js +46 -8
  132. package/dist/elements/RangeMarker.js.map +1 -1
  133. package/dist/elements/RegisteredBodyElement.d.ts +15 -0
  134. package/dist/elements/RegisteredBodyElement.d.ts.map +1 -0
  135. package/dist/elements/RegisteredBodyElement.js +44 -0
  136. package/dist/elements/RegisteredBodyElement.js.map +1 -0
  137. package/dist/elements/Revision.d.ts +8 -8
  138. package/dist/elements/Revision.d.ts.map +1 -1
  139. package/dist/elements/Revision.js +12 -12
  140. package/dist/elements/Revision.js.map +1 -1
  141. package/dist/elements/RevisionContent.d.ts +3 -3
  142. package/dist/elements/RevisionContent.d.ts.map +1 -1
  143. package/dist/elements/RevisionContent.js.map +1 -1
  144. package/dist/elements/RevisionManager.d.ts +2 -2
  145. package/dist/elements/RevisionManager.d.ts.map +1 -1
  146. package/dist/elements/RevisionManager.js +2 -2
  147. package/dist/elements/RevisionManager.js.map +1 -1
  148. package/dist/elements/Run.d.ts +16 -10
  149. package/dist/elements/Run.d.ts.map +1 -1
  150. package/dist/elements/Run.js +199 -173
  151. package/dist/elements/Run.js.map +1 -1
  152. package/dist/elements/Section.d.ts +4 -2
  153. package/dist/elements/Section.d.ts.map +1 -1
  154. package/dist/elements/Section.js +152 -145
  155. package/dist/elements/Section.js.map +1 -1
  156. package/dist/elements/Shape.d.ts +3 -3
  157. package/dist/elements/Shape.d.ts.map +1 -1
  158. package/dist/elements/Shape.js +12 -12
  159. package/dist/elements/Shape.js.map +1 -1
  160. package/dist/elements/StructuredDocumentTag.d.ts +3 -3
  161. package/dist/elements/StructuredDocumentTag.d.ts.map +1 -1
  162. package/dist/elements/StructuredDocumentTag.js +39 -39
  163. package/dist/elements/StructuredDocumentTag.js.map +1 -1
  164. package/dist/elements/Table.d.ts +16 -10
  165. package/dist/elements/Table.d.ts.map +1 -1
  166. package/dist/elements/Table.js +118 -89
  167. package/dist/elements/Table.js.map +1 -1
  168. package/dist/elements/TableCell.d.ts +11 -11
  169. package/dist/elements/TableCell.d.ts.map +1 -1
  170. package/dist/elements/TableCell.js +108 -78
  171. package/dist/elements/TableCell.js.map +1 -1
  172. package/dist/elements/TableGridChange.d.ts +1 -1
  173. package/dist/elements/TableGridChange.d.ts.map +1 -1
  174. package/dist/elements/TableGridChange.js +3 -3
  175. package/dist/elements/TableGridChange.js.map +1 -1
  176. package/dist/elements/TableOfContents.d.ts +1 -1
  177. package/dist/elements/TableOfContents.d.ts.map +1 -1
  178. package/dist/elements/TableOfContents.js +2 -2
  179. package/dist/elements/TableOfContents.js.map +1 -1
  180. package/dist/elements/TableOfContentsElement.d.ts +2 -2
  181. package/dist/elements/TableOfContentsElement.d.ts.map +1 -1
  182. package/dist/elements/TableOfContentsElement.js +5 -5
  183. package/dist/elements/TableOfContentsElement.js.map +1 -1
  184. package/dist/elements/TableRow.d.ts +18 -7
  185. package/dist/elements/TableRow.d.ts.map +1 -1
  186. package/dist/elements/TableRow.js +127 -74
  187. package/dist/elements/TableRow.js.map +1 -1
  188. package/dist/elements/TextBox.d.ts +4 -4
  189. package/dist/elements/TextBox.d.ts.map +1 -1
  190. package/dist/elements/TextBox.js +6 -6
  191. package/dist/elements/TextBox.js.map +1 -1
  192. package/dist/esm/constants/legacyCompatFlags.js +97 -0
  193. package/dist/esm/constants/legacyCompatFlags.js.map +1 -0
  194. package/dist/esm/constants/limits.js +36 -0
  195. package/dist/esm/constants/limits.js.map +1 -0
  196. package/dist/esm/core/Document.js +8498 -0
  197. package/dist/esm/core/Document.js.map +1 -0
  198. package/dist/esm/core/DocumentContent.js +190 -0
  199. package/dist/esm/core/DocumentContent.js.map +1 -0
  200. package/dist/esm/core/DocumentEvents.js +47 -0
  201. package/dist/esm/core/DocumentEvents.js.map +1 -0
  202. package/dist/esm/core/DocumentGenerator.js +764 -0
  203. package/dist/esm/core/DocumentGenerator.js.map +1 -0
  204. package/dist/esm/core/DocumentIdManager.js +67 -0
  205. package/dist/esm/core/DocumentIdManager.js.map +1 -0
  206. package/dist/esm/core/DocumentParser.js +8760 -0
  207. package/dist/esm/core/DocumentParser.js.map +1 -0
  208. package/dist/esm/core/DocumentValidator.js +222 -0
  209. package/dist/esm/core/DocumentValidator.js.map +1 -0
  210. package/dist/esm/core/ElementRegistry.js +24 -0
  211. package/dist/esm/core/ElementRegistry.js.map +1 -0
  212. package/dist/esm/core/Relationship.js +177 -0
  213. package/dist/esm/core/Relationship.js.map +1 -0
  214. package/dist/esm/core/RelationshipManager.js +202 -0
  215. package/dist/esm/core/RelationshipManager.js.map +1 -0
  216. package/dist/esm/elements/AlternateContent.js +19 -0
  217. package/dist/esm/elements/AlternateContent.js.map +1 -0
  218. package/dist/esm/elements/Bookmark.js +115 -0
  219. package/dist/esm/elements/Bookmark.js.map +1 -0
  220. package/dist/esm/elements/BookmarkManager.js +99 -0
  221. package/dist/esm/elements/BookmarkManager.js.map +1 -0
  222. package/dist/esm/elements/Comment.js +181 -0
  223. package/dist/esm/elements/Comment.js.map +1 -0
  224. package/dist/esm/elements/CommentManager.js +233 -0
  225. package/dist/esm/elements/CommentManager.js.map +1 -0
  226. package/dist/esm/elements/CommonTypes.js +106 -0
  227. package/dist/esm/elements/CommonTypes.js.map +1 -0
  228. package/dist/esm/elements/CustomXml.js +19 -0
  229. package/dist/esm/elements/CustomXml.js.map +1 -0
  230. package/dist/esm/elements/Endnote.js +107 -0
  231. package/dist/esm/elements/Endnote.js.map +1 -0
  232. package/dist/esm/elements/EndnoteManager.js +119 -0
  233. package/dist/esm/elements/EndnoteManager.js.map +1 -0
  234. package/dist/esm/elements/Field.js +856 -0
  235. package/dist/esm/elements/Field.js.map +1 -0
  236. package/dist/esm/elements/FieldHelpers.js +134 -0
  237. package/dist/esm/elements/FieldHelpers.js.map +1 -0
  238. package/dist/esm/elements/FontManager.js +158 -0
  239. package/dist/esm/elements/FontManager.js.map +1 -0
  240. package/dist/esm/elements/Footer.js +141 -0
  241. package/dist/esm/elements/Footer.js.map +1 -0
  242. package/dist/esm/elements/Footnote.js +107 -0
  243. package/dist/esm/elements/Footnote.js.map +1 -0
  244. package/dist/esm/elements/FootnoteManager.js +119 -0
  245. package/dist/esm/elements/FootnoteManager.js.map +1 -0
  246. package/dist/esm/elements/Header.js +141 -0
  247. package/dist/esm/elements/Header.js.map +1 -0
  248. package/dist/esm/elements/HeaderFooterManager.js +87 -0
  249. package/dist/esm/elements/HeaderFooterManager.js.map +1 -0
  250. package/dist/esm/elements/Hyperlink.js +586 -0
  251. package/dist/esm/elements/Hyperlink.js.map +1 -0
  252. package/dist/esm/elements/Image.js +1288 -0
  253. package/dist/esm/elements/Image.js.map +1 -0
  254. package/dist/esm/elements/ImageManager.js +223 -0
  255. package/dist/esm/elements/ImageManager.js.map +1 -0
  256. package/dist/esm/elements/ImageRun.js +29 -0
  257. package/dist/esm/elements/ImageRun.js.map +1 -0
  258. package/dist/esm/elements/MathElement.js +37 -0
  259. package/dist/esm/elements/MathElement.js.map +1 -0
  260. package/dist/esm/elements/Paragraph.js +2308 -0
  261. package/dist/esm/elements/Paragraph.js.map +1 -0
  262. package/dist/esm/elements/PreservedElement.js +29 -0
  263. package/dist/esm/elements/PreservedElement.js.map +1 -0
  264. package/dist/esm/elements/PropertyChangeTypes.js +53 -0
  265. package/dist/esm/elements/PropertyChangeTypes.js.map +1 -0
  266. package/dist/esm/elements/RangeMarker.js +219 -0
  267. package/dist/esm/elements/RangeMarker.js.map +1 -0
  268. package/dist/esm/elements/RegisteredBodyElement.js +40 -0
  269. package/dist/esm/elements/RegisteredBodyElement.js.map +1 -0
  270. package/dist/esm/elements/Revision.js +498 -0
  271. package/dist/esm/elements/Revision.js.map +1 -0
  272. package/dist/esm/elements/RevisionContent.js +18 -0
  273. package/dist/esm/elements/RevisionContent.js.map +1 -0
  274. package/dist/esm/elements/RevisionManager.js +486 -0
  275. package/dist/esm/elements/RevisionManager.js.map +1 -0
  276. package/dist/esm/elements/Run.js +1465 -0
  277. package/dist/esm/elements/Run.js.map +1 -0
  278. package/dist/esm/elements/Section.js +978 -0
  279. package/dist/esm/elements/Section.js.map +1 -0
  280. package/dist/esm/elements/Shape.js +493 -0
  281. package/dist/esm/elements/Shape.js.map +1 -0
  282. package/dist/esm/elements/StructuredDocumentTag.js +471 -0
  283. package/dist/esm/elements/StructuredDocumentTag.js.map +1 -0
  284. package/dist/esm/elements/Table.js +1456 -0
  285. package/dist/esm/elements/Table.js.map +1 -0
  286. package/dist/esm/elements/TableCell.js +835 -0
  287. package/dist/esm/elements/TableCell.js.map +1 -0
  288. package/dist/esm/elements/TableGridChange.js +52 -0
  289. package/dist/esm/elements/TableGridChange.js.map +1 -0
  290. package/dist/esm/elements/TableOfContents.js +389 -0
  291. package/dist/esm/elements/TableOfContents.js.map +1 -0
  292. package/dist/esm/elements/TableOfContentsElement.js +29 -0
  293. package/dist/esm/elements/TableOfContentsElement.js.map +1 -0
  294. package/dist/esm/elements/TableRow.js +555 -0
  295. package/dist/esm/elements/TableRow.js.map +1 -0
  296. package/dist/esm/elements/TextBox.js +459 -0
  297. package/dist/esm/elements/TextBox.js.map +1 -0
  298. package/dist/esm/formatting/AbstractNumbering.js +325 -0
  299. package/dist/esm/formatting/AbstractNumbering.js.map +1 -0
  300. package/dist/esm/formatting/NumberingInstance.js +150 -0
  301. package/dist/esm/formatting/NumberingInstance.js.map +1 -0
  302. package/dist/esm/formatting/NumberingLevel.js +608 -0
  303. package/dist/esm/formatting/NumberingLevel.js.map +1 -0
  304. package/dist/esm/formatting/NumberingManager.js +423 -0
  305. package/dist/esm/formatting/NumberingManager.js.map +1 -0
  306. package/dist/esm/formatting/Style.js +1151 -0
  307. package/dist/esm/formatting/Style.js.map +1 -0
  308. package/dist/esm/formatting/StylesManager.js +557 -0
  309. package/dist/esm/formatting/StylesManager.js.map +1 -0
  310. package/dist/esm/helpers/CleanupHelper.js +350 -0
  311. package/dist/esm/helpers/CleanupHelper.js.map +1 -0
  312. package/dist/esm/images/ImageOptimizer.js +161 -0
  313. package/dist/esm/images/ImageOptimizer.js.map +1 -0
  314. package/dist/esm/index.js +75 -0
  315. package/dist/esm/index.js.map +1 -0
  316. package/dist/esm/internal.js +16 -0
  317. package/dist/esm/internal.js.map +1 -0
  318. package/dist/esm/managers/DrawingManager.js +163 -0
  319. package/dist/esm/managers/DrawingManager.js.map +1 -0
  320. package/dist/esm/package.json +3 -0
  321. package/dist/esm/processors/ChangelogGenerator.js +970 -0
  322. package/dist/esm/processors/ChangelogGenerator.js.map +1 -0
  323. package/dist/esm/processors/CompatibilityUpgrader.js +130 -0
  324. package/dist/esm/processors/CompatibilityUpgrader.js.map +1 -0
  325. package/dist/esm/processors/InMemoryRevisionAcceptor.js +530 -0
  326. package/dist/esm/processors/InMemoryRevisionAcceptor.js.map +1 -0
  327. package/dist/esm/processors/MoveOperationHelper.js +57 -0
  328. package/dist/esm/processors/MoveOperationHelper.js.map +1 -0
  329. package/dist/esm/processors/RevisionAwareProcessor.js +232 -0
  330. package/dist/esm/processors/RevisionAwareProcessor.js.map +1 -0
  331. package/dist/esm/processors/RevisionWalker.js +278 -0
  332. package/dist/esm/processors/RevisionWalker.js.map +1 -0
  333. package/dist/{utils → esm/processors}/SelectiveRevisionAcceptor.js +81 -42
  334. package/dist/esm/processors/SelectiveRevisionAcceptor.js.map +1 -0
  335. package/dist/esm/processors/ShadingResolver.js +66 -0
  336. package/dist/esm/processors/ShadingResolver.js.map +1 -0
  337. package/dist/esm/processors/acceptRevisions.js +416 -0
  338. package/dist/esm/processors/acceptRevisions.js.map +1 -0
  339. package/dist/esm/processors/cnfStyleDecoder.js +89 -0
  340. package/dist/esm/processors/cnfStyleDecoder.js.map +1 -0
  341. package/dist/esm/processors/stripTrackedChanges.js +201 -0
  342. package/dist/esm/processors/stripTrackedChanges.js.map +1 -0
  343. package/dist/esm/tracking/DocumentTrackingContext.js +531 -0
  344. package/dist/esm/tracking/DocumentTrackingContext.js.map +1 -0
  345. package/dist/esm/tracking/TrackingContext.js +2 -0
  346. package/dist/esm/tracking/TrackingContext.js.map +1 -0
  347. package/dist/esm/types/compatibility-types.js +8 -0
  348. package/dist/esm/types/compatibility-types.js.map +1 -0
  349. package/dist/esm/types/document-types.js +2 -0
  350. package/dist/esm/types/document-types.js.map +1 -0
  351. package/dist/esm/types/formatting.js +2 -0
  352. package/dist/esm/types/formatting.js.map +1 -0
  353. package/dist/esm/types/list-types.js +2 -0
  354. package/dist/esm/types/list-types.js.map +1 -0
  355. package/dist/esm/types/settings-types.js +2 -0
  356. package/dist/esm/types/settings-types.js.map +1 -0
  357. package/dist/esm/types/styleConfig.js +2 -0
  358. package/dist/esm/types/styleConfig.js.map +1 -0
  359. package/dist/esm/utils/KeyedRegistry.js +32 -0
  360. package/dist/esm/utils/KeyedRegistry.js.map +1 -0
  361. package/dist/esm/utils/corruptionDetection.js +155 -0
  362. package/dist/esm/utils/corruptionDetection.js.map +1 -0
  363. package/dist/esm/utils/dateFormatting.js +4 -0
  364. package/dist/esm/utils/dateFormatting.js.map +1 -0
  365. package/dist/esm/utils/deepClone.js +40 -0
  366. package/dist/esm/utils/deepClone.js.map +1 -0
  367. package/dist/esm/utils/deepEqual.js +47 -0
  368. package/dist/esm/utils/deepEqual.js.map +1 -0
  369. package/dist/esm/utils/diagnostics.js +69 -0
  370. package/dist/esm/utils/diagnostics.js.map +1 -0
  371. package/dist/esm/utils/errorHandling.js +36 -0
  372. package/dist/esm/utils/errorHandling.js.map +1 -0
  373. package/dist/esm/utils/formatting.js +93 -0
  374. package/dist/esm/utils/formatting.js.map +1 -0
  375. package/dist/esm/utils/list-detection.js +148 -0
  376. package/dist/esm/utils/list-detection.js.map +1 -0
  377. package/dist/esm/utils/logger.js +205 -0
  378. package/dist/esm/utils/logger.js.map +1 -0
  379. package/dist/esm/utils/parsingHelpers.js +56 -0
  380. package/dist/esm/utils/parsingHelpers.js.map +1 -0
  381. package/dist/esm/utils/textDiff.js +42 -0
  382. package/dist/esm/utils/textDiff.js.map +1 -0
  383. package/dist/esm/utils/units.js +152 -0
  384. package/dist/esm/utils/units.js.map +1 -0
  385. package/dist/esm/utils/validation.js +285 -0
  386. package/dist/esm/utils/validation.js.map +1 -0
  387. package/dist/esm/utils/xmlSanitization.js +54 -0
  388. package/dist/esm/utils/xmlSanitization.js.map +1 -0
  389. package/dist/esm/validation/RevisionAutoFixer.js +340 -0
  390. package/dist/esm/validation/RevisionAutoFixer.js.map +1 -0
  391. package/dist/esm/validation/RevisionValidator.js +240 -0
  392. package/dist/esm/validation/RevisionValidator.js.map +1 -0
  393. package/dist/esm/validation/ValidationRuleRegistry.js +40 -0
  394. package/dist/esm/validation/ValidationRuleRegistry.js.map +1 -0
  395. package/dist/esm/validation/ValidationRules.js +92 -0
  396. package/dist/esm/validation/ValidationRules.js.map +1 -0
  397. package/dist/esm/validation/index.js +4 -0
  398. package/dist/esm/validation/index.js.map +1 -0
  399. package/dist/esm/xml/XMLBuilder.js +434 -0
  400. package/dist/esm/xml/XMLBuilder.js.map +1 -0
  401. package/dist/esm/xml/XMLParser.js +486 -0
  402. package/dist/esm/xml/XMLParser.js.map +1 -0
  403. package/dist/esm/zip/ZipHandler.js +298 -0
  404. package/dist/esm/zip/ZipHandler.js.map +1 -0
  405. package/dist/esm/zip/ZipReader.js +147 -0
  406. package/dist/esm/zip/ZipReader.js.map +1 -0
  407. package/dist/esm/zip/ZipWriter.js +199 -0
  408. package/dist/esm/zip/ZipWriter.js.map +1 -0
  409. package/dist/esm/zip/errors.js +43 -0
  410. package/dist/esm/zip/errors.js.map +1 -0
  411. package/dist/esm/zip/types.js +31 -0
  412. package/dist/esm/zip/types.js.map +1 -0
  413. package/dist/formatting/AbstractNumbering.d.ts +2 -2
  414. package/dist/formatting/AbstractNumbering.d.ts.map +1 -1
  415. package/dist/formatting/AbstractNumbering.js +33 -33
  416. package/dist/formatting/AbstractNumbering.js.map +1 -1
  417. package/dist/formatting/NumberingInstance.d.ts +2 -2
  418. package/dist/formatting/NumberingInstance.d.ts.map +1 -1
  419. package/dist/formatting/NumberingInstance.js +7 -7
  420. package/dist/formatting/NumberingInstance.js.map +1 -1
  421. package/dist/formatting/NumberingLevel.d.ts +11 -2
  422. package/dist/formatting/NumberingLevel.d.ts.map +1 -1
  423. package/dist/formatting/NumberingLevel.js +111 -25
  424. package/dist/formatting/NumberingLevel.js.map +1 -1
  425. package/dist/formatting/NumberingManager.d.ts +4 -4
  426. package/dist/formatting/NumberingManager.d.ts.map +1 -1
  427. package/dist/formatting/NumberingManager.js +28 -28
  428. package/dist/formatting/NumberingManager.js.map +1 -1
  429. package/dist/formatting/Style.d.ts +14 -7
  430. package/dist/formatting/Style.d.ts.map +1 -1
  431. package/dist/formatting/Style.js +309 -112
  432. package/dist/formatting/Style.js.map +1 -1
  433. package/dist/formatting/StylesManager.d.ts +2 -2
  434. package/dist/formatting/StylesManager.d.ts.map +1 -1
  435. package/dist/formatting/StylesManager.js +52 -52
  436. package/dist/formatting/StylesManager.js.map +1 -1
  437. package/dist/helpers/CleanupHelper.d.ts +1 -1
  438. package/dist/helpers/CleanupHelper.d.ts.map +1 -1
  439. package/dist/helpers/CleanupHelper.js +15 -15
  440. package/dist/helpers/CleanupHelper.js.map +1 -1
  441. package/dist/index.d.ts +81 -90
  442. package/dist/index.d.ts.map +1 -1
  443. package/dist/index.js +286 -317
  444. package/dist/index.js.map +1 -1
  445. package/dist/internal.d.ts +16 -0
  446. package/dist/internal.d.ts.map +1 -0
  447. package/dist/internal.js +42 -0
  448. package/dist/internal.js.map +1 -0
  449. package/dist/managers/DrawingManager.d.ts +3 -3
  450. package/dist/managers/DrawingManager.d.ts.map +1 -1
  451. package/dist/managers/DrawingManager.js +12 -12
  452. package/dist/managers/DrawingManager.js.map +1 -1
  453. package/dist/{utils → processors}/ChangelogGenerator.d.ts +2 -2
  454. package/dist/processors/ChangelogGenerator.d.ts.map +1 -0
  455. package/dist/{utils → processors}/ChangelogGenerator.js +2 -2
  456. package/dist/processors/ChangelogGenerator.js.map +1 -0
  457. package/dist/processors/CompatibilityUpgrader.d.ts.map +1 -0
  458. package/dist/{utils → processors}/CompatibilityUpgrader.js +10 -10
  459. package/dist/processors/CompatibilityUpgrader.js.map +1 -0
  460. package/dist/{utils → processors}/InMemoryRevisionAcceptor.d.ts +3 -3
  461. package/dist/processors/InMemoryRevisionAcceptor.d.ts.map +1 -0
  462. package/dist/{utils → processors}/InMemoryRevisionAcceptor.js +84 -27
  463. package/dist/processors/InMemoryRevisionAcceptor.js.map +1 -0
  464. package/dist/{utils → processors}/MoveOperationHelper.d.ts +4 -4
  465. package/dist/processors/MoveOperationHelper.d.ts.map +1 -0
  466. package/dist/{utils → processors}/MoveOperationHelper.js +10 -10
  467. package/dist/processors/MoveOperationHelper.js.map +1 -0
  468. package/dist/{utils → processors}/RevisionAwareProcessor.d.ts +3 -3
  469. package/dist/processors/RevisionAwareProcessor.d.ts.map +1 -0
  470. package/dist/{utils → processors}/RevisionAwareProcessor.js +2 -2
  471. package/dist/processors/RevisionAwareProcessor.js.map +1 -0
  472. package/dist/{utils → processors}/RevisionWalker.d.ts +2 -1
  473. package/dist/processors/RevisionWalker.d.ts.map +1 -0
  474. package/dist/{utils → processors}/RevisionWalker.js +28 -0
  475. package/dist/processors/RevisionWalker.js.map +1 -0
  476. package/dist/{utils → processors}/SelectiveRevisionAcceptor.d.ts +4 -3
  477. package/dist/processors/SelectiveRevisionAcceptor.d.ts.map +1 -0
  478. package/dist/processors/SelectiveRevisionAcceptor.js +402 -0
  479. package/dist/processors/SelectiveRevisionAcceptor.js.map +1 -0
  480. package/dist/processors/ShadingResolver.d.ts +6 -0
  481. package/dist/processors/ShadingResolver.d.ts.map +1 -0
  482. package/dist/{utils → processors}/ShadingResolver.js +2 -2
  483. package/dist/processors/ShadingResolver.js.map +1 -0
  484. package/dist/{utils → processors}/acceptRevisions.d.ts +1 -1
  485. package/dist/processors/acceptRevisions.d.ts.map +1 -0
  486. package/dist/{utils → processors}/acceptRevisions.js +24 -4
  487. package/dist/processors/acceptRevisions.js.map +1 -0
  488. package/dist/{utils → processors}/cnfStyleDecoder.d.ts +1 -1
  489. package/dist/processors/cnfStyleDecoder.d.ts.map +1 -0
  490. package/dist/processors/cnfStyleDecoder.js.map +1 -0
  491. package/dist/processors/stripTrackedChanges.d.ts +3 -0
  492. package/dist/processors/stripTrackedChanges.d.ts.map +1 -0
  493. package/dist/{utils → processors}/stripTrackedChanges.js +16 -6
  494. package/dist/processors/stripTrackedChanges.js.map +1 -0
  495. package/dist/tracking/DocumentTrackingContext.d.ts +4 -4
  496. package/dist/tracking/DocumentTrackingContext.d.ts.map +1 -1
  497. package/dist/tracking/DocumentTrackingContext.js +38 -43
  498. package/dist/tracking/DocumentTrackingContext.js.map +1 -1
  499. package/dist/tracking/TrackingContext.d.ts +8 -8
  500. package/dist/tracking/TrackingContext.d.ts.map +1 -1
  501. package/dist/tracking/TrackingContext.js.map +1 -1
  502. package/dist/types/document-types.d.ts +28 -0
  503. package/dist/types/document-types.d.ts.map +1 -0
  504. package/dist/types/document-types.js +3 -0
  505. package/dist/types/document-types.js.map +1 -0
  506. package/dist/types/formatting.d.ts +4 -4
  507. package/dist/types/formatting.d.ts.map +1 -1
  508. package/dist/types/formatting.js.map +1 -1
  509. package/dist/types/settings-types.d.ts +6 -0
  510. package/dist/types/settings-types.d.ts.map +1 -1
  511. package/dist/types/settings-types.js.map +1 -1
  512. package/dist/utils/KeyedRegistry.d.ts +13 -0
  513. package/dist/utils/KeyedRegistry.d.ts.map +1 -0
  514. package/dist/utils/KeyedRegistry.js +36 -0
  515. package/dist/utils/KeyedRegistry.js.map +1 -0
  516. package/dist/utils/corruptionDetection.d.ts +1 -1
  517. package/dist/utils/corruptionDetection.d.ts.map +1 -1
  518. package/dist/utils/corruptionDetection.js +4 -4
  519. package/dist/utils/corruptionDetection.js.map +1 -1
  520. package/dist/utils/deepEqual.d.ts +2 -0
  521. package/dist/utils/deepEqual.d.ts.map +1 -0
  522. package/dist/utils/deepEqual.js +50 -0
  523. package/dist/utils/deepEqual.js.map +1 -0
  524. package/dist/utils/list-detection.d.ts +2 -2
  525. package/dist/utils/list-detection.d.ts.map +1 -1
  526. package/dist/utils/list-detection.js.map +1 -1
  527. package/dist/utils/parsingHelpers.d.ts +1 -1
  528. package/dist/utils/parsingHelpers.d.ts.map +1 -1
  529. package/dist/utils/parsingHelpers.js +2 -2
  530. package/dist/utils/parsingHelpers.js.map +1 -1
  531. package/dist/utils/validation.js +7 -7
  532. package/dist/utils/validation.js.map +1 -1
  533. package/dist/utils/xmlSanitization.js +2 -2
  534. package/dist/utils/xmlSanitization.js.map +1 -1
  535. package/dist/validation/RevisionAutoFixer.d.ts +4 -4
  536. package/dist/validation/RevisionAutoFixer.d.ts.map +1 -1
  537. package/dist/validation/RevisionAutoFixer.js +11 -11
  538. package/dist/validation/RevisionAutoFixer.js.map +1 -1
  539. package/dist/validation/RevisionValidator.d.ts +5 -4
  540. package/dist/validation/RevisionValidator.d.ts.map +1 -1
  541. package/dist/validation/RevisionValidator.js +29 -30
  542. package/dist/validation/RevisionValidator.js.map +1 -1
  543. package/dist/validation/ValidationRuleRegistry.d.ts +27 -0
  544. package/dist/validation/ValidationRuleRegistry.d.ts.map +1 -0
  545. package/dist/validation/ValidationRuleRegistry.js +43 -0
  546. package/dist/validation/ValidationRuleRegistry.js.map +1 -0
  547. package/dist/validation/index.d.ts +3 -3
  548. package/dist/validation/index.d.ts.map +1 -1
  549. package/dist/validation/index.js +10 -10
  550. package/dist/validation/index.js.map +1 -1
  551. package/dist/xml/XMLBuilder.d.ts +6 -1
  552. package/dist/xml/XMLBuilder.d.ts.map +1 -1
  553. package/dist/xml/XMLBuilder.js +11 -6
  554. package/dist/xml/XMLBuilder.js.map +1 -1
  555. package/dist/xml/XMLParser.js +6 -6
  556. package/dist/xml/XMLParser.js.map +1 -1
  557. package/dist/zip/ZipHandler.d.ts +1 -1
  558. package/dist/zip/ZipHandler.d.ts.map +1 -1
  559. package/dist/zip/ZipHandler.js +8 -8
  560. package/dist/zip/ZipHandler.js.map +1 -1
  561. package/dist/zip/ZipReader.d.ts +1 -1
  562. package/dist/zip/ZipReader.d.ts.map +1 -1
  563. package/dist/zip/ZipReader.js +14 -14
  564. package/dist/zip/ZipReader.js.map +1 -1
  565. package/dist/zip/ZipWriter.d.ts +1 -1
  566. package/dist/zip/ZipWriter.d.ts.map +1 -1
  567. package/dist/zip/ZipWriter.js +10 -10
  568. package/dist/zip/ZipWriter.js.map +1 -1
  569. package/package.json +20 -4
  570. package/src/constants/legacyCompatFlags.ts +1 -1
  571. package/src/core/Document.ts +478 -167
  572. package/src/core/DocumentContent.ts +14 -11
  573. package/src/core/DocumentEvents.ts +90 -0
  574. package/src/core/DocumentGenerator.ts +49 -22
  575. package/src/core/DocumentParser.ts +2235 -620
  576. package/src/core/DocumentValidator.ts +7 -7
  577. package/src/core/ElementRegistry.ts +69 -0
  578. package/src/core/Relationship.ts +1 -1
  579. package/src/core/RelationshipManager.ts +4 -4
  580. package/src/elements/AlternateContent.ts +1 -1
  581. package/src/elements/Bookmark.ts +52 -4
  582. package/src/elements/BookmarkManager.ts +2 -2
  583. package/src/elements/Comment.ts +3 -3
  584. package/src/elements/CommentManager.ts +4 -4
  585. package/src/elements/CommonTypes.ts +45 -7
  586. package/src/elements/CustomXml.ts +1 -1
  587. package/src/elements/Endnote.ts +2 -2
  588. package/src/elements/EndnoteManager.ts +3 -3
  589. package/src/elements/Field.ts +44 -10
  590. package/src/elements/FieldHelpers.ts +2 -2
  591. package/src/elements/Footer.ts +4 -4
  592. package/src/elements/Footnote.ts +2 -2
  593. package/src/elements/FootnoteManager.ts +3 -3
  594. package/src/elements/Header.ts +4 -4
  595. package/src/elements/HeaderFooterManager.ts +2 -2
  596. package/src/elements/Hyperlink.ts +16 -12
  597. package/src/elements/Image.ts +3 -3
  598. package/src/elements/ImageManager.ts +2 -2
  599. package/src/elements/ImageRun.ts +3 -3
  600. package/src/elements/MathElement.ts +1 -1
  601. package/src/elements/Paragraph.ts +221 -88
  602. package/src/elements/PreservedElement.ts +1 -1
  603. package/src/elements/PropertyChangeTypes.ts +2 -2
  604. package/src/elements/RangeMarker.ts +153 -12
  605. package/src/elements/RegisteredBodyElement.ts +52 -0
  606. package/src/elements/Revision.ts +14 -14
  607. package/src/elements/RevisionContent.ts +3 -3
  608. package/src/elements/RevisionManager.ts +3 -3
  609. package/src/elements/Run.ts +221 -94
  610. package/src/elements/Section.ts +136 -69
  611. package/src/elements/Shape.ts +4 -4
  612. package/src/elements/StructuredDocumentTag.ts +3 -3
  613. package/src/elements/Table.ts +91 -27
  614. package/src/elements/TableCell.ts +62 -34
  615. package/src/elements/TableGridChange.ts +1 -1
  616. package/src/elements/TableOfContents.ts +1 -1
  617. package/src/elements/TableOfContentsElement.ts +2 -2
  618. package/src/elements/TableRow.ts +192 -48
  619. package/src/elements/TextBox.ts +5 -5
  620. package/src/formatting/AbstractNumbering.ts +3 -3
  621. package/src/formatting/NumberingInstance.ts +2 -2
  622. package/src/formatting/NumberingLevel.ts +201 -10
  623. package/src/formatting/NumberingManager.ts +5 -5
  624. package/src/formatting/Style.ts +382 -86
  625. package/src/formatting/StylesManager.ts +4 -4
  626. package/src/helpers/CleanupHelper.ts +6 -6
  627. package/src/index.ts +118 -127
  628. package/src/internal.ts +79 -0
  629. package/src/managers/DrawingManager.ts +3 -3
  630. package/src/{utils → processors}/ChangelogGenerator.ts +3 -3
  631. package/src/{utils → processors}/CompatibilityUpgrader.ts +2 -2
  632. package/src/{utils → processors}/InMemoryRevisionAcceptor.ts +100 -12
  633. package/src/{utils → processors}/MoveOperationHelper.ts +5 -5
  634. package/src/{utils → processors}/RevisionAwareProcessor.ts +3 -3
  635. package/src/{utils → processors}/RevisionWalker.ts +42 -1
  636. package/src/{utils → processors}/SelectiveRevisionAcceptor.ts +98 -39
  637. package/src/{utils → processors}/ShadingResolver.ts +5 -5
  638. package/src/{utils → processors}/acceptRevisions.ts +77 -9
  639. package/src/{utils → processors}/cnfStyleDecoder.ts +1 -1
  640. package/src/{utils → processors}/stripTrackedChanges.ts +35 -10
  641. package/src/tracking/DocumentTrackingContext.ts +12 -14
  642. package/src/tracking/TrackingContext.ts +8 -8
  643. package/src/types/document-types.ts +53 -0
  644. package/src/types/formatting.ts +4 -4
  645. package/src/types/settings-types.ts +32 -0
  646. package/src/utils/KeyedRegistry.ts +41 -0
  647. package/src/utils/corruptionDetection.ts +2 -2
  648. package/src/utils/deepEqual.ts +58 -0
  649. package/src/utils/list-detection.ts +2 -2
  650. package/src/utils/parsingHelpers.ts +11 -3
  651. package/src/utils/validation.ts +3 -3
  652. package/src/utils/xmlSanitization.ts +1 -1
  653. package/src/validation/RevisionAutoFixer.ts +5 -5
  654. package/src/validation/RevisionValidator.ts +39 -28
  655. package/src/validation/ValidationRuleRegistry.ts +86 -0
  656. package/src/validation/index.ts +3 -3
  657. package/src/xml/XMLBuilder.ts +13 -3
  658. package/src/xml/XMLParser.ts +2 -2
  659. package/src/zip/ZipHandler.ts +4 -4
  660. package/src/zip/ZipReader.ts +3 -3
  661. package/src/zip/ZipWriter.ts +3 -3
  662. package/dist/utils/ChangelogGenerator.d.ts.map +0 -1
  663. package/dist/utils/ChangelogGenerator.js.map +0 -1
  664. package/dist/utils/CompatibilityUpgrader.d.ts.map +0 -1
  665. package/dist/utils/CompatibilityUpgrader.js.map +0 -1
  666. package/dist/utils/InMemoryRevisionAcceptor.d.ts.map +0 -1
  667. package/dist/utils/InMemoryRevisionAcceptor.js.map +0 -1
  668. package/dist/utils/MoveOperationHelper.d.ts.map +0 -1
  669. package/dist/utils/MoveOperationHelper.js.map +0 -1
  670. package/dist/utils/RevisionAwareProcessor.d.ts.map +0 -1
  671. package/dist/utils/RevisionAwareProcessor.js.map +0 -1
  672. package/dist/utils/RevisionWalker.d.ts.map +0 -1
  673. package/dist/utils/RevisionWalker.js.map +0 -1
  674. package/dist/utils/SelectiveRevisionAcceptor.d.ts.map +0 -1
  675. package/dist/utils/SelectiveRevisionAcceptor.js.map +0 -1
  676. package/dist/utils/ShadingResolver.d.ts +0 -6
  677. package/dist/utils/ShadingResolver.d.ts.map +0 -1
  678. package/dist/utils/ShadingResolver.js.map +0 -1
  679. package/dist/utils/acceptRevisions.d.ts.map +0 -1
  680. package/dist/utils/acceptRevisions.js.map +0 -1
  681. package/dist/utils/cnfStyleDecoder.d.ts.map +0 -1
  682. package/dist/utils/cnfStyleDecoder.js.map +0 -1
  683. package/dist/utils/stripTrackedChanges.d.ts +0 -3
  684. package/dist/utils/stripTrackedChanges.d.ts.map +0 -1
  685. package/dist/utils/stripTrackedChanges.js.map +0 -1
  686. package/src/__tests__/helper-methods.test.ts +0 -512
  687. package/src/constants/CLAUDE.md +0 -28
  688. package/src/core/CLAUDE.md +0 -113
  689. package/src/elements/CLAUDE.md +0 -142
  690. package/src/formatting/CLAUDE.md +0 -78
  691. package/src/managers/CLAUDE.md +0 -47
  692. package/src/tracking/CLAUDE.md +0 -30
  693. package/src/types/CLAUDE.md +0 -39
  694. package/src/utils/CLAUDE.md +0 -168
  695. package/src/validation/CLAUDE.md +0 -40
  696. package/src/xml/CLAUDE.md +0 -65
  697. package/src/zip/CLAUDE.md +0 -55
  698. /package/dist/{utils → processors}/CompatibilityUpgrader.d.ts +0 -0
  699. /package/dist/{utils → processors}/cnfStyleDecoder.js +0 -0
@@ -0,0 +1,298 @@
1
+ import { ZipReader } from './ZipReader.js';
2
+ import { ZipWriter } from './ZipWriter.js';
3
+ import { DEFAULT_SIZE_LIMITS, } from './types.js';
4
+ import { getGlobalLogger, createScopedLogger } from '../utils/logger.js';
5
+ function getLogger() {
6
+ return createScopedLogger(getGlobalLogger(), 'ZipHandler');
7
+ }
8
+ export class ZipHandler {
9
+ reader;
10
+ writer;
11
+ mode = 'write';
12
+ constructor() {
13
+ this.reader = new ZipReader();
14
+ this.writer = new ZipWriter();
15
+ }
16
+ validateDocumentSize(sizeMB, limits) {
17
+ const logger = getLogger();
18
+ const { warningSizeMB, maxSizeMB } = limits;
19
+ if (maxSizeMB > 0 && sizeMB > maxSizeMB) {
20
+ logger.error('Document exceeds maximum size', { sizeMB: sizeMB.toFixed(1), maxSizeMB });
21
+ throw new Error(`Document size (${sizeMB.toFixed(1)}MB) exceeds maximum supported size (${maxSizeMB}MB). ` +
22
+ `This would likely cause out-of-memory errors. Consider:\n` +
23
+ `- Compressing/optimizing images\n` +
24
+ `- Splitting into multiple documents\n` +
25
+ `- Processing on a machine with more memory\n` +
26
+ `- Increasing maxSizeMB via LoadOptions.sizeLimits (not recommended)`);
27
+ }
28
+ if (warningSizeMB > 0 && sizeMB > warningSizeMB) {
29
+ logger.warn('Large document detected', { sizeMB: sizeMB.toFixed(1), warningSizeMB });
30
+ }
31
+ }
32
+ getSizeLimits(options) {
33
+ return {
34
+ ...DEFAULT_SIZE_LIMITS,
35
+ ...options.sizeLimits,
36
+ };
37
+ }
38
+ async load(filePath, options = {}) {
39
+ const logger = getLogger();
40
+ logger.info('Loading DOCX file', { path: filePath });
41
+ const { promises: fs } = await import('fs');
42
+ const stats = await fs.stat(filePath);
43
+ const sizeMB = stats.size / (1024 * 1024);
44
+ const limits = this.getSizeLimits(options);
45
+ this.validateDocumentSize(sizeMB, limits);
46
+ await this.reader.loadFromFile(filePath, options);
47
+ const files = this.reader.getAllFiles();
48
+ this.writer.clear();
49
+ this.writer.addFiles(files);
50
+ this.mode = 'modify';
51
+ logger.info('DOCX file loaded', { fileCount: files.size, sizeMB: sizeMB.toFixed(2) });
52
+ }
53
+ async loadFromBuffer(buffer, options = {}) {
54
+ const logger = getLogger();
55
+ logger.info('Loading DOCX from buffer', { bufferSize: buffer.length });
56
+ const sizeMB = buffer.length / (1024 * 1024);
57
+ const limits = this.getSizeLimits(options);
58
+ this.validateDocumentSize(sizeMB, limits);
59
+ await this.reader.loadFromBuffer(buffer, options);
60
+ const files = this.reader.getAllFiles();
61
+ this.writer.clear();
62
+ this.writer.addFiles(files);
63
+ this.mode = 'modify';
64
+ logger.info('DOCX buffer loaded', { fileCount: files.size, sizeMB: sizeMB.toFixed(2) });
65
+ }
66
+ addFile(filePath, content, options = {}) {
67
+ this.writer.addFile(filePath, content, options);
68
+ }
69
+ addFiles(files, options = {}) {
70
+ this.writer.addFiles(files, options);
71
+ }
72
+ updateFile(filePath, content, options = {}) {
73
+ if (!this.hasFile(filePath)) {
74
+ return false;
75
+ }
76
+ this.addFile(filePath, content, options);
77
+ return true;
78
+ }
79
+ removeFile(filePath) {
80
+ return this.writer.removeFile(filePath);
81
+ }
82
+ getFile(filePath) {
83
+ return this.writer.getFile(filePath);
84
+ }
85
+ getFileAsString(filePath) {
86
+ const file = this.getFile(filePath);
87
+ if (!file) {
88
+ return undefined;
89
+ }
90
+ if (file.isBinary) {
91
+ throw new Error(`Cannot convert binary file "${filePath}" to string. ` +
92
+ `Binary files (images, fonts, etc.) cannot be safely converted to UTF-8 strings. ` +
93
+ `Use getFileAsBuffer() instead to access the raw bytes.`);
94
+ }
95
+ if (Buffer.isBuffer(file.content)) {
96
+ return file.content.toString('utf8');
97
+ }
98
+ return file.content;
99
+ }
100
+ getFileAsBuffer(filePath) {
101
+ const file = this.getFile(filePath);
102
+ if (!file) {
103
+ return undefined;
104
+ }
105
+ if (Buffer.isBuffer(file.content)) {
106
+ return file.content;
107
+ }
108
+ return Buffer.from(file.content, 'utf8');
109
+ }
110
+ getAllFiles() {
111
+ return this.writer.getAllFiles();
112
+ }
113
+ getFilePaths() {
114
+ return this.writer.getFilePaths();
115
+ }
116
+ hasFile(filePath) {
117
+ return this.writer.hasFile(filePath);
118
+ }
119
+ getFileCount() {
120
+ return this.writer.getFileCount();
121
+ }
122
+ renameFile(oldPath, newPath) {
123
+ const file = this.getFile(oldPath);
124
+ if (!file) {
125
+ return false;
126
+ }
127
+ this.addFile(newPath, file.content, {
128
+ binary: file.isBinary,
129
+ date: file.date,
130
+ });
131
+ this.removeFile(oldPath);
132
+ return true;
133
+ }
134
+ copyFile(srcPath, destPath) {
135
+ const file = this.getFile(srcPath);
136
+ if (!file) {
137
+ return false;
138
+ }
139
+ this.addFile(destPath, file.content, {
140
+ binary: file.isBinary,
141
+ date: file.date,
142
+ });
143
+ return true;
144
+ }
145
+ moveFile(srcPath, destPath) {
146
+ if (!this.copyFile(srcPath, destPath)) {
147
+ return false;
148
+ }
149
+ this.removeFile(srcPath);
150
+ return true;
151
+ }
152
+ existsOrThrow(filePath) {
153
+ if (!this.hasFile(filePath)) {
154
+ throw new Error(`File not found in archive: ${filePath}`);
155
+ }
156
+ }
157
+ removeFiles(filePaths) {
158
+ let count = 0;
159
+ for (const filePath of filePaths) {
160
+ if (this.removeFile(filePath)) {
161
+ count++;
162
+ }
163
+ }
164
+ return count;
165
+ }
166
+ getFilesByExtension(extension) {
167
+ const ext = extension.startsWith('.') ? extension : `.${extension}`;
168
+ const files = [];
169
+ for (const [path, file] of this.getAllFiles()) {
170
+ if (path.toLowerCase().endsWith(ext.toLowerCase())) {
171
+ files.push(file);
172
+ }
173
+ }
174
+ return files;
175
+ }
176
+ getTotalSize() {
177
+ let totalSize = 0;
178
+ for (const file of this.getAllFiles().values()) {
179
+ totalSize += file.size;
180
+ }
181
+ return totalSize;
182
+ }
183
+ getStats() {
184
+ let textCount = 0;
185
+ let binaryCount = 0;
186
+ const totalSize = this.getTotalSize();
187
+ const fileCount = this.getFileCount();
188
+ for (const file of this.getAllFiles().values()) {
189
+ if (file.isBinary) {
190
+ binaryCount++;
191
+ }
192
+ else {
193
+ textCount++;
194
+ }
195
+ }
196
+ return {
197
+ fileCount,
198
+ totalSize,
199
+ textFileCount: textCount,
200
+ binaryFileCount: binaryCount,
201
+ avgFileSize: fileCount > 0 ? Math.round(totalSize / fileCount) : 0,
202
+ };
203
+ }
204
+ isEmpty() {
205
+ return this.getFileCount() === 0;
206
+ }
207
+ getTextFiles() {
208
+ const files = [];
209
+ for (const file of this.getAllFiles().values()) {
210
+ if (!file.isBinary) {
211
+ files.push(file);
212
+ }
213
+ }
214
+ return files;
215
+ }
216
+ getBinaryFiles() {
217
+ const files = [];
218
+ for (const file of this.getAllFiles().values()) {
219
+ if (file.isBinary) {
220
+ files.push(file);
221
+ }
222
+ }
223
+ return files;
224
+ }
225
+ getMediaFiles() {
226
+ const files = [];
227
+ for (const [path, file] of this.getAllFiles()) {
228
+ if (path.startsWith('word/media/')) {
229
+ files.push(file);
230
+ }
231
+ }
232
+ return files;
233
+ }
234
+ async exportFile(internalPath, outputPath) {
235
+ const { promises: fs } = await import('fs');
236
+ const content = this.getFileAsBuffer(internalPath);
237
+ if (!content) {
238
+ throw new Error(`File not found in archive: ${internalPath}`);
239
+ }
240
+ await fs.writeFile(outputPath, content);
241
+ }
242
+ async importFile(sourcePath, internalPath, options = {}) {
243
+ const { promises: fs } = await import('fs');
244
+ const content = await fs.readFile(sourcePath);
245
+ this.addFile(internalPath, content, {
246
+ ...options,
247
+ binary: options.binary !== undefined ? options.binary : true,
248
+ });
249
+ }
250
+ async save(filePath, options = {}) {
251
+ const logger = getLogger();
252
+ logger.info('Saving DOCX file', { path: filePath, fileCount: this.getFileCount() });
253
+ await this.writer.saveToFile(filePath, options);
254
+ logger.info('DOCX file saved', { path: filePath });
255
+ }
256
+ async toBuffer(options = {}) {
257
+ const logger = getLogger();
258
+ logger.info('Generating DOCX buffer', { fileCount: this.getFileCount() });
259
+ const buffer = await this.writer.toBuffer(options);
260
+ logger.info('DOCX buffer generated', { bufferSize: buffer.length });
261
+ return buffer;
262
+ }
263
+ validate() {
264
+ const logger = getLogger();
265
+ logger.info('Validating DOCX structure');
266
+ this.writer.validate();
267
+ logger.info('DOCX structure valid');
268
+ }
269
+ clear() {
270
+ this.reader.clear();
271
+ this.writer.clear();
272
+ this.mode = 'write';
273
+ }
274
+ getMode() {
275
+ return this.mode;
276
+ }
277
+ isLoaded() {
278
+ return this.reader.isLoaded();
279
+ }
280
+ clone() {
281
+ const newHandler = new ZipHandler();
282
+ newHandler.writer = this.writer.clone();
283
+ newHandler.mode = this.mode;
284
+ return newHandler;
285
+ }
286
+ static async modify(inputPath, outputPath, modifier, loadOptions = {}, saveOptions = {}) {
287
+ const handler = new ZipHandler();
288
+ await handler.load(inputPath, loadOptions);
289
+ await modifier(handler);
290
+ await handler.save(outputPath, saveOptions);
291
+ }
292
+ static async create(outputPath, files, saveOptions = {}) {
293
+ const handler = new ZipHandler();
294
+ handler.addFiles(files);
295
+ await handler.save(outputPath, saveOptions);
296
+ }
297
+ }
298
+ //# sourceMappingURL=ZipHandler.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ZipHandler.js","sourceRoot":"","sources":["../../../src/zip/ZipHandler.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * ZipHandler - Main facade for ZIP archive operations\n * Provides a unified interface for reading and writing DOCX files\n */\n\nimport { ZipReader } from './ZipReader.js';\nimport { ZipWriter } from './ZipWriter.js';\nimport {\n ZipFile,\n FileMap,\n LoadOptions,\n SaveOptions,\n AddFileOptions,\n SizeLimitOptions,\n DEFAULT_SIZE_LIMITS,\n} from './types.js';\nimport { getGlobalLogger, createScopedLogger, ILogger } from '../utils/logger.js';\n\n// Create scoped logger for ZipHandler operations\nfunction getLogger(): ILogger {\n return createScopedLogger(getGlobalLogger(), 'ZipHandler');\n}\n\n/**\n * Main class for handling ZIP archives (DOCX files)\n * Combines reading and writing operations into a single interface\n */\nexport class ZipHandler {\n private reader: ZipReader;\n private writer: ZipWriter;\n private mode: 'read' | 'write' | 'modify' = 'write';\n\n constructor() {\n this.reader = new ZipReader();\n this.writer = new ZipWriter();\n }\n\n // ==================== SIZE VALIDATION ====================\n\n /**\n * Validates document size against configured limits\n * @param sizeMB - Size in megabytes\n * @param limits - Size limit options (merged with defaults)\n * @throws Error if size exceeds maximum\n */\n private validateDocumentSize(sizeMB: number, limits: Required<SizeLimitOptions>): void {\n const logger = getLogger();\n const { warningSizeMB, maxSizeMB } = limits;\n\n // Check maximum size (if enabled)\n if (maxSizeMB > 0 && sizeMB > maxSizeMB) {\n logger.error('Document exceeds maximum size', { sizeMB: sizeMB.toFixed(1), maxSizeMB });\n throw new Error(\n `Document size (${sizeMB.toFixed(1)}MB) exceeds maximum supported size (${maxSizeMB}MB). ` +\n `This would likely cause out-of-memory errors. Consider:\\n` +\n `- Compressing/optimizing images\\n` +\n `- Splitting into multiple documents\\n` +\n `- Processing on a machine with more memory\\n` +\n `- Increasing maxSizeMB via LoadOptions.sizeLimits (not recommended)`\n );\n }\n\n // Warn on large files (if enabled)\n if (warningSizeMB > 0 && sizeMB > warningSizeMB) {\n logger.warn('Large document detected', { sizeMB: sizeMB.toFixed(1), warningSizeMB });\n }\n }\n\n /**\n * Gets merged size limits from options and defaults\n * @param options - Load options that may contain size limits\n * @returns Merged size limits with all values defined\n */\n private getSizeLimits(options: LoadOptions): Required<SizeLimitOptions> {\n return {\n ...DEFAULT_SIZE_LIMITS,\n ...options.sizeLimits,\n };\n }\n\n // ==================== LOADING ====================\n\n /**\n * Loads a DOCX file from the filesystem\n * @param filePath - Path to the DOCX file\n * @param options - Load options (including configurable size limits)\n */\n async load(filePath: string, options: LoadOptions = {}): Promise<void> {\n const logger = getLogger();\n logger.info('Loading DOCX file', { path: filePath });\n\n // Check file size before loading\n const { promises: fs } = await import('fs');\n const stats = await fs.stat(filePath);\n const sizeMB = stats.size / (1024 * 1024);\n\n // Validate against configurable limits\n const limits = this.getSizeLimits(options);\n this.validateDocumentSize(sizeMB, limits);\n\n await this.reader.loadFromFile(filePath, options);\n\n // Copy all files from reader to writer for modification\n const files = this.reader.getAllFiles();\n this.writer.clear();\n this.writer.addFiles(files);\n\n this.mode = 'modify';\n logger.info('DOCX file loaded', { fileCount: files.size, sizeMB: sizeMB.toFixed(2) });\n }\n\n /**\n * Loads a DOCX file from a buffer\n * @param buffer - Buffer containing the DOCX data\n * @param options - Load options (including configurable size limits)\n */\n async loadFromBuffer(buffer: Buffer, options: LoadOptions = {}): Promise<void> {\n const logger = getLogger();\n logger.info('Loading DOCX from buffer', { bufferSize: buffer.length });\n\n // Check buffer size before loading\n const sizeMB = buffer.length / (1024 * 1024);\n\n // Validate against configurable limits\n const limits = this.getSizeLimits(options);\n this.validateDocumentSize(sizeMB, limits);\n\n await this.reader.loadFromBuffer(buffer, options);\n\n // Copy all files from reader to writer for modification\n const files = this.reader.getAllFiles();\n this.writer.clear();\n this.writer.addFiles(files);\n\n this.mode = 'modify';\n logger.info('DOCX buffer loaded', { fileCount: files.size, sizeMB: sizeMB.toFixed(2) });\n }\n\n // ==================== FILE OPERATIONS ====================\n\n /**\n * Adds a file to the archive\n * @param filePath - Path where the file will be stored in the archive\n * @param content - File content (string or Buffer)\n * @param options - Options for adding the file\n */\n addFile(filePath: string, content: string | Buffer, options: AddFileOptions = {}): void {\n this.writer.addFile(filePath, content, options);\n }\n\n /**\n * Adds multiple files to the archive\n * @param files - Map of file paths to contents\n * @param options - Options for adding files\n */\n addFiles(files: FileMap, options: AddFileOptions = {}): void {\n this.writer.addFiles(files, options);\n }\n\n /**\n * Updates an existing file in the archive\n * @param filePath - Path to the file to update\n * @param content - New content\n * @param options - Options for updating the file\n * @returns True if the file was updated, false if it didn't exist\n */\n updateFile(filePath: string, content: string | Buffer, options: AddFileOptions = {}): boolean {\n if (!this.hasFile(filePath)) {\n return false;\n }\n this.addFile(filePath, content, options);\n return true;\n }\n\n /**\n * Removes a file from the archive\n * @param filePath - Path to the file to remove\n * @returns True if the file was removed, false if it didn't exist\n */\n removeFile(filePath: string): boolean {\n return this.writer.removeFile(filePath);\n }\n\n /**\n * Gets a specific file from the archive\n * @param filePath - Path to the file within the archive\n * @returns The file data, or undefined if not found\n */\n getFile(filePath: string): ZipFile | undefined {\n return this.writer.getFile(filePath);\n }\n\n /**\n * Gets the content of a specific file as a string\n * @param filePath - Path to the file within the archive\n * @returns The file content as a UTF-8 string, or undefined if not found\n * @throws Error if attempting to convert a binary file to string (Issue #12 fix)\n *\n * **Encoding Note:**\n * - Returns UTF-8 decoded string content for text files\n * - All text content in DOCX files must be UTF-8 per OpenXML specification\n * - For binary files (images, fonts), throws an error to prevent garbage output\n * - Use getFileAsBuffer() for binary files instead\n */\n getFileAsString(filePath: string): string | undefined {\n const file = this.getFile(filePath);\n if (!file) {\n return undefined;\n }\n\n // Issue #12 fix: Prevent converting binary files to UTF-8 strings\n // This produces garbage output for images, fonts, etc.\n if (file.isBinary) {\n throw new Error(\n `Cannot convert binary file \"${filePath}\" to string. ` +\n `Binary files (images, fonts, etc.) cannot be safely converted to UTF-8 strings. ` +\n `Use getFileAsBuffer() instead to access the raw bytes.`\n );\n }\n\n // After ZipWriter fix (Issue #4), check actual content type instead of flag\n // Content from ZipWriter is always Buffer, content from ZipReader may be string\n if (Buffer.isBuffer(file.content)) {\n // Convert buffer to UTF-8 string\n return file.content.toString('utf8');\n }\n\n return file.content;\n }\n\n /**\n * Gets the content of a specific file as a buffer\n * @param filePath - Path to the file within the archive\n * @returns The file content as a Buffer, or undefined if not found\n *\n * **Encoding Note:**\n * - Returns Buffer with UTF-8 encoded content for text files\n * - For binary files, returns raw bytes unchanged\n * - All text content is guaranteed to be UTF-8 encoded\n */\n getFileAsBuffer(filePath: string): Buffer | undefined {\n const file = this.getFile(filePath);\n if (!file) {\n return undefined;\n }\n\n // After ZipWriter fix (Issue #4), check actual content type instead of flag\n // Content from ZipWriter is always Buffer, content from ZipReader may be string\n if (Buffer.isBuffer(file.content)) {\n return file.content;\n }\n\n // Encode string content as UTF-8 Buffer (for content from ZipReader)\n return Buffer.from(file.content, 'utf8');\n }\n\n /**\n * Gets all files from the archive\n * @returns Map of file paths to file data\n */\n getAllFiles(): FileMap {\n return this.writer.getAllFiles();\n }\n\n /**\n * Gets a list of all file paths in the archive\n * @returns Array of file paths\n */\n getFilePaths(): string[] {\n return this.writer.getFilePaths();\n }\n\n /**\n * Checks if a file exists in the archive\n * @param filePath - Path to check\n * @returns True if the file exists\n */\n hasFile(filePath: string): boolean {\n return this.writer.hasFile(filePath);\n }\n\n /**\n * Gets the number of files in the archive\n * @returns Number of files\n */\n getFileCount(): number {\n return this.writer.getFileCount();\n }\n\n // ==================== HELPER METHODS ====================\n\n /**\n * Renames a file in the archive\n * @param oldPath - Current path of the file\n * @param newPath - New path for the file\n * @returns True if the file was renamed, false if it didn't exist\n */\n renameFile(oldPath: string, newPath: string): boolean {\n const file = this.getFile(oldPath);\n if (!file) {\n return false;\n }\n this.addFile(newPath, file.content, {\n binary: file.isBinary,\n date: file.date,\n });\n this.removeFile(oldPath);\n return true;\n }\n\n /**\n * Copies a file within the archive\n * @param srcPath - Source file path\n * @param destPath - Destination file path\n * @returns True if the file was copied, false if source didn't exist\n */\n copyFile(srcPath: string, destPath: string): boolean {\n const file = this.getFile(srcPath);\n if (!file) {\n return false;\n }\n this.addFile(destPath, file.content, {\n binary: file.isBinary,\n date: file.date,\n });\n return true;\n }\n\n /**\n * Moves a file within the archive (copy and delete)\n * @param srcPath - Source file path\n * @param destPath - Destination file path\n * @returns True if the file was moved, false if source didn't exist\n */\n moveFile(srcPath: string, destPath: string): boolean {\n if (!this.copyFile(srcPath, destPath)) {\n return false;\n }\n this.removeFile(srcPath);\n return true;\n }\n\n /**\n * Checks if a file exists, throws if it doesn't\n * @param filePath - Path to check\n * @throws {Error} If file doesn't exist\n */\n existsOrThrow(filePath: string): void {\n if (!this.hasFile(filePath)) {\n throw new Error(`File not found in archive: ${filePath}`);\n }\n }\n\n /**\n * Removes multiple files from the archive\n * @param filePaths - Array of file paths to remove\n * @returns Number of files successfully removed\n */\n removeFiles(filePaths: string[]): number {\n let count = 0;\n for (const filePath of filePaths) {\n if (this.removeFile(filePath)) {\n count++;\n }\n }\n return count;\n }\n\n /**\n * Gets all files with a specific extension\n * @param extension - File extension (with or without leading dot)\n * @returns Array of files with the specified extension\n */\n getFilesByExtension(extension: string): ZipFile[] {\n const ext = extension.startsWith('.') ? extension : `.${extension}`;\n const files: ZipFile[] = [];\n for (const [path, file] of this.getAllFiles()) {\n if (path.toLowerCase().endsWith(ext.toLowerCase())) {\n files.push(file);\n }\n }\n return files;\n }\n\n /**\n * Gets the total uncompressed size of all files in the archive\n * @returns Total size in bytes\n */\n getTotalSize(): number {\n let totalSize = 0;\n for (const file of this.getAllFiles().values()) {\n totalSize += file.size;\n }\n return totalSize;\n }\n\n /**\n * Gets comprehensive statistics about the archive\n * @returns Statistics object\n */\n getStats(): {\n fileCount: number;\n totalSize: number;\n textFileCount: number;\n binaryFileCount: number;\n avgFileSize: number;\n } {\n let textCount = 0;\n let binaryCount = 0;\n const totalSize = this.getTotalSize();\n const fileCount = this.getFileCount();\n\n for (const file of this.getAllFiles().values()) {\n if (file.isBinary) {\n binaryCount++;\n } else {\n textCount++;\n }\n }\n\n return {\n fileCount,\n totalSize,\n textFileCount: textCount,\n binaryFileCount: binaryCount,\n avgFileSize: fileCount > 0 ? Math.round(totalSize / fileCount) : 0,\n };\n }\n\n /**\n * Checks if the archive is empty\n * @returns True if the archive has no files\n */\n isEmpty(): boolean {\n return this.getFileCount() === 0;\n }\n\n /**\n * Gets all text (non-binary) files from the archive\n * @returns Array of text files\n */\n getTextFiles(): ZipFile[] {\n const files: ZipFile[] = [];\n for (const file of this.getAllFiles().values()) {\n if (!file.isBinary) {\n files.push(file);\n }\n }\n return files;\n }\n\n /**\n * Gets all binary files from the archive\n * @returns Array of binary files\n */\n getBinaryFiles(): ZipFile[] {\n const files: ZipFile[] = [];\n for (const file of this.getAllFiles().values()) {\n if (file.isBinary) {\n files.push(file);\n }\n }\n return files;\n }\n\n /**\n * Gets all media files from the word/media/ directory\n * @returns Array of media files\n */\n getMediaFiles(): ZipFile[] {\n const files: ZipFile[] = [];\n for (const [path, file] of this.getAllFiles()) {\n if (path.startsWith('word/media/')) {\n files.push(file);\n }\n }\n return files;\n }\n\n /**\n * Exports a file from the archive to the filesystem\n * @param internalPath - Path of the file within the archive\n * @param outputPath - Path where the file will be saved\n */\n async exportFile(internalPath: string, outputPath: string): Promise<void> {\n const { promises: fs } = await import('fs');\n const content = this.getFileAsBuffer(internalPath);\n if (!content) {\n throw new Error(`File not found in archive: ${internalPath}`);\n }\n await fs.writeFile(outputPath, content);\n }\n\n /**\n * Imports a file from the filesystem into the archive\n * @param sourcePath - Path to the file on filesystem\n * @param internalPath - Path where the file will be stored in archive\n * @param options - Options for adding the file\n */\n async importFile(\n sourcePath: string,\n internalPath: string,\n options: AddFileOptions = {}\n ): Promise<void> {\n const { promises: fs } = await import('fs');\n const content = await fs.readFile(sourcePath);\n this.addFile(internalPath, content, {\n ...options,\n binary: options.binary !== undefined ? options.binary : true,\n });\n }\n\n // ==================== SAVING ====================\n\n /**\n * Saves the archive to a file\n * @param filePath - Path where the file will be saved\n * @param options - Save options\n */\n async save(filePath: string, options: SaveOptions = {}): Promise<void> {\n const logger = getLogger();\n logger.info('Saving DOCX file', { path: filePath, fileCount: this.getFileCount() });\n await this.writer.saveToFile(filePath, options);\n logger.info('DOCX file saved', { path: filePath });\n }\n\n /**\n * Generates the ZIP archive as a buffer\n * @param options - Save options\n * @returns Buffer containing the ZIP archive\n */\n async toBuffer(options: SaveOptions = {}): Promise<Buffer> {\n const logger = getLogger();\n logger.info('Generating DOCX buffer', { fileCount: this.getFileCount() });\n const buffer = await this.writer.toBuffer(options);\n logger.info('DOCX buffer generated', { bufferSize: buffer.length });\n return buffer;\n }\n\n // ==================== VALIDATION ====================\n\n /**\n * Validates the DOCX structure\n * @throws {MissingRequiredFileError} If required files are missing\n */\n validate(): void {\n const logger = getLogger();\n logger.info('Validating DOCX structure');\n this.writer.validate();\n logger.info('DOCX structure valid');\n }\n\n // ==================== UTILITY ====================\n\n /**\n * Creates a new empty archive\n */\n clear(): void {\n this.reader.clear();\n this.writer.clear();\n this.mode = 'write';\n }\n\n /**\n * Gets the current mode of the handler\n * @returns Current mode ('read', 'write', or 'modify')\n */\n getMode(): 'read' | 'write' | 'modify' {\n return this.mode;\n }\n\n /**\n * Checks if the archive has been loaded\n * @returns True if loaded\n */\n isLoaded(): boolean {\n return this.reader.isLoaded();\n }\n\n /**\n * Creates a clone of this handler with all its files\n * @returns A new ZipHandler instance with the same files\n */\n clone(): ZipHandler {\n const newHandler = new ZipHandler();\n newHandler.writer = this.writer.clone();\n newHandler.mode = this.mode;\n return newHandler;\n }\n\n // ==================== CONVENIENCE METHODS ====================\n\n /**\n * Reads a DOCX file, modifies it, and saves it back\n * @param inputPath - Path to the input DOCX file\n * @param outputPath - Path to save the modified file\n * @param modifier - Function that modifies the handler\n * @param loadOptions - Options for loading\n * @param saveOptions - Options for saving\n */\n static async modify(\n inputPath: string,\n outputPath: string,\n modifier: (handler: ZipHandler) => void | Promise<void>,\n loadOptions: LoadOptions = {},\n saveOptions: SaveOptions = {}\n ): Promise<void> {\n const handler = new ZipHandler();\n await handler.load(inputPath, loadOptions);\n await modifier(handler);\n await handler.save(outputPath, saveOptions);\n }\n\n /**\n * Creates a new DOCX file with the provided files\n * @param outputPath - Path to save the new file\n * @param files - Map of file paths to contents\n * @param saveOptions - Options for saving\n */\n static async create(\n outputPath: string,\n files: FileMap,\n saveOptions: SaveOptions = {}\n ): Promise<void> {\n const handler = new ZipHandler();\n handler.addFiles(files);\n await handler.save(outputPath, saveOptions);\n }\n}\n"]}
@@ -0,0 +1,147 @@
1
+ import JSZip from 'jszip';
2
+ import { promises as fs } from 'fs';
3
+ import { DocxNotFoundError, InvalidDocxError, CorruptedArchiveError, FileOperationError, } from './errors.js';
4
+ import { validateDocxStructure, isBinaryFile, normalizePath, isValidZipBuffer, } from '../utils/validation.js';
5
+ export class ZipReader {
6
+ zip = null;
7
+ files = new Map();
8
+ loaded = false;
9
+ async loadFromFile(filePath, options = {}) {
10
+ try {
11
+ try {
12
+ await fs.access(filePath);
13
+ }
14
+ catch {
15
+ throw new DocxNotFoundError(filePath);
16
+ }
17
+ const buffer = await fs.readFile(filePath);
18
+ await this.loadFromBuffer(buffer, options);
19
+ }
20
+ catch (error) {
21
+ if (error instanceof DocxNotFoundError) {
22
+ throw error;
23
+ }
24
+ const message = error instanceof Error ? error.message : String(error);
25
+ throw new FileOperationError('read', message);
26
+ }
27
+ }
28
+ async loadFromBuffer(buffer, options = {}) {
29
+ const { validate = true } = options;
30
+ try {
31
+ if (!isValidZipBuffer(buffer)) {
32
+ throw new InvalidDocxError('File is not a valid ZIP archive');
33
+ }
34
+ this.zip = await JSZip.loadAsync(buffer);
35
+ await this.extractFiles();
36
+ if (validate) {
37
+ this.validate();
38
+ }
39
+ this.loaded = true;
40
+ }
41
+ catch (error) {
42
+ if (error instanceof InvalidDocxError) {
43
+ throw error;
44
+ }
45
+ const message = error instanceof Error ? error.message : String(error);
46
+ throw new CorruptedArchiveError(message);
47
+ }
48
+ }
49
+ async extractFiles() {
50
+ if (!this.zip) {
51
+ throw new Error('ZIP archive not loaded');
52
+ }
53
+ this.files.clear();
54
+ const filePaths = Object.keys(this.zip.files).filter((path) => !this.zip.files[path].dir);
55
+ for (const filePath of filePaths) {
56
+ const normalizedPath = normalizePath(filePath);
57
+ const zipObject = this.zip.files[filePath];
58
+ if (!zipObject) {
59
+ continue;
60
+ }
61
+ const isBinary = isBinaryFile(normalizedPath);
62
+ let content;
63
+ if (isBinary) {
64
+ content = await zipObject.async('nodebuffer');
65
+ }
66
+ else {
67
+ content = await zipObject.async('string');
68
+ }
69
+ const date = zipObject.date;
70
+ this.files.set(normalizedPath, {
71
+ path: normalizedPath,
72
+ content,
73
+ isBinary,
74
+ size: isBinary ? content.length : content.length,
75
+ date,
76
+ });
77
+ }
78
+ }
79
+ validate() {
80
+ const filePaths = Array.from(this.files.keys());
81
+ validateDocxStructure(filePaths);
82
+ }
83
+ getFile(filePath) {
84
+ this.ensureLoaded();
85
+ const normalizedPath = normalizePath(filePath);
86
+ return this.files.get(normalizedPath);
87
+ }
88
+ getFileAsString(filePath) {
89
+ const file = this.getFile(filePath);
90
+ if (!file) {
91
+ return undefined;
92
+ }
93
+ if (Buffer.isBuffer(file.content)) {
94
+ return file.content.toString('utf8');
95
+ }
96
+ return file.content;
97
+ }
98
+ getFileAsBuffer(filePath) {
99
+ const file = this.getFile(filePath);
100
+ if (!file) {
101
+ return undefined;
102
+ }
103
+ if (Buffer.isBuffer(file.content)) {
104
+ return file.content;
105
+ }
106
+ return Buffer.from(file.content, 'utf8');
107
+ }
108
+ getAllFiles() {
109
+ this.ensureLoaded();
110
+ return new Map(this.files);
111
+ }
112
+ getFilePaths() {
113
+ this.ensureLoaded();
114
+ return Array.from(this.files.keys());
115
+ }
116
+ hasFile(filePath) {
117
+ this.ensureLoaded();
118
+ const normalizedPath = normalizePath(filePath);
119
+ return this.files.has(normalizedPath);
120
+ }
121
+ getFilesByPattern(pattern) {
122
+ this.ensureLoaded();
123
+ const regexPattern = pattern.replace(/\*/g, '.*').replace(/\?/g, '.');
124
+ const regex = new RegExp(`^${regexPattern}$`);
125
+ const matchingFiles = [];
126
+ for (const [path, file] of this.files) {
127
+ if (regex.test(path)) {
128
+ matchingFiles.push(file);
129
+ }
130
+ }
131
+ return matchingFiles;
132
+ }
133
+ ensureLoaded() {
134
+ if (!this.loaded) {
135
+ throw new Error('Archive not loaded. Call loadFromFile() or loadFromBuffer() first.');
136
+ }
137
+ }
138
+ isLoaded() {
139
+ return this.loaded;
140
+ }
141
+ clear() {
142
+ this.zip = null;
143
+ this.files.clear();
144
+ this.loaded = false;
145
+ }
146
+ }
147
+ //# sourceMappingURL=ZipReader.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ZipReader.js","sourceRoot":"","sources":["../../../src/zip/ZipReader.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * ZipReader - Handles reading ZIP archives (DOCX files)\n */\n\nimport JSZip from 'jszip';\nimport { promises as fs } from 'fs';\nimport { ZipFile, FileMap, LoadOptions } from './types.js';\nimport {\n DocxNotFoundError,\n InvalidDocxError,\n CorruptedArchiveError,\n FileOperationError,\n} from './errors.js';\nimport {\n validateDocxStructure,\n isBinaryFile,\n normalizePath,\n isValidZipBuffer,\n} from '../utils/validation.js';\n\n/**\n * Handles reading operations on ZIP archives\n */\nexport class ZipReader {\n private zip: JSZip | null = null;\n private files: FileMap = new Map();\n private loaded = false;\n\n /**\n * Loads a DOCX file from the filesystem\n * @param filePath - Path to the DOCX file\n * @param options - Load options\n */\n async loadFromFile(filePath: string, options: LoadOptions = {}): Promise<void> {\n try {\n // Check if file exists\n try {\n await fs.access(filePath);\n } catch {\n throw new DocxNotFoundError(filePath);\n }\n\n // Read file as buffer\n const buffer = await fs.readFile(filePath);\n await this.loadFromBuffer(buffer, options);\n } catch (error: unknown) {\n if (error instanceof DocxNotFoundError) {\n throw error;\n }\n const message = error instanceof Error ? error.message : String(error);\n throw new FileOperationError('read', message);\n }\n }\n\n /**\n * Loads a DOCX file from a buffer\n * @param buffer - Buffer containing the DOCX data\n * @param options - Load options\n */\n async loadFromBuffer(buffer: Buffer, options: LoadOptions = {}): Promise<void> {\n const { validate = true } = options;\n\n try {\n // Validate ZIP signature\n if (!isValidZipBuffer(buffer)) {\n throw new InvalidDocxError('File is not a valid ZIP archive');\n }\n\n // Load ZIP archive\n this.zip = await JSZip.loadAsync(buffer);\n\n // Extract all files\n await this.extractFiles();\n\n // Validate DOCX structure if requested\n if (validate) {\n this.validate();\n }\n\n this.loaded = true;\n } catch (error: unknown) {\n if (error instanceof InvalidDocxError) {\n throw error;\n }\n const message = error instanceof Error ? error.message : String(error);\n throw new CorruptedArchiveError(message);\n }\n }\n\n /**\n * Extracts all files from the ZIP archive into memory\n *\n * **Encoding Note:**\n * - Text files (XML, etc.) are extracted as UTF-8 strings using `async('string')`\n * - JSZip automatically decodes UTF-8 when extracting as 'string'\n * - Binary files are extracted as Buffers to preserve exact content\n * - All text content is guaranteed to be valid UTF-8\n */\n private async extractFiles(): Promise<void> {\n if (!this.zip) {\n throw new Error('ZIP archive not loaded');\n }\n\n this.files.clear();\n\n // Get all file paths\n const filePaths = Object.keys(this.zip.files).filter((path) => !this.zip!.files[path]!.dir);\n\n // Extract each file\n for (const filePath of filePaths) {\n const normalizedPath = normalizePath(filePath);\n const zipObject = this.zip.files[filePath];\n\n if (!zipObject) {\n continue;\n }\n\n const isBinary = isBinaryFile(normalizedPath);\n\n // Extract content based on type\n // For text files: JSZip's async('string') automatically uses UTF-8 decoding\n // For binary files: async('nodebuffer') preserves exact bytes\n let content;\n if (isBinary) {\n content = await zipObject.async('nodebuffer');\n } else {\n // Text files are extracted as UTF-8 strings\n // JSZip automatically handles UTF-8 decoding for 'string' type\n content = await zipObject.async('string');\n }\n\n // Get file metadata\n const date = zipObject.date;\n\n // Store file information\n this.files.set(normalizedPath, {\n path: normalizedPath,\n content,\n isBinary,\n size: isBinary ? (content as Buffer).length : (content as string).length,\n date,\n });\n }\n }\n\n /**\n * Validates the DOCX structure\n * @throws {MissingRequiredFileError} If required files are missing\n */\n private validate(): void {\n const filePaths = Array.from(this.files.keys());\n validateDocxStructure(filePaths);\n }\n\n /**\n * Gets a specific file from the archive\n * @param filePath - Path to the file within the archive\n * @returns The file data, or undefined if not found\n */\n getFile(filePath: string): ZipFile | undefined {\n this.ensureLoaded();\n const normalizedPath = normalizePath(filePath);\n return this.files.get(normalizedPath);\n }\n\n /**\n * Gets the content of a specific file as a string\n * @param filePath - Path to the file within the archive\n * @returns The file content as a UTF-8 string, or undefined if not found\n *\n * **Encoding Note:**\n * - Returns UTF-8 decoded string content\n * - For binary files, converts the Buffer to UTF-8 string\n * - Assumes all text content is UTF-8 encoded (per OpenXML standard)\n */\n getFileAsString(filePath: string): string | undefined {\n const file = this.getFile(filePath);\n if (!file) {\n return undefined;\n }\n\n // Check actual content type instead of flag (Issue #4)\n // Content is Buffer for binary files, string for text files\n if (Buffer.isBuffer(file.content)) {\n // Convert binary buffer to UTF-8 string\n return file.content.toString('utf8');\n }\n\n return file.content;\n }\n\n /**\n * Gets the content of a specific file as a buffer\n * @param filePath - Path to the file within the archive\n * @returns The file content as a Buffer, or undefined if not found\n *\n * **Encoding Note:**\n * - Returns Buffer with UTF-8 encoded content for text files\n * - For binary files, returns raw bytes\n * - String content is explicitly encoded as UTF-8\n */\n getFileAsBuffer(filePath: string): Buffer | undefined {\n const file = this.getFile(filePath);\n if (!file) {\n return undefined;\n }\n\n // Check actual content type instead of flag (Issue #4)\n // Content is Buffer for binary files, string for text files\n if (Buffer.isBuffer(file.content)) {\n return file.content;\n }\n\n // Encode string content as UTF-8 Buffer\n return Buffer.from(file.content, 'utf8');\n }\n\n /**\n * Gets all files from the archive\n * @returns Map of file paths to file data\n */\n getAllFiles(): FileMap {\n this.ensureLoaded();\n return new Map(this.files);\n }\n\n /**\n * Gets a list of all file paths in the archive\n * @returns Array of file paths\n */\n getFilePaths(): string[] {\n this.ensureLoaded();\n return Array.from(this.files.keys());\n }\n\n /**\n * Checks if a file exists in the archive\n * @param filePath - Path to check\n * @returns True if the file exists\n */\n hasFile(filePath: string): boolean {\n this.ensureLoaded();\n const normalizedPath = normalizePath(filePath);\n return this.files.has(normalizedPath);\n }\n\n /**\n * Gets files matching a pattern (simple glob)\n * @param pattern - Pattern to match (supports * wildcard)\n * @returns Array of matching files\n */\n getFilesByPattern(pattern: string): ZipFile[] {\n this.ensureLoaded();\n\n // Convert simple glob pattern to regex\n const regexPattern = pattern.replace(/\\*/g, '.*').replace(/\\?/g, '.');\n const regex = new RegExp(`^${regexPattern}$`);\n\n const matchingFiles: ZipFile[] = [];\n for (const [path, file] of this.files) {\n if (regex.test(path)) {\n matchingFiles.push(file);\n }\n }\n\n return matchingFiles;\n }\n\n /**\n * Ensures the archive is loaded before operations\n * @throws {Error} If archive is not loaded\n */\n private ensureLoaded(): void {\n if (!this.loaded) {\n throw new Error('Archive not loaded. Call loadFromFile() or loadFromBuffer() first.');\n }\n }\n\n /**\n * Checks if the archive is loaded\n * @returns True if loaded\n */\n isLoaded(): boolean {\n return this.loaded;\n }\n\n /**\n * Clears all loaded data\n */\n clear(): void {\n this.zip = null;\n this.files.clear();\n this.loaded = false;\n }\n}\n"]}