docxmlater 10.4.1 → 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.
- package/README.md +3 -3
- package/dist/constants/legacyCompatFlags.d.ts +1 -1
- package/dist/constants/legacyCompatFlags.d.ts.map +1 -1
- package/dist/constants/legacyCompatFlags.js.map +1 -1
- package/dist/core/Document.d.ts +74 -67
- package/dist/core/Document.d.ts.map +1 -1
- package/dist/core/Document.js +605 -414
- package/dist/core/Document.js.map +1 -1
- package/dist/core/DocumentContent.d.ts +11 -10
- package/dist/core/DocumentContent.d.ts.map +1 -1
- package/dist/core/DocumentContent.js +19 -19
- package/dist/core/DocumentContent.js.map +1 -1
- package/dist/core/DocumentEvents.d.ts +39 -0
- package/dist/core/DocumentEvents.d.ts.map +1 -0
- package/dist/core/DocumentEvents.js +51 -0
- package/dist/core/DocumentEvents.js.map +1 -0
- package/dist/core/DocumentGenerator.d.ts +11 -11
- package/dist/core/DocumentGenerator.d.ts.map +1 -1
- package/dist/core/DocumentGenerator.js +72 -52
- package/dist/core/DocumentGenerator.js.map +1 -1
- package/dist/core/DocumentParser.d.ts +15 -15
- package/dist/core/DocumentParser.d.ts.map +1 -1
- package/dist/core/DocumentParser.js +2056 -1073
- package/dist/core/DocumentParser.js.map +1 -1
- package/dist/core/DocumentValidator.d.ts +3 -3
- package/dist/core/DocumentValidator.d.ts.map +1 -1
- package/dist/core/DocumentValidator.js +31 -31
- package/dist/core/DocumentValidator.js.map +1 -1
- package/dist/core/ElementRegistry.d.ts +22 -0
- package/dist/core/ElementRegistry.d.ts.map +1 -0
- package/dist/core/ElementRegistry.js +27 -0
- package/dist/core/ElementRegistry.js.map +1 -0
- package/dist/core/Relationship.js +4 -4
- package/dist/core/Relationship.js.map +1 -1
- package/dist/core/RelationshipManager.d.ts +1 -1
- package/dist/core/RelationshipManager.d.ts.map +1 -1
- package/dist/core/RelationshipManager.js +32 -32
- package/dist/core/RelationshipManager.js.map +1 -1
- package/dist/elements/AlternateContent.d.ts +1 -1
- package/dist/elements/AlternateContent.d.ts.map +1 -1
- package/dist/elements/AlternateContent.js.map +1 -1
- package/dist/elements/Bookmark.d.ts +6 -1
- package/dist/elements/Bookmark.d.ts.map +1 -1
- package/dist/elements/Bookmark.js +19 -3
- package/dist/elements/Bookmark.js.map +1 -1
- package/dist/elements/BookmarkManager.d.ts +1 -1
- package/dist/elements/BookmarkManager.d.ts.map +1 -1
- package/dist/elements/BookmarkManager.js +7 -7
- package/dist/elements/BookmarkManager.js.map +1 -1
- package/dist/elements/Comment.d.ts +2 -2
- package/dist/elements/Comment.d.ts.map +1 -1
- package/dist/elements/Comment.js +4 -4
- package/dist/elements/Comment.js.map +1 -1
- package/dist/elements/CommentManager.d.ts +2 -2
- package/dist/elements/CommentManager.d.ts.map +1 -1
- package/dist/elements/CommentManager.js +9 -9
- package/dist/elements/CommentManager.js.map +1 -1
- package/dist/elements/CommonTypes.d.ts +9 -4
- package/dist/elements/CommonTypes.d.ts.map +1 -1
- package/dist/elements/CommonTypes.js +1 -0
- package/dist/elements/CommonTypes.js.map +1 -1
- package/dist/elements/CustomXml.d.ts +1 -1
- package/dist/elements/CustomXml.d.ts.map +1 -1
- package/dist/elements/CustomXml.js.map +1 -1
- package/dist/elements/Endnote.d.ts +2 -2
- package/dist/elements/Endnote.d.ts.map +1 -1
- package/dist/elements/Endnote.js +9 -9
- package/dist/elements/Endnote.js.map +1 -1
- package/dist/elements/EndnoteManager.d.ts +1 -1
- package/dist/elements/EndnoteManager.d.ts.map +1 -1
- package/dist/elements/EndnoteManager.js +11 -11
- package/dist/elements/EndnoteManager.js.map +1 -1
- package/dist/elements/Field.d.ts +9 -5
- package/dist/elements/Field.d.ts.map +1 -1
- package/dist/elements/Field.js +21 -9
- package/dist/elements/Field.js.map +1 -1
- package/dist/elements/FieldHelpers.d.ts +1 -1
- package/dist/elements/FieldHelpers.d.ts.map +1 -1
- package/dist/elements/FieldHelpers.js +10 -10
- package/dist/elements/FieldHelpers.js.map +1 -1
- package/dist/elements/Footer.d.ts +3 -3
- package/dist/elements/Footer.d.ts.map +1 -1
- package/dist/elements/Footer.js +5 -5
- package/dist/elements/Footer.js.map +1 -1
- package/dist/elements/Footnote.d.ts +2 -2
- package/dist/elements/Footnote.d.ts.map +1 -1
- package/dist/elements/Footnote.js +9 -9
- package/dist/elements/Footnote.js.map +1 -1
- package/dist/elements/FootnoteManager.d.ts +1 -1
- package/dist/elements/FootnoteManager.d.ts.map +1 -1
- package/dist/elements/FootnoteManager.js +11 -11
- package/dist/elements/FootnoteManager.js.map +1 -1
- package/dist/elements/Header.d.ts +3 -3
- package/dist/elements/Header.d.ts.map +1 -1
- package/dist/elements/Header.js +5 -5
- package/dist/elements/Header.js.map +1 -1
- package/dist/elements/HeaderFooterManager.d.ts +2 -2
- package/dist/elements/HeaderFooterManager.d.ts.map +1 -1
- package/dist/elements/HeaderFooterManager.js.map +1 -1
- package/dist/elements/Hyperlink.d.ts +5 -5
- package/dist/elements/Hyperlink.d.ts.map +1 -1
- package/dist/elements/Hyperlink.js +29 -29
- package/dist/elements/Hyperlink.js.map +1 -1
- package/dist/elements/Image.d.ts +1 -1
- package/dist/elements/Image.d.ts.map +1 -1
- package/dist/elements/Image.js +67 -67
- package/dist/elements/Image.js.map +1 -1
- package/dist/elements/ImageManager.d.ts +1 -1
- package/dist/elements/ImageManager.d.ts.map +1 -1
- package/dist/elements/ImageManager.js +4 -4
- package/dist/elements/ImageManager.js.map +1 -1
- package/dist/elements/ImageRun.d.ts +3 -3
- package/dist/elements/ImageRun.d.ts.map +1 -1
- package/dist/elements/ImageRun.js +2 -2
- package/dist/elements/ImageRun.js.map +1 -1
- package/dist/elements/MathElement.d.ts +1 -1
- package/dist/elements/MathElement.d.ts.map +1 -1
- package/dist/elements/MathElement.js.map +1 -1
- package/dist/elements/Paragraph.d.ts +34 -19
- package/dist/elements/Paragraph.d.ts.map +1 -1
- package/dist/elements/Paragraph.js +286 -231
- package/dist/elements/Paragraph.js.map +1 -1
- package/dist/elements/PreservedElement.d.ts +1 -1
- package/dist/elements/PreservedElement.d.ts.map +1 -1
- package/dist/elements/PreservedElement.js.map +1 -1
- package/dist/elements/PropertyChangeTypes.d.ts +2 -2
- package/dist/elements/PropertyChangeTypes.d.ts.map +1 -1
- package/dist/elements/PropertyChangeTypes.js.map +1 -1
- package/dist/elements/RangeMarker.d.ts +14 -1
- package/dist/elements/RangeMarker.d.ts.map +1 -1
- package/dist/elements/RangeMarker.js +46 -8
- package/dist/elements/RangeMarker.js.map +1 -1
- package/dist/elements/RegisteredBodyElement.d.ts +15 -0
- package/dist/elements/RegisteredBodyElement.d.ts.map +1 -0
- package/dist/elements/RegisteredBodyElement.js +44 -0
- package/dist/elements/RegisteredBodyElement.js.map +1 -0
- package/dist/elements/Revision.d.ts +8 -8
- package/dist/elements/Revision.d.ts.map +1 -1
- package/dist/elements/Revision.js +12 -12
- package/dist/elements/Revision.js.map +1 -1
- package/dist/elements/RevisionContent.d.ts +3 -3
- package/dist/elements/RevisionContent.d.ts.map +1 -1
- package/dist/elements/RevisionContent.js.map +1 -1
- package/dist/elements/RevisionManager.d.ts +2 -2
- package/dist/elements/RevisionManager.d.ts.map +1 -1
- package/dist/elements/RevisionManager.js +2 -2
- package/dist/elements/RevisionManager.js.map +1 -1
- package/dist/elements/Run.d.ts +16 -10
- package/dist/elements/Run.d.ts.map +1 -1
- package/dist/elements/Run.js +199 -173
- package/dist/elements/Run.js.map +1 -1
- package/dist/elements/Section.d.ts +4 -2
- package/dist/elements/Section.d.ts.map +1 -1
- package/dist/elements/Section.js +152 -145
- package/dist/elements/Section.js.map +1 -1
- package/dist/elements/Shape.d.ts +3 -3
- package/dist/elements/Shape.d.ts.map +1 -1
- package/dist/elements/Shape.js +12 -12
- package/dist/elements/Shape.js.map +1 -1
- package/dist/elements/StructuredDocumentTag.d.ts +3 -3
- package/dist/elements/StructuredDocumentTag.d.ts.map +1 -1
- package/dist/elements/StructuredDocumentTag.js +39 -39
- package/dist/elements/StructuredDocumentTag.js.map +1 -1
- package/dist/elements/Table.d.ts +16 -10
- package/dist/elements/Table.d.ts.map +1 -1
- package/dist/elements/Table.js +118 -89
- package/dist/elements/Table.js.map +1 -1
- package/dist/elements/TableCell.d.ts +11 -11
- package/dist/elements/TableCell.d.ts.map +1 -1
- package/dist/elements/TableCell.js +108 -78
- package/dist/elements/TableCell.js.map +1 -1
- package/dist/elements/TableGridChange.d.ts +1 -1
- package/dist/elements/TableGridChange.d.ts.map +1 -1
- package/dist/elements/TableGridChange.js +3 -3
- package/dist/elements/TableGridChange.js.map +1 -1
- package/dist/elements/TableOfContents.d.ts +1 -1
- package/dist/elements/TableOfContents.d.ts.map +1 -1
- package/dist/elements/TableOfContents.js +2 -2
- package/dist/elements/TableOfContents.js.map +1 -1
- package/dist/elements/TableOfContentsElement.d.ts +2 -2
- package/dist/elements/TableOfContentsElement.d.ts.map +1 -1
- package/dist/elements/TableOfContentsElement.js +5 -5
- package/dist/elements/TableOfContentsElement.js.map +1 -1
- package/dist/elements/TableRow.d.ts +18 -7
- package/dist/elements/TableRow.d.ts.map +1 -1
- package/dist/elements/TableRow.js +127 -74
- package/dist/elements/TableRow.js.map +1 -1
- package/dist/elements/TextBox.d.ts +4 -4
- package/dist/elements/TextBox.d.ts.map +1 -1
- package/dist/elements/TextBox.js +6 -6
- package/dist/elements/TextBox.js.map +1 -1
- package/dist/esm/constants/legacyCompatFlags.js +97 -0
- package/dist/esm/constants/legacyCompatFlags.js.map +1 -0
- package/dist/esm/constants/limits.js +36 -0
- package/dist/esm/constants/limits.js.map +1 -0
- package/dist/esm/core/Document.js +8498 -0
- package/dist/esm/core/Document.js.map +1 -0
- package/dist/esm/core/DocumentContent.js +190 -0
- package/dist/esm/core/DocumentContent.js.map +1 -0
- package/dist/esm/core/DocumentEvents.js +47 -0
- package/dist/esm/core/DocumentEvents.js.map +1 -0
- package/dist/esm/core/DocumentGenerator.js +764 -0
- package/dist/esm/core/DocumentGenerator.js.map +1 -0
- package/dist/esm/core/DocumentIdManager.js +67 -0
- package/dist/esm/core/DocumentIdManager.js.map +1 -0
- package/dist/esm/core/DocumentParser.js +8760 -0
- package/dist/esm/core/DocumentParser.js.map +1 -0
- package/dist/esm/core/DocumentValidator.js +222 -0
- package/dist/esm/core/DocumentValidator.js.map +1 -0
- package/dist/esm/core/ElementRegistry.js +24 -0
- package/dist/esm/core/ElementRegistry.js.map +1 -0
- package/dist/esm/core/Relationship.js +177 -0
- package/dist/esm/core/Relationship.js.map +1 -0
- package/dist/esm/core/RelationshipManager.js +202 -0
- package/dist/esm/core/RelationshipManager.js.map +1 -0
- package/dist/esm/elements/AlternateContent.js +19 -0
- package/dist/esm/elements/AlternateContent.js.map +1 -0
- package/dist/esm/elements/Bookmark.js +115 -0
- package/dist/esm/elements/Bookmark.js.map +1 -0
- package/dist/esm/elements/BookmarkManager.js +99 -0
- package/dist/esm/elements/BookmarkManager.js.map +1 -0
- package/dist/esm/elements/Comment.js +181 -0
- package/dist/esm/elements/Comment.js.map +1 -0
- package/dist/esm/elements/CommentManager.js +233 -0
- package/dist/esm/elements/CommentManager.js.map +1 -0
- package/dist/esm/elements/CommonTypes.js +106 -0
- package/dist/esm/elements/CommonTypes.js.map +1 -0
- package/dist/esm/elements/CustomXml.js +19 -0
- package/dist/esm/elements/CustomXml.js.map +1 -0
- package/dist/esm/elements/Endnote.js +107 -0
- package/dist/esm/elements/Endnote.js.map +1 -0
- package/dist/esm/elements/EndnoteManager.js +119 -0
- package/dist/esm/elements/EndnoteManager.js.map +1 -0
- package/dist/esm/elements/Field.js +856 -0
- package/dist/esm/elements/Field.js.map +1 -0
- package/dist/esm/elements/FieldHelpers.js +134 -0
- package/dist/esm/elements/FieldHelpers.js.map +1 -0
- package/dist/esm/elements/FontManager.js +158 -0
- package/dist/esm/elements/FontManager.js.map +1 -0
- package/dist/esm/elements/Footer.js +141 -0
- package/dist/esm/elements/Footer.js.map +1 -0
- package/dist/esm/elements/Footnote.js +107 -0
- package/dist/esm/elements/Footnote.js.map +1 -0
- package/dist/esm/elements/FootnoteManager.js +119 -0
- package/dist/esm/elements/FootnoteManager.js.map +1 -0
- package/dist/esm/elements/Header.js +141 -0
- package/dist/esm/elements/Header.js.map +1 -0
- package/dist/esm/elements/HeaderFooterManager.js +87 -0
- package/dist/esm/elements/HeaderFooterManager.js.map +1 -0
- package/dist/esm/elements/Hyperlink.js +586 -0
- package/dist/esm/elements/Hyperlink.js.map +1 -0
- package/dist/esm/elements/Image.js +1288 -0
- package/dist/esm/elements/Image.js.map +1 -0
- package/dist/esm/elements/ImageManager.js +223 -0
- package/dist/esm/elements/ImageManager.js.map +1 -0
- package/dist/esm/elements/ImageRun.js +29 -0
- package/dist/esm/elements/ImageRun.js.map +1 -0
- package/dist/esm/elements/MathElement.js +37 -0
- package/dist/esm/elements/MathElement.js.map +1 -0
- package/dist/esm/elements/Paragraph.js +2308 -0
- package/dist/esm/elements/Paragraph.js.map +1 -0
- package/dist/esm/elements/PreservedElement.js +29 -0
- package/dist/esm/elements/PreservedElement.js.map +1 -0
- package/dist/esm/elements/PropertyChangeTypes.js +53 -0
- package/dist/esm/elements/PropertyChangeTypes.js.map +1 -0
- package/dist/esm/elements/RangeMarker.js +219 -0
- package/dist/esm/elements/RangeMarker.js.map +1 -0
- package/dist/esm/elements/RegisteredBodyElement.js +40 -0
- package/dist/esm/elements/RegisteredBodyElement.js.map +1 -0
- package/dist/esm/elements/Revision.js +498 -0
- package/dist/esm/elements/Revision.js.map +1 -0
- package/dist/esm/elements/RevisionContent.js +18 -0
- package/dist/esm/elements/RevisionContent.js.map +1 -0
- package/dist/esm/elements/RevisionManager.js +486 -0
- package/dist/esm/elements/RevisionManager.js.map +1 -0
- package/dist/esm/elements/Run.js +1465 -0
- package/dist/esm/elements/Run.js.map +1 -0
- package/dist/esm/elements/Section.js +978 -0
- package/dist/esm/elements/Section.js.map +1 -0
- package/dist/esm/elements/Shape.js +493 -0
- package/dist/esm/elements/Shape.js.map +1 -0
- package/dist/esm/elements/StructuredDocumentTag.js +471 -0
- package/dist/esm/elements/StructuredDocumentTag.js.map +1 -0
- package/dist/esm/elements/Table.js +1456 -0
- package/dist/esm/elements/Table.js.map +1 -0
- package/dist/esm/elements/TableCell.js +835 -0
- package/dist/esm/elements/TableCell.js.map +1 -0
- package/dist/esm/elements/TableGridChange.js +52 -0
- package/dist/esm/elements/TableGridChange.js.map +1 -0
- package/dist/esm/elements/TableOfContents.js +389 -0
- package/dist/esm/elements/TableOfContents.js.map +1 -0
- package/dist/esm/elements/TableOfContentsElement.js +29 -0
- package/dist/esm/elements/TableOfContentsElement.js.map +1 -0
- package/dist/esm/elements/TableRow.js +555 -0
- package/dist/esm/elements/TableRow.js.map +1 -0
- package/dist/esm/elements/TextBox.js +459 -0
- package/dist/esm/elements/TextBox.js.map +1 -0
- package/dist/esm/formatting/AbstractNumbering.js +325 -0
- package/dist/esm/formatting/AbstractNumbering.js.map +1 -0
- package/dist/esm/formatting/NumberingInstance.js +150 -0
- package/dist/esm/formatting/NumberingInstance.js.map +1 -0
- package/dist/esm/formatting/NumberingLevel.js +608 -0
- package/dist/esm/formatting/NumberingLevel.js.map +1 -0
- package/dist/esm/formatting/NumberingManager.js +423 -0
- package/dist/esm/formatting/NumberingManager.js.map +1 -0
- package/dist/esm/formatting/Style.js +1151 -0
- package/dist/esm/formatting/Style.js.map +1 -0
- package/dist/esm/formatting/StylesManager.js +557 -0
- package/dist/esm/formatting/StylesManager.js.map +1 -0
- package/dist/esm/helpers/CleanupHelper.js +350 -0
- package/dist/esm/helpers/CleanupHelper.js.map +1 -0
- package/dist/esm/images/ImageOptimizer.js +161 -0
- package/dist/esm/images/ImageOptimizer.js.map +1 -0
- package/dist/esm/index.js +75 -0
- package/dist/esm/index.js.map +1 -0
- package/dist/esm/internal.js +16 -0
- package/dist/esm/internal.js.map +1 -0
- package/dist/esm/managers/DrawingManager.js +163 -0
- package/dist/esm/managers/DrawingManager.js.map +1 -0
- package/dist/esm/package.json +3 -0
- package/dist/esm/processors/ChangelogGenerator.js +970 -0
- package/dist/esm/processors/ChangelogGenerator.js.map +1 -0
- package/dist/esm/processors/CompatibilityUpgrader.js +130 -0
- package/dist/esm/processors/CompatibilityUpgrader.js.map +1 -0
- package/dist/esm/processors/InMemoryRevisionAcceptor.js +530 -0
- package/dist/esm/processors/InMemoryRevisionAcceptor.js.map +1 -0
- package/dist/esm/processors/MoveOperationHelper.js +57 -0
- package/dist/esm/processors/MoveOperationHelper.js.map +1 -0
- package/dist/esm/processors/RevisionAwareProcessor.js +232 -0
- package/dist/esm/processors/RevisionAwareProcessor.js.map +1 -0
- package/dist/esm/processors/RevisionWalker.js +278 -0
- package/dist/esm/processors/RevisionWalker.js.map +1 -0
- package/dist/{utils → esm/processors}/SelectiveRevisionAcceptor.js +81 -42
- package/dist/esm/processors/SelectiveRevisionAcceptor.js.map +1 -0
- package/dist/esm/processors/ShadingResolver.js +66 -0
- package/dist/esm/processors/ShadingResolver.js.map +1 -0
- package/dist/esm/processors/acceptRevisions.js +416 -0
- package/dist/esm/processors/acceptRevisions.js.map +1 -0
- package/dist/esm/processors/cnfStyleDecoder.js +89 -0
- package/dist/esm/processors/cnfStyleDecoder.js.map +1 -0
- package/dist/esm/processors/stripTrackedChanges.js +201 -0
- package/dist/esm/processors/stripTrackedChanges.js.map +1 -0
- package/dist/esm/tracking/DocumentTrackingContext.js +531 -0
- package/dist/esm/tracking/DocumentTrackingContext.js.map +1 -0
- package/dist/esm/tracking/TrackingContext.js +2 -0
- package/dist/esm/tracking/TrackingContext.js.map +1 -0
- package/dist/esm/types/compatibility-types.js +8 -0
- package/dist/esm/types/compatibility-types.js.map +1 -0
- package/dist/esm/types/document-types.js +2 -0
- package/dist/esm/types/document-types.js.map +1 -0
- package/dist/esm/types/formatting.js +2 -0
- package/dist/esm/types/formatting.js.map +1 -0
- package/dist/esm/types/list-types.js +2 -0
- package/dist/esm/types/list-types.js.map +1 -0
- package/dist/esm/types/settings-types.js +2 -0
- package/dist/esm/types/settings-types.js.map +1 -0
- package/dist/esm/types/styleConfig.js +2 -0
- package/dist/esm/types/styleConfig.js.map +1 -0
- package/dist/esm/utils/KeyedRegistry.js +32 -0
- package/dist/esm/utils/KeyedRegistry.js.map +1 -0
- package/dist/esm/utils/corruptionDetection.js +155 -0
- package/dist/esm/utils/corruptionDetection.js.map +1 -0
- package/dist/esm/utils/dateFormatting.js +4 -0
- package/dist/esm/utils/dateFormatting.js.map +1 -0
- package/dist/esm/utils/deepClone.js +40 -0
- package/dist/esm/utils/deepClone.js.map +1 -0
- package/dist/esm/utils/deepEqual.js +47 -0
- package/dist/esm/utils/deepEqual.js.map +1 -0
- package/dist/esm/utils/diagnostics.js +69 -0
- package/dist/esm/utils/diagnostics.js.map +1 -0
- package/dist/esm/utils/errorHandling.js +36 -0
- package/dist/esm/utils/errorHandling.js.map +1 -0
- package/dist/esm/utils/formatting.js +93 -0
- package/dist/esm/utils/formatting.js.map +1 -0
- package/dist/esm/utils/list-detection.js +148 -0
- package/dist/esm/utils/list-detection.js.map +1 -0
- package/dist/esm/utils/logger.js +205 -0
- package/dist/esm/utils/logger.js.map +1 -0
- package/dist/esm/utils/parsingHelpers.js +56 -0
- package/dist/esm/utils/parsingHelpers.js.map +1 -0
- package/dist/esm/utils/textDiff.js +42 -0
- package/dist/esm/utils/textDiff.js.map +1 -0
- package/dist/esm/utils/units.js +152 -0
- package/dist/esm/utils/units.js.map +1 -0
- package/dist/esm/utils/validation.js +285 -0
- package/dist/esm/utils/validation.js.map +1 -0
- package/dist/esm/utils/xmlSanitization.js +54 -0
- package/dist/esm/utils/xmlSanitization.js.map +1 -0
- package/dist/esm/validation/RevisionAutoFixer.js +340 -0
- package/dist/esm/validation/RevisionAutoFixer.js.map +1 -0
- package/dist/esm/validation/RevisionValidator.js +240 -0
- package/dist/esm/validation/RevisionValidator.js.map +1 -0
- package/dist/esm/validation/ValidationRuleRegistry.js +40 -0
- package/dist/esm/validation/ValidationRuleRegistry.js.map +1 -0
- package/dist/esm/validation/ValidationRules.js +92 -0
- package/dist/esm/validation/ValidationRules.js.map +1 -0
- package/dist/esm/validation/index.js +4 -0
- package/dist/esm/validation/index.js.map +1 -0
- package/dist/esm/xml/XMLBuilder.js +434 -0
- package/dist/esm/xml/XMLBuilder.js.map +1 -0
- package/dist/esm/xml/XMLParser.js +486 -0
- package/dist/esm/xml/XMLParser.js.map +1 -0
- package/dist/esm/zip/ZipHandler.js +298 -0
- package/dist/esm/zip/ZipHandler.js.map +1 -0
- package/dist/esm/zip/ZipReader.js +147 -0
- package/dist/esm/zip/ZipReader.js.map +1 -0
- package/dist/esm/zip/ZipWriter.js +199 -0
- package/dist/esm/zip/ZipWriter.js.map +1 -0
- package/dist/esm/zip/errors.js +43 -0
- package/dist/esm/zip/errors.js.map +1 -0
- package/dist/esm/zip/types.js +31 -0
- package/dist/esm/zip/types.js.map +1 -0
- package/dist/formatting/AbstractNumbering.d.ts +2 -2
- package/dist/formatting/AbstractNumbering.d.ts.map +1 -1
- package/dist/formatting/AbstractNumbering.js +33 -33
- package/dist/formatting/AbstractNumbering.js.map +1 -1
- package/dist/formatting/NumberingInstance.d.ts +2 -2
- package/dist/formatting/NumberingInstance.d.ts.map +1 -1
- package/dist/formatting/NumberingInstance.js +7 -7
- package/dist/formatting/NumberingInstance.js.map +1 -1
- package/dist/formatting/NumberingLevel.d.ts +11 -2
- package/dist/formatting/NumberingLevel.d.ts.map +1 -1
- package/dist/formatting/NumberingLevel.js +111 -25
- package/dist/formatting/NumberingLevel.js.map +1 -1
- package/dist/formatting/NumberingManager.d.ts +4 -4
- package/dist/formatting/NumberingManager.d.ts.map +1 -1
- package/dist/formatting/NumberingManager.js +28 -28
- package/dist/formatting/NumberingManager.js.map +1 -1
- package/dist/formatting/Style.d.ts +14 -7
- package/dist/formatting/Style.d.ts.map +1 -1
- package/dist/formatting/Style.js +309 -112
- package/dist/formatting/Style.js.map +1 -1
- package/dist/formatting/StylesManager.d.ts +2 -2
- package/dist/formatting/StylesManager.d.ts.map +1 -1
- package/dist/formatting/StylesManager.js +52 -52
- package/dist/formatting/StylesManager.js.map +1 -1
- package/dist/helpers/CleanupHelper.d.ts +1 -1
- package/dist/helpers/CleanupHelper.d.ts.map +1 -1
- package/dist/helpers/CleanupHelper.js +15 -15
- package/dist/helpers/CleanupHelper.js.map +1 -1
- package/dist/index.d.ts +81 -90
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +286 -317
- package/dist/index.js.map +1 -1
- package/dist/internal.d.ts +16 -0
- package/dist/internal.d.ts.map +1 -0
- package/dist/internal.js +42 -0
- package/dist/internal.js.map +1 -0
- package/dist/managers/DrawingManager.d.ts +3 -3
- package/dist/managers/DrawingManager.d.ts.map +1 -1
- package/dist/managers/DrawingManager.js +12 -12
- package/dist/managers/DrawingManager.js.map +1 -1
- package/dist/{utils → processors}/ChangelogGenerator.d.ts +2 -2
- package/dist/processors/ChangelogGenerator.d.ts.map +1 -0
- package/dist/{utils → processors}/ChangelogGenerator.js +2 -2
- package/dist/processors/ChangelogGenerator.js.map +1 -0
- package/dist/processors/CompatibilityUpgrader.d.ts.map +1 -0
- package/dist/{utils → processors}/CompatibilityUpgrader.js +10 -10
- package/dist/processors/CompatibilityUpgrader.js.map +1 -0
- package/dist/{utils → processors}/InMemoryRevisionAcceptor.d.ts +3 -3
- package/dist/processors/InMemoryRevisionAcceptor.d.ts.map +1 -0
- package/dist/{utils → processors}/InMemoryRevisionAcceptor.js +84 -27
- package/dist/processors/InMemoryRevisionAcceptor.js.map +1 -0
- package/dist/{utils → processors}/MoveOperationHelper.d.ts +4 -4
- package/dist/processors/MoveOperationHelper.d.ts.map +1 -0
- package/dist/{utils → processors}/MoveOperationHelper.js +10 -10
- package/dist/processors/MoveOperationHelper.js.map +1 -0
- package/dist/{utils → processors}/RevisionAwareProcessor.d.ts +3 -3
- package/dist/processors/RevisionAwareProcessor.d.ts.map +1 -0
- package/dist/{utils → processors}/RevisionAwareProcessor.js +2 -2
- package/dist/processors/RevisionAwareProcessor.js.map +1 -0
- package/dist/{utils → processors}/RevisionWalker.d.ts +2 -1
- package/dist/processors/RevisionWalker.d.ts.map +1 -0
- package/dist/{utils → processors}/RevisionWalker.js +28 -0
- package/dist/processors/RevisionWalker.js.map +1 -0
- package/dist/{utils → processors}/SelectiveRevisionAcceptor.d.ts +4 -3
- package/dist/processors/SelectiveRevisionAcceptor.d.ts.map +1 -0
- package/dist/processors/SelectiveRevisionAcceptor.js +402 -0
- package/dist/processors/SelectiveRevisionAcceptor.js.map +1 -0
- package/dist/processors/ShadingResolver.d.ts +6 -0
- package/dist/processors/ShadingResolver.d.ts.map +1 -0
- package/dist/{utils → processors}/ShadingResolver.js +2 -2
- package/dist/processors/ShadingResolver.js.map +1 -0
- package/dist/{utils → processors}/acceptRevisions.d.ts +1 -1
- package/dist/processors/acceptRevisions.d.ts.map +1 -0
- package/dist/{utils → processors}/acceptRevisions.js +24 -4
- package/dist/processors/acceptRevisions.js.map +1 -0
- package/dist/{utils → processors}/cnfStyleDecoder.d.ts +1 -1
- package/dist/processors/cnfStyleDecoder.d.ts.map +1 -0
- package/dist/processors/cnfStyleDecoder.js.map +1 -0
- package/dist/processors/stripTrackedChanges.d.ts +3 -0
- package/dist/processors/stripTrackedChanges.d.ts.map +1 -0
- package/dist/{utils → processors}/stripTrackedChanges.js +16 -6
- package/dist/processors/stripTrackedChanges.js.map +1 -0
- package/dist/tracking/DocumentTrackingContext.d.ts +4 -4
- package/dist/tracking/DocumentTrackingContext.d.ts.map +1 -1
- package/dist/tracking/DocumentTrackingContext.js +38 -43
- package/dist/tracking/DocumentTrackingContext.js.map +1 -1
- package/dist/tracking/TrackingContext.d.ts +8 -8
- package/dist/tracking/TrackingContext.d.ts.map +1 -1
- package/dist/tracking/TrackingContext.js.map +1 -1
- package/dist/types/document-types.d.ts +28 -0
- package/dist/types/document-types.d.ts.map +1 -0
- package/dist/types/document-types.js +3 -0
- package/dist/types/document-types.js.map +1 -0
- package/dist/types/formatting.d.ts +4 -4
- package/dist/types/formatting.d.ts.map +1 -1
- package/dist/types/formatting.js.map +1 -1
- package/dist/types/settings-types.d.ts +6 -0
- package/dist/types/settings-types.d.ts.map +1 -1
- package/dist/types/settings-types.js.map +1 -1
- package/dist/utils/KeyedRegistry.d.ts +13 -0
- package/dist/utils/KeyedRegistry.d.ts.map +1 -0
- package/dist/utils/KeyedRegistry.js +36 -0
- package/dist/utils/KeyedRegistry.js.map +1 -0
- package/dist/utils/corruptionDetection.d.ts +1 -1
- package/dist/utils/corruptionDetection.d.ts.map +1 -1
- package/dist/utils/corruptionDetection.js +4 -4
- package/dist/utils/corruptionDetection.js.map +1 -1
- package/dist/utils/deepEqual.d.ts +2 -0
- package/dist/utils/deepEqual.d.ts.map +1 -0
- package/dist/utils/deepEqual.js +50 -0
- package/dist/utils/deepEqual.js.map +1 -0
- package/dist/utils/list-detection.d.ts +2 -2
- package/dist/utils/list-detection.d.ts.map +1 -1
- package/dist/utils/list-detection.js.map +1 -1
- package/dist/utils/parsingHelpers.d.ts +1 -1
- package/dist/utils/parsingHelpers.d.ts.map +1 -1
- package/dist/utils/parsingHelpers.js +2 -2
- package/dist/utils/parsingHelpers.js.map +1 -1
- package/dist/utils/validation.js +7 -7
- package/dist/utils/validation.js.map +1 -1
- package/dist/utils/xmlSanitization.js +2 -2
- package/dist/utils/xmlSanitization.js.map +1 -1
- package/dist/validation/RevisionAutoFixer.d.ts +4 -4
- package/dist/validation/RevisionAutoFixer.d.ts.map +1 -1
- package/dist/validation/RevisionAutoFixer.js +11 -11
- package/dist/validation/RevisionAutoFixer.js.map +1 -1
- package/dist/validation/RevisionValidator.d.ts +5 -4
- package/dist/validation/RevisionValidator.d.ts.map +1 -1
- package/dist/validation/RevisionValidator.js +29 -30
- package/dist/validation/RevisionValidator.js.map +1 -1
- package/dist/validation/ValidationRuleRegistry.d.ts +27 -0
- package/dist/validation/ValidationRuleRegistry.d.ts.map +1 -0
- package/dist/validation/ValidationRuleRegistry.js +43 -0
- package/dist/validation/ValidationRuleRegistry.js.map +1 -0
- package/dist/validation/index.d.ts +3 -3
- package/dist/validation/index.d.ts.map +1 -1
- package/dist/validation/index.js +10 -10
- package/dist/validation/index.js.map +1 -1
- package/dist/xml/XMLBuilder.d.ts +6 -1
- package/dist/xml/XMLBuilder.d.ts.map +1 -1
- package/dist/xml/XMLBuilder.js +11 -6
- package/dist/xml/XMLBuilder.js.map +1 -1
- package/dist/xml/XMLParser.js +6 -6
- package/dist/xml/XMLParser.js.map +1 -1
- package/dist/zip/ZipHandler.d.ts +1 -1
- package/dist/zip/ZipHandler.d.ts.map +1 -1
- package/dist/zip/ZipHandler.js +8 -8
- package/dist/zip/ZipHandler.js.map +1 -1
- package/dist/zip/ZipReader.d.ts +1 -1
- package/dist/zip/ZipReader.d.ts.map +1 -1
- package/dist/zip/ZipReader.js +14 -14
- package/dist/zip/ZipReader.js.map +1 -1
- package/dist/zip/ZipWriter.d.ts +1 -1
- package/dist/zip/ZipWriter.d.ts.map +1 -1
- package/dist/zip/ZipWriter.js +10 -10
- package/dist/zip/ZipWriter.js.map +1 -1
- package/package.json +20 -4
- package/src/constants/legacyCompatFlags.ts +1 -1
- package/src/core/Document.ts +461 -167
- package/src/core/DocumentContent.ts +14 -11
- package/src/core/DocumentEvents.ts +90 -0
- package/src/core/DocumentGenerator.ts +49 -22
- package/src/core/DocumentParser.ts +2180 -617
- package/src/core/DocumentValidator.ts +7 -7
- package/src/core/ElementRegistry.ts +69 -0
- package/src/core/Relationship.ts +1 -1
- package/src/core/RelationshipManager.ts +4 -4
- package/src/elements/AlternateContent.ts +1 -1
- package/src/elements/Bookmark.ts +52 -4
- package/src/elements/BookmarkManager.ts +2 -2
- package/src/elements/Comment.ts +3 -3
- package/src/elements/CommentManager.ts +4 -4
- package/src/elements/CommonTypes.ts +45 -7
- package/src/elements/CustomXml.ts +1 -1
- package/src/elements/Endnote.ts +2 -2
- package/src/elements/EndnoteManager.ts +3 -3
- package/src/elements/Field.ts +44 -10
- package/src/elements/FieldHelpers.ts +2 -2
- package/src/elements/Footer.ts +4 -4
- package/src/elements/Footnote.ts +2 -2
- package/src/elements/FootnoteManager.ts +3 -3
- package/src/elements/Header.ts +4 -4
- package/src/elements/HeaderFooterManager.ts +2 -2
- package/src/elements/Hyperlink.ts +16 -12
- package/src/elements/Image.ts +3 -3
- package/src/elements/ImageManager.ts +2 -2
- package/src/elements/ImageRun.ts +3 -3
- package/src/elements/MathElement.ts +1 -1
- package/src/elements/Paragraph.ts +221 -88
- package/src/elements/PreservedElement.ts +1 -1
- package/src/elements/PropertyChangeTypes.ts +2 -2
- package/src/elements/RangeMarker.ts +153 -12
- package/src/elements/RegisteredBodyElement.ts +52 -0
- package/src/elements/Revision.ts +14 -14
- package/src/elements/RevisionContent.ts +3 -3
- package/src/elements/RevisionManager.ts +3 -3
- package/src/elements/Run.ts +221 -94
- package/src/elements/Section.ts +136 -69
- package/src/elements/Shape.ts +4 -4
- package/src/elements/StructuredDocumentTag.ts +3 -3
- package/src/elements/Table.ts +91 -27
- package/src/elements/TableCell.ts +62 -34
- package/src/elements/TableGridChange.ts +1 -1
- package/src/elements/TableOfContents.ts +1 -1
- package/src/elements/TableOfContentsElement.ts +2 -2
- package/src/elements/TableRow.ts +192 -48
- package/src/elements/TextBox.ts +5 -5
- package/src/formatting/AbstractNumbering.ts +3 -3
- package/src/formatting/NumberingInstance.ts +2 -2
- package/src/formatting/NumberingLevel.ts +201 -10
- package/src/formatting/NumberingManager.ts +5 -5
- package/src/formatting/Style.ts +382 -86
- package/src/formatting/StylesManager.ts +4 -4
- package/src/helpers/CleanupHelper.ts +6 -6
- package/src/index.ts +118 -127
- package/src/internal.ts +79 -0
- package/src/managers/DrawingManager.ts +3 -3
- package/src/{utils → processors}/ChangelogGenerator.ts +3 -3
- package/src/{utils → processors}/CompatibilityUpgrader.ts +2 -2
- package/src/{utils → processors}/InMemoryRevisionAcceptor.ts +100 -12
- package/src/{utils → processors}/MoveOperationHelper.ts +5 -5
- package/src/{utils → processors}/RevisionAwareProcessor.ts +3 -3
- package/src/{utils → processors}/RevisionWalker.ts +42 -1
- package/src/{utils → processors}/SelectiveRevisionAcceptor.ts +98 -39
- package/src/{utils → processors}/ShadingResolver.ts +5 -5
- package/src/{utils → processors}/acceptRevisions.ts +77 -9
- package/src/{utils → processors}/cnfStyleDecoder.ts +1 -1
- package/src/{utils → processors}/stripTrackedChanges.ts +35 -10
- package/src/tracking/DocumentTrackingContext.ts +12 -14
- package/src/tracking/TrackingContext.ts +8 -8
- package/src/types/document-types.ts +53 -0
- package/src/types/formatting.ts +4 -4
- package/src/types/settings-types.ts +32 -0
- package/src/utils/KeyedRegistry.ts +41 -0
- package/src/utils/corruptionDetection.ts +2 -2
- package/src/utils/deepEqual.ts +58 -0
- package/src/utils/list-detection.ts +2 -2
- package/src/utils/parsingHelpers.ts +11 -3
- package/src/utils/validation.ts +3 -3
- package/src/utils/xmlSanitization.ts +1 -1
- package/src/validation/RevisionAutoFixer.ts +5 -5
- package/src/validation/RevisionValidator.ts +39 -28
- package/src/validation/ValidationRuleRegistry.ts +86 -0
- package/src/validation/index.ts +3 -3
- package/src/xml/XMLBuilder.ts +13 -3
- package/src/xml/XMLParser.ts +2 -2
- package/src/zip/ZipHandler.ts +4 -4
- package/src/zip/ZipReader.ts +3 -3
- package/src/zip/ZipWriter.ts +3 -3
- package/dist/utils/ChangelogGenerator.d.ts.map +0 -1
- package/dist/utils/ChangelogGenerator.js.map +0 -1
- package/dist/utils/CompatibilityUpgrader.d.ts.map +0 -1
- package/dist/utils/CompatibilityUpgrader.js.map +0 -1
- package/dist/utils/InMemoryRevisionAcceptor.d.ts.map +0 -1
- package/dist/utils/InMemoryRevisionAcceptor.js.map +0 -1
- package/dist/utils/MoveOperationHelper.d.ts.map +0 -1
- package/dist/utils/MoveOperationHelper.js.map +0 -1
- package/dist/utils/RevisionAwareProcessor.d.ts.map +0 -1
- package/dist/utils/RevisionAwareProcessor.js.map +0 -1
- package/dist/utils/RevisionWalker.d.ts.map +0 -1
- package/dist/utils/RevisionWalker.js.map +0 -1
- package/dist/utils/SelectiveRevisionAcceptor.d.ts.map +0 -1
- package/dist/utils/SelectiveRevisionAcceptor.js.map +0 -1
- package/dist/utils/ShadingResolver.d.ts +0 -6
- package/dist/utils/ShadingResolver.d.ts.map +0 -1
- package/dist/utils/ShadingResolver.js.map +0 -1
- package/dist/utils/acceptRevisions.d.ts.map +0 -1
- package/dist/utils/acceptRevisions.js.map +0 -1
- package/dist/utils/cnfStyleDecoder.d.ts.map +0 -1
- package/dist/utils/cnfStyleDecoder.js.map +0 -1
- package/dist/utils/stripTrackedChanges.d.ts +0 -3
- package/dist/utils/stripTrackedChanges.d.ts.map +0 -1
- package/dist/utils/stripTrackedChanges.js.map +0 -1
- package/src/__tests__/helper-methods.test.ts +0 -512
- package/src/constants/CLAUDE.md +0 -28
- package/src/core/CLAUDE.md +0 -113
- package/src/elements/CLAUDE.md +0 -142
- package/src/formatting/CLAUDE.md +0 -78
- package/src/managers/CLAUDE.md +0 -47
- package/src/tracking/CLAUDE.md +0 -30
- package/src/types/CLAUDE.md +0 -39
- package/src/utils/CLAUDE.md +0 -168
- package/src/validation/CLAUDE.md +0 -40
- package/src/xml/CLAUDE.md +0 -65
- package/src/zip/CLAUDE.md +0 -55
- /package/dist/{utils → processors}/CompatibilityUpgrader.d.ts +0 -0
- /package/dist/{utils → processors}/cnfStyleDecoder.js +0 -0
|
@@ -0,0 +1,486 @@
|
|
|
1
|
+
import { getGlobalLogger, createScopedLogger } from '../utils/logger.js';
|
|
2
|
+
import { XMLBuilder } from './XMLBuilder.js';
|
|
3
|
+
function getLogger() {
|
|
4
|
+
return createScopedLogger(getGlobalLogger(), 'XMLParser');
|
|
5
|
+
}
|
|
6
|
+
export const DEFAULT_MAX_NESTING_DEPTH = 256;
|
|
7
|
+
export class XMLParser {
|
|
8
|
+
static extractBody(docXml) {
|
|
9
|
+
const startTag = '<w:body';
|
|
10
|
+
const endTag = '</w:body>';
|
|
11
|
+
const startIdx = docXml.indexOf(startTag);
|
|
12
|
+
if (startIdx === -1)
|
|
13
|
+
return '';
|
|
14
|
+
const openEnd = docXml.indexOf('>', startIdx);
|
|
15
|
+
if (openEnd === -1)
|
|
16
|
+
return '';
|
|
17
|
+
const endIdx = docXml.indexOf(endTag, openEnd);
|
|
18
|
+
if (endIdx === -1)
|
|
19
|
+
return '';
|
|
20
|
+
return docXml.substring(openEnd + 1, endIdx);
|
|
21
|
+
}
|
|
22
|
+
static extractElements(xml, tagName) {
|
|
23
|
+
const elements = [];
|
|
24
|
+
const openTag = `<${tagName}`;
|
|
25
|
+
const closeTag = `</${tagName}>`;
|
|
26
|
+
const selfClosingEnd = '/>';
|
|
27
|
+
let pos = 0;
|
|
28
|
+
while (pos < xml.length) {
|
|
29
|
+
const startIdx = xml.indexOf(openTag, pos);
|
|
30
|
+
if (startIdx === -1)
|
|
31
|
+
break;
|
|
32
|
+
const charAfterTag = xml[startIdx + openTag.length];
|
|
33
|
+
if (charAfterTag &&
|
|
34
|
+
charAfterTag !== '>' &&
|
|
35
|
+
charAfterTag !== '/' &&
|
|
36
|
+
charAfterTag !== ' ' &&
|
|
37
|
+
charAfterTag !== '\t' &&
|
|
38
|
+
charAfterTag !== '\n' &&
|
|
39
|
+
charAfterTag !== '\r' &&
|
|
40
|
+
charAfterTag !== '=') {
|
|
41
|
+
pos = startIdx + openTag.length;
|
|
42
|
+
continue;
|
|
43
|
+
}
|
|
44
|
+
const openEnd = xml.indexOf('>', startIdx);
|
|
45
|
+
if (openEnd === -1)
|
|
46
|
+
break;
|
|
47
|
+
if (xml.substring(openEnd - 1, openEnd + 1) === selfClosingEnd) {
|
|
48
|
+
elements.push(xml.substring(startIdx, openEnd + 1));
|
|
49
|
+
pos = openEnd + 1;
|
|
50
|
+
continue;
|
|
51
|
+
}
|
|
52
|
+
let depth = 1;
|
|
53
|
+
let searchPos = openEnd + 1;
|
|
54
|
+
while (depth > 0 && searchPos < xml.length) {
|
|
55
|
+
let nextOpen = -1;
|
|
56
|
+
let openSearchPos = searchPos;
|
|
57
|
+
while (true) {
|
|
58
|
+
const candidateOpen = xml.indexOf(openTag, openSearchPos);
|
|
59
|
+
if (candidateOpen === -1) {
|
|
60
|
+
break;
|
|
61
|
+
}
|
|
62
|
+
const charAfter = xml[candidateOpen + openTag.length];
|
|
63
|
+
if (charAfter &&
|
|
64
|
+
charAfter !== '>' &&
|
|
65
|
+
charAfter !== '/' &&
|
|
66
|
+
charAfter !== ' ' &&
|
|
67
|
+
charAfter !== '\t' &&
|
|
68
|
+
charAfter !== '\n' &&
|
|
69
|
+
charAfter !== '\r') {
|
|
70
|
+
openSearchPos = candidateOpen + openTag.length;
|
|
71
|
+
continue;
|
|
72
|
+
}
|
|
73
|
+
nextOpen = candidateOpen;
|
|
74
|
+
break;
|
|
75
|
+
}
|
|
76
|
+
const nextClose = xml.indexOf(closeTag, searchPos);
|
|
77
|
+
if (nextClose === -1)
|
|
78
|
+
break;
|
|
79
|
+
if (nextOpen !== -1 && nextOpen < nextClose) {
|
|
80
|
+
depth++;
|
|
81
|
+
searchPos = nextOpen + openTag.length;
|
|
82
|
+
}
|
|
83
|
+
else {
|
|
84
|
+
depth--;
|
|
85
|
+
if (depth === 0) {
|
|
86
|
+
elements.push(xml.substring(startIdx, nextClose + closeTag.length));
|
|
87
|
+
pos = nextClose + closeTag.length;
|
|
88
|
+
}
|
|
89
|
+
else {
|
|
90
|
+
searchPos = nextClose + closeTag.length;
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
if (depth > 0) {
|
|
95
|
+
pos = startIdx + openTag.length;
|
|
96
|
+
}
|
|
97
|
+
}
|
|
98
|
+
return elements;
|
|
99
|
+
}
|
|
100
|
+
static extractAttribute(xml, attributeName) {
|
|
101
|
+
const attrPattern = `${attributeName}="`;
|
|
102
|
+
const startIdx = xml.indexOf(attrPattern);
|
|
103
|
+
if (startIdx === -1)
|
|
104
|
+
return undefined;
|
|
105
|
+
const valueStart = startIdx + attrPattern.length;
|
|
106
|
+
const valueEnd = xml.indexOf('"', valueStart);
|
|
107
|
+
if (valueEnd === -1)
|
|
108
|
+
return undefined;
|
|
109
|
+
const rawValue = xml.substring(valueStart, valueEnd);
|
|
110
|
+
return XMLBuilder.unescapeXml(rawValue);
|
|
111
|
+
}
|
|
112
|
+
static hasSelfClosingTag(xml, tagName) {
|
|
113
|
+
return xml.includes(`<${tagName}/>`) || xml.includes(`<${tagName} `);
|
|
114
|
+
}
|
|
115
|
+
static hasBooleanProperty(xml, tagName) {
|
|
116
|
+
if (xml.includes(`<${tagName} w:val="1"`) || xml.includes(`<${tagName} w:val="true"`)) {
|
|
117
|
+
return true;
|
|
118
|
+
}
|
|
119
|
+
if (xml.includes(`<${tagName}/>`)) {
|
|
120
|
+
return true;
|
|
121
|
+
}
|
|
122
|
+
return false;
|
|
123
|
+
}
|
|
124
|
+
static extractText(xml) {
|
|
125
|
+
const texts = [];
|
|
126
|
+
const openTag = '<w:t';
|
|
127
|
+
const closeTag = '</w:t>';
|
|
128
|
+
let pos = 0;
|
|
129
|
+
while (pos < xml.length) {
|
|
130
|
+
const startIdx = xml.indexOf(openTag, pos);
|
|
131
|
+
if (startIdx === -1)
|
|
132
|
+
break;
|
|
133
|
+
const openEnd = xml.indexOf('>', startIdx);
|
|
134
|
+
if (openEnd === -1)
|
|
135
|
+
break;
|
|
136
|
+
const closeIdx = xml.indexOf(closeTag, openEnd);
|
|
137
|
+
if (closeIdx === -1)
|
|
138
|
+
break;
|
|
139
|
+
const text = xml.substring(openEnd + 1, closeIdx);
|
|
140
|
+
texts.push(text);
|
|
141
|
+
pos = closeIdx + closeTag.length;
|
|
142
|
+
}
|
|
143
|
+
return texts.join('');
|
|
144
|
+
}
|
|
145
|
+
static validateSize(xml, maxSize = 10 * 1024 * 1024) {
|
|
146
|
+
if (xml.length > maxSize) {
|
|
147
|
+
throw new Error(`XML content too large for parsing (${(xml.length / 1024 / 1024).toFixed(1)}MB). ` +
|
|
148
|
+
`Maximum allowed: ${(maxSize / 1024 / 1024).toFixed(0)}MB`);
|
|
149
|
+
}
|
|
150
|
+
}
|
|
151
|
+
static extractBetweenTags(xml, startTag, endTag) {
|
|
152
|
+
const startIdx = xml.indexOf(startTag);
|
|
153
|
+
if (startIdx === -1)
|
|
154
|
+
return undefined;
|
|
155
|
+
const openEnd = xml.indexOf('>', startIdx);
|
|
156
|
+
if (openEnd === -1)
|
|
157
|
+
return undefined;
|
|
158
|
+
const endIdx = xml.indexOf(endTag, openEnd);
|
|
159
|
+
if (endIdx === -1)
|
|
160
|
+
return undefined;
|
|
161
|
+
return xml.substring(openEnd + 1, endIdx);
|
|
162
|
+
}
|
|
163
|
+
static extractSelfClosingTag(xml, tagName) {
|
|
164
|
+
const startPattern = `<${tagName}`;
|
|
165
|
+
let searchPos = 0;
|
|
166
|
+
while (true) {
|
|
167
|
+
const startIdx = xml.indexOf(startPattern, searchPos);
|
|
168
|
+
if (startIdx === -1)
|
|
169
|
+
return undefined;
|
|
170
|
+
const charAfterTag = xml[startIdx + startPattern.length];
|
|
171
|
+
if (charAfterTag === ' ' || charAfterTag === '/' || charAfterTag === '>') {
|
|
172
|
+
const endIdx = xml.indexOf('/>', startIdx);
|
|
173
|
+
if (endIdx === -1) {
|
|
174
|
+
const closeTagStart = xml.indexOf('>', startIdx);
|
|
175
|
+
if (closeTagStart === -1)
|
|
176
|
+
return undefined;
|
|
177
|
+
return xml.substring(startIdx + startPattern.length, closeTagStart);
|
|
178
|
+
}
|
|
179
|
+
return xml.substring(startIdx + startPattern.length, endIdx);
|
|
180
|
+
}
|
|
181
|
+
searchPos = startIdx + 1;
|
|
182
|
+
}
|
|
183
|
+
}
|
|
184
|
+
static parseToObject(xml, options) {
|
|
185
|
+
const logger = getLogger();
|
|
186
|
+
logger.debug('Parsing XML to object', { xmlSize: xml.length });
|
|
187
|
+
const opts = {
|
|
188
|
+
ignoreAttributes: options?.ignoreAttributes ?? false,
|
|
189
|
+
attributeNamePrefix: options?.attributeNamePrefix ?? '@_',
|
|
190
|
+
textNodeName: options?.textNodeName ?? '#text',
|
|
191
|
+
ignoreNamespace: options?.ignoreNamespace ?? false,
|
|
192
|
+
parseAttributeValue: options?.parseAttributeValue ?? true,
|
|
193
|
+
trimValues: options?.trimValues ?? true,
|
|
194
|
+
alwaysArray: options?.alwaysArray ?? false,
|
|
195
|
+
maxNestingDepth: options?.maxNestingDepth ?? DEFAULT_MAX_NESTING_DEPTH,
|
|
196
|
+
};
|
|
197
|
+
XMLParser.validateSize(xml);
|
|
198
|
+
xml = xml.replace(/<\?xml[^>]*\?>\s*/g, '').trim();
|
|
199
|
+
if (!xml) {
|
|
200
|
+
return {};
|
|
201
|
+
}
|
|
202
|
+
const result = XMLParser.parseElementToObject(xml, 0, opts, 0);
|
|
203
|
+
logger.debug('XML parsed to object');
|
|
204
|
+
return result.value;
|
|
205
|
+
}
|
|
206
|
+
static parseElementToObject(xml, startPos, options, depth) {
|
|
207
|
+
if (depth > options.maxNestingDepth) {
|
|
208
|
+
throw new Error(`XML nesting depth exceeds maximum of ${options.maxNestingDepth}. ` +
|
|
209
|
+
`This may indicate malformed XML or an attack attempt. ` +
|
|
210
|
+
`Use the maxNestingDepth option to increase the limit if needed.`);
|
|
211
|
+
}
|
|
212
|
+
const openTagStart = xml.indexOf('<', startPos);
|
|
213
|
+
if (openTagStart === -1) {
|
|
214
|
+
return { value: {}, endPos: xml.length };
|
|
215
|
+
}
|
|
216
|
+
if (xml.substring(openTagStart, openTagStart + 4) === '<!--') {
|
|
217
|
+
const commentEnd = xml.indexOf('-->', openTagStart + 4);
|
|
218
|
+
if (commentEnd !== -1) {
|
|
219
|
+
return XMLParser.parseElementToObject(xml, commentEnd + 3, options, depth);
|
|
220
|
+
}
|
|
221
|
+
return { value: {}, endPos: xml.length };
|
|
222
|
+
}
|
|
223
|
+
const nameMatch = /^([a-zA-Z0-9:_-]+)/.exec(xml.substring(openTagStart + 1));
|
|
224
|
+
if (!nameMatch) {
|
|
225
|
+
return { value: {}, endPos: openTagStart + 1 };
|
|
226
|
+
}
|
|
227
|
+
const originalElementName = nameMatch[1] || '';
|
|
228
|
+
let elementName = originalElementName;
|
|
229
|
+
const tagHeaderEnd = xml.indexOf('>', openTagStart);
|
|
230
|
+
if (tagHeaderEnd === -1) {
|
|
231
|
+
return { value: {}, endPos: xml.length };
|
|
232
|
+
}
|
|
233
|
+
if (options.ignoreNamespace && elementName.includes(':')) {
|
|
234
|
+
elementName = elementName.split(':')[1] || elementName;
|
|
235
|
+
}
|
|
236
|
+
const tagHeader = xml.substring(openTagStart + 1 + originalElementName.length, tagHeaderEnd);
|
|
237
|
+
const attributes = XMLParser.extractAttributesFromTag(tagHeader, options);
|
|
238
|
+
const isSelfClosing = tagHeader.trim().endsWith('/') || xml[tagHeaderEnd - 1] === '/';
|
|
239
|
+
if (isSelfClosing) {
|
|
240
|
+
const elementValue = { ...attributes };
|
|
241
|
+
return {
|
|
242
|
+
value: { [elementName]: elementValue },
|
|
243
|
+
endPos: tagHeaderEnd + 1,
|
|
244
|
+
};
|
|
245
|
+
}
|
|
246
|
+
const closingTag = `</${originalElementName}>`;
|
|
247
|
+
const contentStart = tagHeaderEnd + 1;
|
|
248
|
+
const closingTagPos = XMLParser.findClosingTag(xml, originalElementName, contentStart);
|
|
249
|
+
if (closingTagPos === -1) {
|
|
250
|
+
return {
|
|
251
|
+
value: { [elementName]: { ...attributes } },
|
|
252
|
+
endPos: tagHeaderEnd + 1,
|
|
253
|
+
};
|
|
254
|
+
}
|
|
255
|
+
const content = xml.substring(contentStart, closingTagPos);
|
|
256
|
+
const children = [];
|
|
257
|
+
let textContent = '';
|
|
258
|
+
let pos = 0;
|
|
259
|
+
while (pos < content.length) {
|
|
260
|
+
const nextTag = content.indexOf('<', pos);
|
|
261
|
+
if (nextTag === -1) {
|
|
262
|
+
const text = content.substring(pos);
|
|
263
|
+
if (text.length > 0 && (!options.trimValues || text.trim())) {
|
|
264
|
+
textContent += XMLBuilder.unescapeXml(text);
|
|
265
|
+
}
|
|
266
|
+
break;
|
|
267
|
+
}
|
|
268
|
+
if (nextTag > pos) {
|
|
269
|
+
const text = content.substring(pos, nextTag);
|
|
270
|
+
if (text.length > 0 && (!options.trimValues || text.trim())) {
|
|
271
|
+
textContent += XMLBuilder.unescapeXml(text);
|
|
272
|
+
}
|
|
273
|
+
}
|
|
274
|
+
const childResult = XMLParser.parseElementToObject(content, nextTag, options, depth + 1);
|
|
275
|
+
const childObj = childResult.value;
|
|
276
|
+
const childKeys = Object.keys(childObj);
|
|
277
|
+
if (childKeys.length > 0) {
|
|
278
|
+
const childName = childKeys[0];
|
|
279
|
+
if (childName) {
|
|
280
|
+
const childValue = childObj[childName];
|
|
281
|
+
children.push({ name: childName, value: childValue });
|
|
282
|
+
}
|
|
283
|
+
}
|
|
284
|
+
pos = childResult.endPos;
|
|
285
|
+
}
|
|
286
|
+
let elementValue = {};
|
|
287
|
+
if (!options.ignoreAttributes && Object.keys(attributes).length > 0) {
|
|
288
|
+
elementValue = { ...attributes };
|
|
289
|
+
}
|
|
290
|
+
if (textContent.length > 0 && (!options.trimValues || textContent.trim())) {
|
|
291
|
+
const text = options.trimValues ? textContent.trim() : textContent;
|
|
292
|
+
if (typeof elementValue === 'object' && !Array.isArray(elementValue)) {
|
|
293
|
+
if (Object.keys(elementValue).length === 0) {
|
|
294
|
+
elementValue = text;
|
|
295
|
+
}
|
|
296
|
+
else {
|
|
297
|
+
elementValue[options.textNodeName] = text;
|
|
298
|
+
}
|
|
299
|
+
}
|
|
300
|
+
}
|
|
301
|
+
if (children.length > 0) {
|
|
302
|
+
const coalescedChildren = XMLParser.coalesceChildren(children, options);
|
|
303
|
+
if (typeof elementValue === 'object' && !Array.isArray(elementValue)) {
|
|
304
|
+
elementValue = { ...elementValue, ...coalescedChildren };
|
|
305
|
+
}
|
|
306
|
+
else {
|
|
307
|
+
elementValue = coalescedChildren;
|
|
308
|
+
}
|
|
309
|
+
}
|
|
310
|
+
if (typeof elementValue === 'object' &&
|
|
311
|
+
!Array.isArray(elementValue) &&
|
|
312
|
+
Object.keys(elementValue).length === 0) {
|
|
313
|
+
elementValue = {};
|
|
314
|
+
}
|
|
315
|
+
return {
|
|
316
|
+
value: { [elementName]: elementValue },
|
|
317
|
+
endPos: closingTagPos + closingTag.length,
|
|
318
|
+
};
|
|
319
|
+
}
|
|
320
|
+
static extractAttributesFromTag(tagHeader, options) {
|
|
321
|
+
const attributes = {};
|
|
322
|
+
if (options.ignoreAttributes) {
|
|
323
|
+
return attributes;
|
|
324
|
+
}
|
|
325
|
+
let pos = 0;
|
|
326
|
+
while (pos < tagHeader.length) {
|
|
327
|
+
while (pos < tagHeader.length) {
|
|
328
|
+
const char = tagHeader[pos];
|
|
329
|
+
if (char && /\s/.test(char)) {
|
|
330
|
+
pos++;
|
|
331
|
+
}
|
|
332
|
+
else {
|
|
333
|
+
break;
|
|
334
|
+
}
|
|
335
|
+
}
|
|
336
|
+
if (pos >= tagHeader.length || tagHeader[pos] === '/') {
|
|
337
|
+
break;
|
|
338
|
+
}
|
|
339
|
+
const nameStart = pos;
|
|
340
|
+
while (pos < tagHeader.length) {
|
|
341
|
+
const char = tagHeader[pos];
|
|
342
|
+
if (char && /[a-zA-Z0-9:_-]/.test(char)) {
|
|
343
|
+
pos++;
|
|
344
|
+
}
|
|
345
|
+
else {
|
|
346
|
+
break;
|
|
347
|
+
}
|
|
348
|
+
}
|
|
349
|
+
if (pos === nameStart) {
|
|
350
|
+
break;
|
|
351
|
+
}
|
|
352
|
+
let attrName = tagHeader.substring(nameStart, pos);
|
|
353
|
+
while (pos < tagHeader.length) {
|
|
354
|
+
const char = tagHeader[pos];
|
|
355
|
+
if (char && /[\s=]/.test(char)) {
|
|
356
|
+
pos++;
|
|
357
|
+
}
|
|
358
|
+
else {
|
|
359
|
+
break;
|
|
360
|
+
}
|
|
361
|
+
}
|
|
362
|
+
let attrValue = '';
|
|
363
|
+
if (pos < tagHeader.length && (tagHeader[pos] === '"' || tagHeader[pos] === "'")) {
|
|
364
|
+
const quote = tagHeader[pos];
|
|
365
|
+
pos++;
|
|
366
|
+
const valueStart = pos;
|
|
367
|
+
while (pos < tagHeader.length && tagHeader[pos] !== quote) {
|
|
368
|
+
pos++;
|
|
369
|
+
}
|
|
370
|
+
attrValue = tagHeader.substring(valueStart, pos);
|
|
371
|
+
pos++;
|
|
372
|
+
}
|
|
373
|
+
if (options.ignoreNamespace && attrName.includes(':')) {
|
|
374
|
+
attrName = attrName.split(':')[1] || attrName;
|
|
375
|
+
}
|
|
376
|
+
const prefixedName = options.attributeNamePrefix + attrName;
|
|
377
|
+
attributes[prefixedName] = options.parseAttributeValue
|
|
378
|
+
? XMLParser.parseValue(attrValue)
|
|
379
|
+
: attrValue;
|
|
380
|
+
}
|
|
381
|
+
return attributes;
|
|
382
|
+
}
|
|
383
|
+
static findClosingTag(xml, elementName, startPos) {
|
|
384
|
+
const openTag = `<${elementName}`;
|
|
385
|
+
const closeTag = `</${elementName}>`;
|
|
386
|
+
let depth = 1;
|
|
387
|
+
let pos = startPos;
|
|
388
|
+
while (depth > 0 && pos < xml.length) {
|
|
389
|
+
const nextClose = xml.indexOf(closeTag, pos);
|
|
390
|
+
if (nextClose === -1) {
|
|
391
|
+
return -1;
|
|
392
|
+
}
|
|
393
|
+
let realOpenPos = -1;
|
|
394
|
+
let searchPos = pos;
|
|
395
|
+
while (searchPos < nextClose) {
|
|
396
|
+
const candidateOpen = xml.indexOf(openTag, searchPos);
|
|
397
|
+
if (candidateOpen === -1 || candidateOpen >= nextClose) {
|
|
398
|
+
break;
|
|
399
|
+
}
|
|
400
|
+
const charAfter = xml[candidateOpen + openTag.length];
|
|
401
|
+
if (charAfter === '>' ||
|
|
402
|
+
charAfter === ' ' ||
|
|
403
|
+
charAfter === '/' ||
|
|
404
|
+
charAfter === '\t' ||
|
|
405
|
+
charAfter === '\n' ||
|
|
406
|
+
charAfter === '\r') {
|
|
407
|
+
const tagEnd = xml.indexOf('>', candidateOpen);
|
|
408
|
+
if (tagEnd !== -1 && xml[tagEnd - 1] === '/') {
|
|
409
|
+
searchPos = tagEnd + 1;
|
|
410
|
+
continue;
|
|
411
|
+
}
|
|
412
|
+
realOpenPos = candidateOpen;
|
|
413
|
+
break;
|
|
414
|
+
}
|
|
415
|
+
searchPos = candidateOpen + openTag.length;
|
|
416
|
+
}
|
|
417
|
+
if (realOpenPos !== -1) {
|
|
418
|
+
depth++;
|
|
419
|
+
pos = realOpenPos + openTag.length;
|
|
420
|
+
}
|
|
421
|
+
else {
|
|
422
|
+
depth--;
|
|
423
|
+
if (depth === 0) {
|
|
424
|
+
return nextClose;
|
|
425
|
+
}
|
|
426
|
+
pos = nextClose + closeTag.length;
|
|
427
|
+
}
|
|
428
|
+
}
|
|
429
|
+
return -1;
|
|
430
|
+
}
|
|
431
|
+
static coalesceChildren(children, options) {
|
|
432
|
+
const result = {};
|
|
433
|
+
const nameCounts = {};
|
|
434
|
+
const nameIndices = {};
|
|
435
|
+
const orderedChildren = [];
|
|
436
|
+
for (const child of children) {
|
|
437
|
+
nameCounts[child.name] = (nameCounts[child.name] || 0) + 1;
|
|
438
|
+
}
|
|
439
|
+
for (const child of children) {
|
|
440
|
+
const shouldBeArray = options.alwaysArray || (nameCounts[child.name] || 0) > 1;
|
|
441
|
+
const currentIndex = nameIndices[child.name] || 0;
|
|
442
|
+
orderedChildren.push({ type: child.name, index: currentIndex });
|
|
443
|
+
nameIndices[child.name] = currentIndex + 1;
|
|
444
|
+
if (shouldBeArray) {
|
|
445
|
+
if (!result[child.name]) {
|
|
446
|
+
result[child.name] = [];
|
|
447
|
+
}
|
|
448
|
+
result[child.name].push(child.value);
|
|
449
|
+
}
|
|
450
|
+
else {
|
|
451
|
+
result[child.name] = child.value;
|
|
452
|
+
}
|
|
453
|
+
}
|
|
454
|
+
if (orderedChildren.length > 0) {
|
|
455
|
+
result._orderedChildren = orderedChildren;
|
|
456
|
+
}
|
|
457
|
+
return result;
|
|
458
|
+
}
|
|
459
|
+
static parseValue(value) {
|
|
460
|
+
if (value === 'true')
|
|
461
|
+
return true;
|
|
462
|
+
if (value === 'false')
|
|
463
|
+
return false;
|
|
464
|
+
if (/^[0-9A-Fa-f]{6}$/.test(value)) {
|
|
465
|
+
return value.toUpperCase();
|
|
466
|
+
}
|
|
467
|
+
if (/^\d{7,}$/.test(value)) {
|
|
468
|
+
return value;
|
|
469
|
+
}
|
|
470
|
+
if (/^-?\d+$/.test(value)) {
|
|
471
|
+
const num = parseInt(value, 10);
|
|
472
|
+
if (!isNaN(num))
|
|
473
|
+
return num;
|
|
474
|
+
}
|
|
475
|
+
if (/^-?\d+\.\d+$/.test(value)) {
|
|
476
|
+
const num = parseFloat(value);
|
|
477
|
+
if (!isNaN(num))
|
|
478
|
+
return num;
|
|
479
|
+
}
|
|
480
|
+
if (/^[0-9A-Fa-f]{3}$/.test(value) && /[A-Fa-f]/.test(value)) {
|
|
481
|
+
return value.toUpperCase();
|
|
482
|
+
}
|
|
483
|
+
return value;
|
|
484
|
+
}
|
|
485
|
+
}
|
|
486
|
+
//# sourceMappingURL=XMLParser.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"XMLParser.js","sourceRoot":"","sources":["../../../src/xml/XMLParser.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * XMLParser - Simple position-based XML parser\n * Avoids regex backtracking issues that can cause ReDoS attacks\n * Completes the DocXML framework (XMLBuilder + XMLParser)\n */\n\nimport { getGlobalLogger, createScopedLogger, ILogger } from '../utils/logger.js';\nimport { XMLBuilder } from './XMLBuilder.js';\n\n// Create scoped logger for XMLParser operations\nfunction getLogger(): ILogger {\n return createScopedLogger(getGlobalLogger(), 'XMLParser');\n}\n\n/**\n * Default maximum nesting depth for XML parsing.\n * Prevents stack overflow on deeply nested documents.\n */\nexport const DEFAULT_MAX_NESTING_DEPTH = 256;\n\n/**\n * Options for XML-to-object parsing\n */\nexport interface ParseToObjectOptions {\n /** Ignore attributes (default: false) */\n ignoreAttributes?: boolean;\n\n /** Attribute name prefix (default: '@_') */\n attributeNamePrefix?: string;\n\n /** Text node property name (default: '#text') */\n textNodeName?: string;\n\n /** Remove namespace prefixes from element names (default: false) */\n ignoreNamespace?: boolean;\n\n /** Parse numeric attribute values (default: true) */\n parseAttributeValue?: boolean;\n\n /** Trim whitespace from text values (default: true) */\n trimValues?: boolean;\n\n /** Always return arrays for elements (default: false) */\n alwaysArray?: boolean;\n\n /** Maximum nesting depth (default: 256). Prevents stack overflow on deeply nested documents. */\n maxNestingDepth?: number;\n}\n\n/**\n * Parsed XML object structure\n * Can be a string, object, array, or nested structure\n */\nexport type ParsedXMLValue =\n | string\n | number\n | boolean\n | ParsedXMLObject\n | ParsedXMLObject[]\n | null\n | undefined;\n\n/**\n * Parsed XML object with dynamic keys\n */\nexport interface ParsedXMLObject {\n [key: string]: ParsedXMLValue;\n}\n\n/**\n * Internal structure for tracking parsed elements during parsing\n */\ninterface ParsedElement {\n name: string;\n value: ParsedXMLValue;\n}\n\n/**\n * Simple XML parser using position-based parsing instead of regex\n * Prevents catastrophic backtracking (ReDoS attacks) by avoiding nested regex patterns\n */\nexport class XMLParser {\n /**\n * Extracts the body content from a Word document XML\n * @param docXml - The complete document.xml content\n * @returns The body content, or empty string if not found\n */\n static extractBody(docXml: string): string {\n const startTag = '<w:body';\n const endTag = '</w:body>';\n\n const startIdx = docXml.indexOf(startTag);\n if (startIdx === -1) return '';\n\n // Find the closing > of opening tag\n const openEnd = docXml.indexOf('>', startIdx);\n if (openEnd === -1) return '';\n\n // Find matching closing tag\n const endIdx = docXml.indexOf(endTag, openEnd);\n if (endIdx === -1) return '';\n\n return docXml.substring(openEnd + 1, endIdx);\n }\n\n /**\n * Extracts all elements of a given type using position-based parsing\n * Handles nested tags correctly by tracking depth\n * @param xml - XML content to parse\n * @param tagName - Tag name to extract (e.g., 'w:p', 'w:r')\n * @returns Array of XML strings for each element\n */\n static extractElements(xml: string, tagName: string): string[] {\n const elements: string[] = [];\n const openTag = `<${tagName}`;\n const closeTag = `</${tagName}>`;\n const selfClosingEnd = '/>';\n\n let pos = 0;\n while (pos < xml.length) {\n const startIdx = xml.indexOf(openTag, pos);\n if (startIdx === -1) break;\n\n // Verify this is the exact tag (not a prefix match like <w:p matching <w:pPr>)\n // The character after the tag name must be either '>', '/', whitespace, or '=' (for attributes)\n const charAfterTag = xml[startIdx + openTag.length];\n if (\n charAfterTag &&\n charAfterTag !== '>' &&\n charAfterTag !== '/' &&\n charAfterTag !== ' ' &&\n charAfterTag !== '\\t' &&\n charAfterTag !== '\\n' &&\n charAfterTag !== '\\r' &&\n charAfterTag !== '='\n ) {\n // This is a prefix match (e.g., <w:pPr> when looking for <w:p>), skip it (Issue #5)\n pos = startIdx + openTag.length;\n continue;\n }\n\n // Find the end of opening tag\n const openEnd = xml.indexOf('>', startIdx);\n if (openEnd === -1) break;\n\n // Check if self-closing\n if (xml.substring(openEnd - 1, openEnd + 1) === selfClosingEnd) {\n elements.push(xml.substring(startIdx, openEnd + 1));\n pos = openEnd + 1;\n continue;\n }\n\n // Find matching closing tag (handle nesting)\n let depth = 1;\n let searchPos = openEnd + 1;\n\n while (depth > 0 && searchPos < xml.length) {\n // Find next potential opening tag\n let nextOpen = -1;\n let openSearchPos = searchPos;\n while (true) {\n const candidateOpen = xml.indexOf(openTag, openSearchPos);\n if (candidateOpen === -1) {\n break;\n }\n // Verify it's an exact match (not a prefix)\n const charAfter = xml[candidateOpen + openTag.length];\n if (\n charAfter &&\n charAfter !== '>' &&\n charAfter !== '/' &&\n charAfter !== ' ' &&\n charAfter !== '\\t' &&\n charAfter !== '\\n' &&\n charAfter !== '\\r'\n ) {\n // Prefix match, keep searching\n openSearchPos = candidateOpen + openTag.length;\n continue;\n }\n nextOpen = candidateOpen;\n break;\n }\n\n const nextClose = xml.indexOf(closeTag, searchPos);\n\n if (nextClose === -1) break;\n\n if (nextOpen !== -1 && nextOpen < nextClose) {\n depth++;\n searchPos = nextOpen + openTag.length;\n } else {\n depth--;\n if (depth === 0) {\n elements.push(xml.substring(startIdx, nextClose + closeTag.length));\n pos = nextClose + closeTag.length;\n } else {\n searchPos = nextClose + closeTag.length;\n }\n }\n }\n\n if (depth > 0) {\n // Unclosed tag - skip it\n pos = startIdx + openTag.length;\n }\n }\n\n return elements;\n }\n\n /**\n * Extracts attribute value from an XML string\n * @param xml - XML content\n * @param attributeName - Attribute name (e.g., 'w:val')\n * @returns Attribute value or undefined\n */\n static extractAttribute(xml: string, attributeName: string): string | undefined {\n // Use simple indexOf for bounded string search (safe)\n const attrPattern = `${attributeName}=\"`;\n const startIdx = xml.indexOf(attrPattern);\n if (startIdx === -1) return undefined;\n\n const valueStart = startIdx + attrPattern.length;\n const valueEnd = xml.indexOf('\"', valueStart);\n if (valueEnd === -1) return undefined;\n\n const rawValue = xml.substring(valueStart, valueEnd);\n // Unescape XML entities to get the actual value\n // This prevents double-escaping when the value is later re-serialized\n return XMLBuilder.unescapeXml(rawValue);\n }\n\n /**\n * Checks if an XML string contains a self-closing tag\n * @param xml - XML content\n * @param tagName - Tag name to check\n * @returns True if the tag exists as self-closing\n */\n static hasSelfClosingTag(xml: string, tagName: string): boolean {\n return xml.includes(`<${tagName}/>`) || xml.includes(`<${tagName} `);\n }\n\n /**\n * Checks if a boolean property tag is enabled (w:val=\"1\" or w:val=\"true\")\n * Per ECMA-376, boolean properties can be:\n * - Present with w:val=\"1\" or w:val=\"true\" (enabled)\n * - Present with w:val=\"0\" or w:val=\"false\" (explicitly disabled)\n * - Absent (disabled by default)\n *\n * @param xml - XML content to search\n * @param tagName - Tag name (e.g., 'w:keepNext')\n * @returns True if tag exists with w:val=\"1\" or w:val=\"true\", false otherwise\n *\n * @example\n * hasBooleanProperty('<w:pPr><w:keepNext w:val=\"1\"/></w:pPr>', 'w:keepNext'); // true\n * hasBooleanProperty('<w:pPr><w:keepNext w:val=\"0\"/></w:pPr>', 'w:keepNext'); // false\n * hasBooleanProperty('<w:pPr><w:spacing/></w:pPr>', 'w:keepNext'); // false\n */\n static hasBooleanProperty(xml: string, tagName: string): boolean {\n // Check for tag with w:val=\"1\" or w:val=\"true\"\n if (xml.includes(`<${tagName} w:val=\"1\"`) || xml.includes(`<${tagName} w:val=\"true\"`)) {\n return true;\n }\n\n // Check for self-closing tag without w:val attribute (means true per ECMA-376)\n // Example: <w:b/> means bold=true\n if (xml.includes(`<${tagName}/>`)) {\n return true;\n }\n\n return false;\n }\n\n /**\n * Extracts text content from within tags\n * Finds all <w:t>...</w:t> tags and extracts their text\n * @param xml - XML content\n * @returns Combined text content\n */\n static extractText(xml: string): string {\n const texts: string[] = [];\n const openTag = '<w:t';\n const closeTag = '</w:t>';\n\n let pos = 0;\n while (pos < xml.length) {\n const startIdx = xml.indexOf(openTag, pos);\n if (startIdx === -1) break;\n\n // Find the end of opening tag\n const openEnd = xml.indexOf('>', startIdx);\n if (openEnd === -1) break;\n\n // Find closing tag\n const closeIdx = xml.indexOf(closeTag, openEnd);\n if (closeIdx === -1) break;\n\n // Extract text between tags\n const text = xml.substring(openEnd + 1, closeIdx);\n texts.push(text);\n\n pos = closeIdx + closeTag.length;\n }\n\n return texts.join('');\n }\n\n /**\n * Validates input size to prevent excessive memory usage\n * @param xml - XML content\n * @param maxSize - Maximum size in bytes (default: 10MB)\n * @throws Error if XML exceeds max size\n */\n static validateSize(xml: string, maxSize: number = 10 * 1024 * 1024): void {\n if (xml.length > maxSize) {\n throw new Error(\n `XML content too large for parsing (${(xml.length / 1024 / 1024).toFixed(1)}MB). ` +\n `Maximum allowed: ${(maxSize / 1024 / 1024).toFixed(0)}MB`\n );\n }\n }\n\n /**\n * Extracts content between two specific tags\n * More efficient than regex for large documents\n * @param xml - XML content\n * @param startTag - Opening tag (e.g., '<w:pPr')\n * @param endTag - Closing tag (e.g., '</w:pPr>')\n * @returns Content between tags, or undefined if not found\n */\n static extractBetweenTags(xml: string, startTag: string, endTag: string): string | undefined {\n const startIdx = xml.indexOf(startTag);\n if (startIdx === -1) return undefined;\n\n // Find the end of the opening tag\n const openEnd = xml.indexOf('>', startIdx);\n if (openEnd === -1) return undefined;\n\n // Find the closing tag\n const endIdx = xml.indexOf(endTag, openEnd);\n if (endIdx === -1) return undefined;\n\n return xml.substring(openEnd + 1, endIdx);\n }\n\n /**\n * Extracts a complete self-closing tag with its attributes\n * Handles cases where multiple similar tags exist (e.g., <w:sz.../> and <w:szCs.../>)\n *\n * @param xml - XML string to search\n * @param tagName - Tag name to find (e.g., \"w:color\", \"w:sz\")\n * @returns The complete tag content (attributes portion) or undefined if not found\n *\n * @example\n * const xml = '<w:sz w:val=\"36\"/><w:color w:val=\"FF0000\"/>';\n * const colorTag = XMLParser.extractSelfClosingTag(xml, 'w:color');\n * // Returns: ' w:val=\"FF0000\"'\n */\n static extractSelfClosingTag(xml: string, tagName: string): string | undefined {\n const startPattern = `<${tagName}`;\n let searchPos = 0;\n\n // Search for the exact tag (not tags that start with this pattern)\n while (true) {\n const startIdx = xml.indexOf(startPattern, searchPos);\n if (startIdx === -1) return undefined;\n\n // Check what character follows the tag name\n const charAfterTag = xml[startIdx + startPattern.length];\n\n // Valid separators after tag name: space, '/', or '>'\n if (charAfterTag === ' ' || charAfterTag === '/' || charAfterTag === '>') {\n // Found the exact tag, now find its end\n const endIdx = xml.indexOf('/>', startIdx);\n if (endIdx === -1) {\n // Try finding a closing tag instead (non-self-closing)\n const closeTagStart = xml.indexOf('>', startIdx);\n if (closeTagStart === -1) return undefined;\n\n // Return attributes portion\n return xml.substring(startIdx + startPattern.length, closeTagStart);\n }\n\n // Return attributes portion (between tag name and />)\n return xml.substring(startIdx + startPattern.length, endIdx);\n }\n\n // Not the exact tag (e.g., found \"w:sz\" when looking for \"w:s\")\n // Continue searching\n searchPos = startIdx + 1;\n }\n }\n\n /**\n * Parse XML string to JavaScript object\n * Compatible with fast-xml-parser output format\n *\n * @param xml - XML string to parse\n * @param options - Parsing options\n * @returns Parsed JavaScript object\n *\n * @example\n * const xml = '<Relationships><Relationship Id=\"rId1\" Target=\"https://example.com\"/></Relationships>';\n * const obj = XMLParser.parseToObject(xml);\n * // Returns: { Relationships: { Relationship: { '@_Id': 'rId1', '@_Target': 'https://example.com' } } }\n *\n * @example\n * // Multiple elements become arrays\n * const xml = '<Items><Item id=\"1\"/><Item id=\"2\"/></Items>';\n * const obj = XMLParser.parseToObject(xml);\n * // Returns: { Items: { Item: [{ '@_id': '1' }, { '@_id': '2' }] } }\n */\n static parseToObject(xml: string, options?: ParseToObjectOptions): ParsedXMLObject {\n const logger = getLogger();\n logger.debug('Parsing XML to object', { xmlSize: xml.length });\n\n // Default options\n const opts: Required<ParseToObjectOptions> = {\n ignoreAttributes: options?.ignoreAttributes ?? false,\n attributeNamePrefix: options?.attributeNamePrefix ?? '@_',\n textNodeName: options?.textNodeName ?? '#text',\n ignoreNamespace: options?.ignoreNamespace ?? false,\n parseAttributeValue: options?.parseAttributeValue ?? true,\n trimValues: options?.trimValues ?? true,\n alwaysArray: options?.alwaysArray ?? false,\n maxNestingDepth: options?.maxNestingDepth ?? DEFAULT_MAX_NESTING_DEPTH,\n };\n\n // Validate input size\n XMLParser.validateSize(xml);\n\n // Remove XML declaration and trim\n xml = xml.replace(/<\\?xml[^>]*\\?>\\s*/g, '').trim();\n\n if (!xml) {\n return {};\n }\n\n // Parse root element (start at depth 0)\n const result = XMLParser.parseElementToObject(xml, 0, opts, 0);\n logger.debug('XML parsed to object');\n return result.value as ParsedXMLObject;\n }\n\n /**\n * Parses a single XML element into an object\n * @private\n */\n private static parseElementToObject(\n xml: string,\n startPos: number,\n options: Required<ParseToObjectOptions>,\n depth: number\n ): { value: ParsedXMLValue; endPos: number } {\n // Check nesting depth to prevent stack overflow\n if (depth > options.maxNestingDepth) {\n throw new Error(\n `XML nesting depth exceeds maximum of ${options.maxNestingDepth}. ` +\n `This may indicate malformed XML or an attack attempt. ` +\n `Use the maxNestingDepth option to increase the limit if needed.`\n );\n }\n\n // Find opening tag\n const openTagStart = xml.indexOf('<', startPos);\n if (openTagStart === -1) {\n return { value: {}, endPos: xml.length };\n }\n\n // Skip comments\n if (xml.substring(openTagStart, openTagStart + 4) === '<!--') {\n const commentEnd = xml.indexOf('-->', openTagStart + 4);\n if (commentEnd !== -1) {\n return XMLParser.parseElementToObject(xml, commentEnd + 3, options, depth);\n }\n return { value: {}, endPos: xml.length };\n }\n\n // Extract element name\n const nameMatch = /^([a-zA-Z0-9:_-]+)/.exec(xml.substring(openTagStart + 1));\n if (!nameMatch) {\n return { value: {}, endPos: openTagStart + 1 };\n }\n\n const originalElementName: string = nameMatch[1] || '';\n let elementName: string = originalElementName;\n const tagHeaderEnd = xml.indexOf('>', openTagStart);\n if (tagHeaderEnd === -1) {\n return { value: {}, endPos: xml.length };\n }\n\n // Remove namespace if requested (but keep original for offset calculations)\n if (options.ignoreNamespace && elementName.includes(':')) {\n elementName = elementName.split(':')[1] || elementName;\n }\n\n // Extract attributes using ORIGINAL element name length for correct offset\n const tagHeader = xml.substring(openTagStart + 1 + originalElementName.length, tagHeaderEnd);\n const attributes = XMLParser.extractAttributesFromTag(tagHeader, options);\n\n // Check if self-closing\n const isSelfClosing = tagHeader.trim().endsWith('/') || xml[tagHeaderEnd - 1] === '/';\n\n if (isSelfClosing) {\n // Self-closing tag - return object with attributes only\n const elementValue: ParsedXMLObject = { ...attributes };\n return {\n value: { [elementName]: elementValue },\n endPos: tagHeaderEnd + 1,\n };\n }\n\n // Find closing tag (use original name with namespace for correct matching)\n const closingTag = `</${originalElementName}>`;\n const contentStart = tagHeaderEnd + 1;\n const closingTagPos = XMLParser.findClosingTag(xml, originalElementName, contentStart);\n\n if (closingTagPos === -1) {\n // No closing tag found - treat as self-closing\n return {\n value: { [elementName]: { ...attributes } },\n endPos: tagHeaderEnd + 1,\n };\n }\n\n // Extract content between tags\n const content = xml.substring(contentStart, closingTagPos);\n\n // Parse content (children or text)\n const children: ParsedElement[] = [];\n let textContent = '';\n let pos = 0;\n\n while (pos < content.length) {\n const nextTag = content.indexOf('<', pos);\n\n if (nextTag === -1) {\n // No more tags - rest is text\n const text = content.substring(pos);\n // When trimValues is false, preserve whitespace-only text\n // When trimValues is true, only include text that has non-whitespace content\n if (text.length > 0 && (!options.trimValues || text.trim())) {\n // Unescape XML entities in text content (e.g., < -> <)\n textContent += XMLBuilder.unescapeXml(text);\n }\n break;\n }\n\n // Collect text before next tag\n if (nextTag > pos) {\n const text = content.substring(pos, nextTag);\n // When trimValues is false, preserve whitespace-only text\n // When trimValues is true, only include text that has non-whitespace content\n if (text.length > 0 && (!options.trimValues || text.trim())) {\n // Unescape XML entities in text content (e.g., < -> <)\n textContent += XMLBuilder.unescapeXml(text);\n }\n }\n\n // Parse child element (increment depth for children)\n const childResult = XMLParser.parseElementToObject(content, nextTag, options, depth + 1);\n const childObj = childResult.value as ParsedXMLObject;\n\n // Extract child name and value\n const childKeys = Object.keys(childObj);\n if (childKeys.length > 0) {\n const childName = childKeys[0];\n if (childName) {\n const childValue = childObj[childName];\n children.push({ name: childName, value: childValue });\n }\n }\n\n pos = childResult.endPos;\n }\n\n // Build element value\n let elementValue: ParsedXMLValue = {};\n\n // Add attributes\n if (!options.ignoreAttributes && Object.keys(attributes).length > 0) {\n elementValue = { ...attributes };\n }\n\n // Add text content\n // When trimValues is false, include whitespace-only text\n // When trimValues is true, only include text with non-whitespace content\n if (textContent.length > 0 && (!options.trimValues || textContent.trim())) {\n const text = options.trimValues ? textContent.trim() : textContent;\n if (typeof elementValue === 'object' && !Array.isArray(elementValue)) {\n if (Object.keys(elementValue).length === 0) {\n // Only text, no attributes - return as direct value if simple\n elementValue = text;\n } else {\n // Text with attributes\n elementValue[options.textNodeName] = text;\n }\n }\n }\n\n // Add children\n if (children.length > 0) {\n const coalescedChildren = XMLParser.coalesceChildren(children, options);\n if (typeof elementValue === 'object' && !Array.isArray(elementValue)) {\n elementValue = { ...elementValue, ...coalescedChildren };\n } else {\n elementValue = coalescedChildren;\n }\n }\n\n // If element has no content, attributes, or children - return empty object\n if (\n typeof elementValue === 'object' &&\n !Array.isArray(elementValue) &&\n Object.keys(elementValue).length === 0\n ) {\n elementValue = {};\n }\n\n return {\n value: { [elementName]: elementValue },\n endPos: closingTagPos + closingTag.length,\n };\n }\n\n /**\n * Extracts attributes from a tag header\n * @private\n */\n private static extractAttributesFromTag(\n tagHeader: string,\n options: Required<ParseToObjectOptions>\n ): Record<string, string | number | boolean> {\n const attributes: Record<string, string | number | boolean> = {};\n\n if (options.ignoreAttributes) {\n return attributes;\n }\n\n // Simple attribute extraction using position-based parsing\n let pos = 0;\n while (pos < tagHeader.length) {\n // Skip whitespace\n while (pos < tagHeader.length) {\n const char = tagHeader[pos];\n if (char && /\\s/.test(char)) {\n pos++;\n } else {\n break;\n }\n }\n\n if (pos >= tagHeader.length || tagHeader[pos] === '/') {\n break;\n }\n\n // Extract attribute name\n const nameStart = pos;\n while (pos < tagHeader.length) {\n const char = tagHeader[pos];\n if (char && /[a-zA-Z0-9:_-]/.test(char)) {\n pos++;\n } else {\n break;\n }\n }\n\n if (pos === nameStart) {\n break;\n }\n\n let attrName = tagHeader.substring(nameStart, pos);\n\n // Skip whitespace and '='\n while (pos < tagHeader.length) {\n const char = tagHeader[pos];\n if (char && /[\\s=]/.test(char)) {\n pos++;\n } else {\n break;\n }\n }\n\n // Extract attribute value\n let attrValue = '';\n if (pos < tagHeader.length && (tagHeader[pos] === '\"' || tagHeader[pos] === \"'\")) {\n const quote = tagHeader[pos];\n pos++; // Skip opening quote\n const valueStart = pos;\n\n while (pos < tagHeader.length && tagHeader[pos] !== quote) {\n pos++;\n }\n\n attrValue = tagHeader.substring(valueStart, pos);\n pos++; // Skip closing quote\n }\n\n // Remove namespace from attribute name if requested\n if (options.ignoreNamespace && attrName.includes(':')) {\n attrName = attrName.split(':')[1] || attrName;\n }\n\n // Add prefix to attribute name\n const prefixedName = options.attributeNamePrefix + attrName;\n\n // Parse attribute value\n attributes[prefixedName] = options.parseAttributeValue\n ? XMLParser.parseValue(attrValue)\n : attrValue;\n }\n\n return attributes;\n }\n\n /**\n * Finds the closing tag for an element, handling nesting\n * @private\n */\n private static findClosingTag(xml: string, elementName: string, startPos: number): number {\n const openTag = `<${elementName}`;\n const closeTag = `</${elementName}>`;\n let depth = 1;\n let pos = startPos;\n\n while (depth > 0 && pos < xml.length) {\n const nextClose = xml.indexOf(closeTag, pos);\n\n if (nextClose === -1) {\n return -1; // No closing tag found\n }\n\n // Find the next REAL opening tag (not a prefix match like <w:pPrChange for <w:pPr)\n // Must search for all potential matches and verify each one\n let realOpenPos = -1;\n let searchPos = pos;\n while (searchPos < nextClose) {\n const candidateOpen = xml.indexOf(openTag, searchPos);\n if (candidateOpen === -1 || candidateOpen >= nextClose) {\n break; // No more candidates before the closing tag\n }\n\n const charAfter = xml[candidateOpen + openTag.length];\n if (\n charAfter === '>' ||\n charAfter === ' ' ||\n charAfter === '/' ||\n charAfter === '\\t' ||\n charAfter === '\\n' ||\n charAfter === '\\r'\n ) {\n // This looks like a real opening tag - but check if it's self-closing\n // Self-closing tags like <w:rPr/> should NOT increase depth\n const tagEnd = xml.indexOf('>', candidateOpen);\n if (tagEnd !== -1 && xml[tagEnd - 1] === '/') {\n // Self-closing tag - skip it (don't affect depth)\n searchPos = tagEnd + 1;\n continue;\n }\n // This is a real opening tag (not self-closing)\n realOpenPos = candidateOpen;\n break;\n }\n\n // False positive (e.g., <w:pPrChange when looking for <w:pPr)\n // Keep searching from after this position\n searchPos = candidateOpen + openTag.length;\n }\n\n if (realOpenPos !== -1) {\n // Found a real opening tag before the closing tag - increase depth\n depth++;\n pos = realOpenPos + openTag.length;\n } else {\n // No real opening tag before this closing tag - decrease depth\n depth--;\n if (depth === 0) {\n return nextClose;\n }\n pos = nextClose + closeTag.length;\n }\n }\n\n return -1;\n }\n\n /**\n * Coalesces children with duplicate names into arrays\n * @private\n */\n private static coalesceChildren(\n children: ParsedElement[],\n options: Required<ParseToObjectOptions>\n ): ParsedXMLObject {\n const result: ParsedXMLObject = {};\n const nameCounts: Record<string, number> = {};\n const nameIndices: Record<string, number> = {};\n\n // Track element order for correct run content parsing (tabs, breaks, text)\n // This is critical for preserving the order of mixed content like: text -> tab -> text\n const orderedChildren: { type: string; index: number }[] = [];\n\n // Count occurrences of each child name\n for (const child of children) {\n nameCounts[child.name] = (nameCounts[child.name] || 0) + 1;\n }\n\n // Build result object while tracking order\n for (const child of children) {\n const shouldBeArray = options.alwaysArray || (nameCounts[child.name] || 0) > 1;\n\n // Track element order with its index in the array\n const currentIndex = nameIndices[child.name] || 0;\n orderedChildren.push({ type: child.name, index: currentIndex });\n nameIndices[child.name] = currentIndex + 1;\n\n if (shouldBeArray) {\n if (!result[child.name]) {\n result[child.name] = [];\n }\n (result[child.name] as ParsedXMLValue[]).push(child.value);\n } else {\n result[child.name] = child.value;\n }\n }\n\n // Add _orderedChildren to track element order (used by DocumentParser for runs)\n if (orderedChildren.length > 0) {\n result._orderedChildren = orderedChildren;\n }\n\n return result;\n }\n\n /**\n * Parses a string value to number or boolean if applicable\n * @private\n */\n private static parseValue(value: string): string | number | boolean {\n if (value === 'true') return true;\n if (value === 'false') return false;\n\n // Preserve 6-character hex color codes (OpenXML standard for colors)\n // This includes \"000000\" (black) which should stay as a string\n if (/^[0-9A-Fa-f]{6}$/.test(value)) {\n return value.toUpperCase(); // Normalize to uppercase per Microsoft convention\n }\n\n // Preserve long digit-only strings (e.g., cnfStyle binary strings like \"100000000000\")\n // These should not be converted to numbers to avoid losing leading zeros\n if (/^\\d{7,}$/.test(value)) {\n return value; // Keep as string for values with 7+ digits\n }\n\n // Try parsing as number\n // 3-character values like \"240\" will be parsed as numbers\n // 6-character hex values are already handled above\n if (/^-?\\d+$/.test(value)) {\n const num = parseInt(value, 10);\n if (!isNaN(num)) return num;\n }\n\n if (/^-?\\d+\\.\\d+$/.test(value)) {\n const num = parseFloat(value);\n if (!isNaN(num)) return num;\n }\n\n // Preserve 3-character hex codes (like \"F0A\") that have letters\n // Pure numeric 3-char values (like \"240\") are already parsed as numbers above\n if (/^[0-9A-Fa-f]{3}$/.test(value) && /[A-Fa-f]/.test(value)) {\n return value.toUpperCase();\n }\n\n return value;\n }\n}\n"]}
|