This is an automated email from the ASF dual-hosted git repository.
Claudenw pushed a commit to branch master
in repository https://gitbox.apache.org/repos/asf/creadur-rat.git
The following commit(s) were added to refs/heads/master by this push:
new da14353f RAT-539: Update xml writer (#627)
da14353f is described below
commit da14353f901d7259d01d4d389fdc3e02132564fc
Author: Claude Warren <[email protected]>
AuthorDate: Mon May 4 15:59:37 2026 +0200
RAT-539: Update xml writer (#627)
* moved Char handling to XMLChar, switched XmlWriter to write to appender
* added checkstyle supressions to and made XMLChar final
* Fixed issue with not closing stream before tryiong to read
* updated test
* added append() method to IXmlWriter
* fixed XXE issue
* fixed checkstyle error
* added checkstyle supression for XMLChar.java
* fixed spotbugs issues
* attempt at sonar exclusion
* fixed XXE issue and added testing
* resolved XsdGenerator issue
* attempt to fix windows line end problem
* fixed sonar issues
* added pom comment
* fixed formatting issue
* LHF: fix javadoc
* Add reference to RAT-539 in pom comment
* Add reference to RAT-539 in checkstylecomment
---------
Co-authored-by: P. Ottlinger <[email protected]>
---
apache-rat-core/pom.xml | 4 +
.../src/main/java/org/apache/rat/Reporter.java | 5 +-
.../apache/rat/report/xml/writer/IXmlWriter.java | 9 +
.../org/apache/rat/report/xml/writer/XMLChar.java | 1068 ++++++++++++++++++++
.../apache/rat/report/xml/writer/XmlWriter.java | 563 ++---------
.../org/apache/rat/utils/StandardXmlFactory.java | 67 ++
.../report/xml/writer/impl/base/XmlWriterTest.java | 136 ++-
.../org/apache/rat/tools/xsd/XsdGenerator.java | 20 +-
src/conf/checkstyle-suppressions.xml | 4 +
9 files changed, 1349 insertions(+), 527 deletions(-)
diff --git a/apache-rat-core/pom.xml b/apache-rat-core/pom.xml
index 3984e84a..b60c6cc8 100644
--- a/apache-rat-core/pom.xml
+++ b/apache-rat-core/pom.xml
@@ -26,6 +26,10 @@
<packaging>jar</packaging>
<name>Apache Creadur RAT::Core</name>
<description>The core functionality of RAT that is used by all
clients.</description>
+ <properties>
+ <!-- RAT-539: XMLChar.java is copied from Xerces codebase and so does no
meet our sonar tests -->
+
<sonar.exclusions>src/main/java/org/apache/rat/report/xml/writer/XMLChar.java</sonar.exclusions>
+ </properties>
<build>
<resources>
<resource>
diff --git a/apache-rat-core/src/main/java/org/apache/rat/Reporter.java
b/apache-rat-core/src/main/java/org/apache/rat/Reporter.java
index ec32f340..2e58c582 100644
--- a/apache-rat-core/src/main/java/org/apache/rat/Reporter.java
+++ b/apache-rat-core/src/main/java/org/apache/rat/Reporter.java
@@ -94,10 +94,9 @@ public class Reporter {
report.startReport();
configuration.getSources().build().run(report);
report.endReport();
-
- InputStream inputStream = new
ByteArrayInputStream(outputStream.toByteArray());
- document =
DocumentBuilderFactory.newInstance().newDocumentBuilder().parse(inputStream);
}
+ InputStream inputStream = new
ByteArrayInputStream(outputStream.toByteArray());
+ document =
DocumentBuilderFactory.newInstance().newDocumentBuilder().parse(inputStream);
} else {
document =
DocumentBuilderFactory.newInstance().newDocumentBuilder().newDocument();
statistic = new ClaimStatistic();
diff --git
a/apache-rat-core/src/main/java/org/apache/rat/report/xml/writer/IXmlWriter.java
b/apache-rat-core/src/main/java/org/apache/rat/report/xml/writer/IXmlWriter.java
index 509287f5..a227b9c5 100644
---
a/apache-rat-core/src/main/java/org/apache/rat/report/xml/writer/IXmlWriter.java
+++
b/apache-rat-core/src/main/java/org/apache/rat/report/xml/writer/IXmlWriter.java
@@ -21,6 +21,7 @@ package org.apache.rat.report.xml.writer;
import java.io.IOException;
import org.apache.rat.report.xml.XmlElements;
+import org.w3c.dom.Document;
/**
* Simple interface for creating basic XML documents.
@@ -147,4 +148,12 @@ public interface IXmlWriter extends AutoCloseable {
* if called before any call to {@link #openElement(CharSequence)}
*/
IXmlWriter closeDocument() throws IOException;
+
+ /**
+ * Append an XML document into this one.
+ * @param document the document to append
+ * @return this object
+ * @throws IOException on error.
+ */
+ IXmlWriter append(Document document) throws IOException;
}
diff --git
a/apache-rat-core/src/main/java/org/apache/rat/report/xml/writer/XMLChar.java
b/apache-rat-core/src/main/java/org/apache/rat/report/xml/writer/XMLChar.java
new file mode 100644
index 00000000..644b82c8
--- /dev/null
+++
b/apache-rat-core/src/main/java/org/apache/rat/report/xml/writer/XMLChar.java
@@ -0,0 +1,1068 @@
+package org.apache.rat.report.xml.writer;
+
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one or more
+ * contributor license agreements. See the NOTICE file distributed with
+ * this work for additional information regarding copyright ownership.
+ * The ASF licenses this file to You under the Apache License, Version 2.0
+ * (the "License"); you may not use this file except in compliance with
+ * the License. You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+// this code is lifted from
https://github.com/apache/xerces2-j/blob/trunk/src/org/apache/xerces/util/XMLChar.java
+
+import java.util.Arrays;
+
+/**
+ * This class defines the basic XML character properties. The data
+ * in this class can be used to verify that a character is a valid
+ * XML character or if the character is a space, name start, or name
+ * character.
+ * <p>
+ * A series of convenience methods are supplied to ease the burden
+ * of the developer. Because inlining the checks can improve per
+ * character performance, the tables of character properties are
+ * public. Using the character as an index into the <code>CHARS</code>
+ * array and applying the appropriate mask flag (e.g.
+ * <code>MASK_VALID</code>), yields the same results as calling the
+ * convenience methods. There is one exception: check the comments
+ * for the <code>isValid</code> method for details.
+ *
+ * @author Glenn Marcy, IBM
+ * @author Andy Clark, IBM
+ * @author Eric Ye, IBM
+ * @author Arnaud Le Hors, IBM
+ * @author Michael Glavassevich, IBM
+ * @author Rahul Srivastava, Sun Microsystems Inc.
+ *
+ * @version $Id$
+ */
+public final class XMLChar {
+
+ private XMLChar() {
+ // do not instantiate.
+ }
+
+ //
+ // Constants
+ //
+
+ /** Character flags. */
+ private static final byte[] CHARS = new byte[1 << 16];
+
+ /** Valid character mask. */
+ public static final int MASK_VALID = 0x01;
+
+ /** Space character mask. */
+ public static final int MASK_SPACE = 0x02;
+
+ /** Name start character mask. */
+ public static final int MASK_NAME_START = 0x04;
+
+ /** Name character mask. */
+ public static final int MASK_NAME = 0x08;
+
+ /** Pubid character mask. */
+ public static final int MASK_PUBID = 0x10;
+
+ /**
+ * Content character mask. Special characters are those that can
+ * be considered the start of markup, such as '<' and '&'.
+ * The various newline characters are considered special as well.
+ * All other valid XML characters can be considered content.
+ * <p>
+ * This is an optimization for the inner loop of character scanning.
+ */
+ public static final int MASK_CONTENT = 0x20;
+
+ /** NCName start character mask. */
+ public static final int MASK_NCNAME_START = 0x40;
+
+ /** NCName character mask. */
+ public static final int MASK_NCNAME = 0x80;
+
+ //
+ // Static initialization
+ //
+
+ static {
+
+ // Initializing the Character Flag Array
+ // Code generated by: XMLCharGenerator.
+
+ CHARS[9] = 35;
+ CHARS[10] = 19;
+ CHARS[13] = 19;
+ CHARS[32] = 51;
+ CHARS[33] = 49;
+ CHARS[34] = 33;
+ Arrays.fill(CHARS, 35, 38, (byte) 49 ); // Fill 3 of value (byte) 49
+ CHARS[38] = 1;
+ Arrays.fill(CHARS, 39, 45, (byte) 49 ); // Fill 6 of value (byte) 49
+ Arrays.fill(CHARS, 45, 47, (byte) -71 ); // Fill 2 of value (byte) -71
+ CHARS[47] = 49;
+ Arrays.fill(CHARS, 48, 58, (byte) -71 ); // Fill 10 of value (byte) -71
+ CHARS[58] = 61;
+ CHARS[59] = 49;
+ CHARS[60] = 1;
+ CHARS[61] = 49;
+ CHARS[62] = 33;
+ Arrays.fill(CHARS, 63, 65, (byte) 49 ); // Fill 2 of value (byte) 49
+ Arrays.fill(CHARS, 65, 91, (byte) -3 ); // Fill 26 of value (byte) -3
+ Arrays.fill(CHARS, 91, 93, (byte) 33 ); // Fill 2 of value (byte) 33
+ CHARS[93] = 1;
+ CHARS[94] = 33;
+ CHARS[95] = -3;
+ CHARS[96] = 33;
+ Arrays.fill(CHARS, 97, 123, (byte) -3 ); // Fill 26 of value (byte) -3
+ Arrays.fill(CHARS, 123, 183, (byte) 33 ); // Fill 60 of value (byte) 33
+ CHARS[183] = -87;
+ Arrays.fill(CHARS, 184, 192, (byte) 33 ); // Fill 8 of value (byte) 33
+ Arrays.fill(CHARS, 192, 215, (byte) -19 ); // Fill 23 of value (byte)
-19
+ CHARS[215] = 33;
+ Arrays.fill(CHARS, 216, 247, (byte) -19 ); // Fill 31 of value (byte)
-19
+ CHARS[247] = 33;
+ Arrays.fill(CHARS, 248, 306, (byte) -19 ); // Fill 58 of value (byte)
-19
+ Arrays.fill(CHARS, 306, 308, (byte) 33 ); // Fill 2 of value (byte) 33
+ Arrays.fill(CHARS, 308, 319, (byte) -19 ); // Fill 11 of value (byte)
-19
+ Arrays.fill(CHARS, 319, 321, (byte) 33 ); // Fill 2 of value (byte) 33
+ Arrays.fill(CHARS, 321, 329, (byte) -19 ); // Fill 8 of value (byte)
-19
+ CHARS[329] = 33;
+ Arrays.fill(CHARS, 330, 383, (byte) -19 ); // Fill 53 of value (byte)
-19
+ CHARS[383] = 33;
+ Arrays.fill(CHARS, 384, 452, (byte) -19 ); // Fill 68 of value (byte)
-19
+ Arrays.fill(CHARS, 452, 461, (byte) 33 ); // Fill 9 of value (byte) 33
+ Arrays.fill(CHARS, 461, 497, (byte) -19 ); // Fill 36 of value (byte)
-19
+ Arrays.fill(CHARS, 497, 500, (byte) 33 ); // Fill 3 of value (byte) 33
+ Arrays.fill(CHARS, 500, 502, (byte) -19 ); // Fill 2 of value (byte)
-19
+ Arrays.fill(CHARS, 502, 506, (byte) 33 ); // Fill 4 of value (byte) 33
+ Arrays.fill(CHARS, 506, 536, (byte) -19 ); // Fill 30 of value (byte)
-19
+ Arrays.fill(CHARS, 536, 592, (byte) 33 ); // Fill 56 of value (byte) 33
+ Arrays.fill(CHARS, 592, 681, (byte) -19 ); // Fill 89 of value (byte)
-19
+ Arrays.fill(CHARS, 681, 699, (byte) 33 ); // Fill 18 of value (byte) 33
+ Arrays.fill(CHARS, 699, 706, (byte) -19 ); // Fill 7 of value (byte)
-19
+ Arrays.fill(CHARS, 706, 720, (byte) 33 ); // Fill 14 of value (byte) 33
+ Arrays.fill(CHARS, 720, 722, (byte) -87 ); // Fill 2 of value (byte)
-87
+ Arrays.fill(CHARS, 722, 768, (byte) 33 ); // Fill 46 of value (byte) 33
+ Arrays.fill(CHARS, 768, 838, (byte) -87 ); // Fill 70 of value (byte)
-87
+ Arrays.fill(CHARS, 838, 864, (byte) 33 ); // Fill 26 of value (byte) 33
+ Arrays.fill(CHARS, 864, 866, (byte) -87 ); // Fill 2 of value (byte)
-87
+ Arrays.fill(CHARS, 866, 902, (byte) 33 ); // Fill 36 of value (byte) 33
+ CHARS[902] = -19;
+ CHARS[903] = -87;
+ Arrays.fill(CHARS, 904, 907, (byte) -19 ); // Fill 3 of value (byte)
-19
+ CHARS[907] = 33;
+ CHARS[908] = -19;
+ CHARS[909] = 33;
+ Arrays.fill(CHARS, 910, 930, (byte) -19 ); // Fill 20 of value (byte)
-19
+ CHARS[930] = 33;
+ Arrays.fill(CHARS, 931, 975, (byte) -19 ); // Fill 44 of value (byte)
-19
+ CHARS[975] = 33;
+ Arrays.fill(CHARS, 976, 983, (byte) -19 ); // Fill 7 of value (byte)
-19
+ Arrays.fill(CHARS, 983, 986, (byte) 33 ); // Fill 3 of value (byte) 33
+ CHARS[986] = -19;
+ CHARS[987] = 33;
+ CHARS[988] = -19;
+ CHARS[989] = 33;
+ CHARS[990] = -19;
+ CHARS[991] = 33;
+ CHARS[992] = -19;
+ CHARS[993] = 33;
+ Arrays.fill(CHARS, 994, 1012, (byte) -19 ); // Fill 18 of value (byte)
-19
+ Arrays.fill(CHARS, 1012, 1025, (byte) 33 ); // Fill 13 of value (byte)
33
+ Arrays.fill(CHARS, 1025, 1037, (byte) -19 ); // Fill 12 of value
(byte) -19
+ CHARS[1037] = 33;
+ Arrays.fill(CHARS, 1038, 1104, (byte) -19 ); // Fill 66 of value
(byte) -19
+ CHARS[1104] = 33;
+ Arrays.fill(CHARS, 1105, 1117, (byte) -19 ); // Fill 12 of value
(byte) -19
+ CHARS[1117] = 33;
+ Arrays.fill(CHARS, 1118, 1154, (byte) -19 ); // Fill 36 of value
(byte) -19
+ CHARS[1154] = 33;
+ Arrays.fill(CHARS, 1155, 1159, (byte) -87 ); // Fill 4 of value (byte)
-87
+ Arrays.fill(CHARS, 1159, 1168, (byte) 33 ); // Fill 9 of value (byte)
33
+ Arrays.fill(CHARS, 1168, 1221, (byte) -19 ); // Fill 53 of value
(byte) -19
+ Arrays.fill(CHARS, 1221, 1223, (byte) 33 ); // Fill 2 of value (byte)
33
+ Arrays.fill(CHARS, 1223, 1225, (byte) -19 ); // Fill 2 of value (byte)
-19
+ Arrays.fill(CHARS, 1225, 1227, (byte) 33 ); // Fill 2 of value (byte)
33
+ Arrays.fill(CHARS, 1227, 1229, (byte) -19 ); // Fill 2 of value (byte)
-19
+ Arrays.fill(CHARS, 1229, 1232, (byte) 33 ); // Fill 3 of value (byte)
33
+ Arrays.fill(CHARS, 1232, 1260, (byte) -19 ); // Fill 28 of value
(byte) -19
+ Arrays.fill(CHARS, 1260, 1262, (byte) 33 ); // Fill 2 of value (byte)
33
+ Arrays.fill(CHARS, 1262, 1270, (byte) -19 ); // Fill 8 of value (byte)
-19
+ Arrays.fill(CHARS, 1270, 1272, (byte) 33 ); // Fill 2 of value (byte)
33
+ Arrays.fill(CHARS, 1272, 1274, (byte) -19 ); // Fill 2 of value (byte)
-19
+ Arrays.fill(CHARS, 1274, 1329, (byte) 33 ); // Fill 55 of value (byte)
33
+ Arrays.fill(CHARS, 1329, 1367, (byte) -19 ); // Fill 38 of value
(byte) -19
+ Arrays.fill(CHARS, 1367, 1369, (byte) 33 ); // Fill 2 of value (byte)
33
+ CHARS[1369] = -19;
+ Arrays.fill(CHARS, 1370, 1377, (byte) 33 ); // Fill 7 of value (byte)
33
+ Arrays.fill(CHARS, 1377, 1415, (byte) -19 ); // Fill 38 of value
(byte) -19
+ Arrays.fill(CHARS, 1415, 1425, (byte) 33 ); // Fill 10 of value (byte)
33
+ Arrays.fill(CHARS, 1425, 1442, (byte) -87 ); // Fill 17 of value
(byte) -87
+ CHARS[1442] = 33;
+ Arrays.fill(CHARS, 1443, 1466, (byte) -87 ); // Fill 23 of value
(byte) -87
+ CHARS[1466] = 33;
+ Arrays.fill(CHARS, 1467, 1470, (byte) -87 ); // Fill 3 of value (byte)
-87
+ CHARS[1470] = 33;
+ CHARS[1471] = -87;
+ CHARS[1472] = 33;
+ Arrays.fill(CHARS, 1473, 1475, (byte) -87 ); // Fill 2 of value (byte)
-87
+ CHARS[1475] = 33;
+ CHARS[1476] = -87;
+ Arrays.fill(CHARS, 1477, 1488, (byte) 33 ); // Fill 11 of value (byte)
33
+ Arrays.fill(CHARS, 1488, 1515, (byte) -19 ); // Fill 27 of value
(byte) -19
+ Arrays.fill(CHARS, 1515, 1520, (byte) 33 ); // Fill 5 of value (byte)
33
+ Arrays.fill(CHARS, 1520, 1523, (byte) -19 ); // Fill 3 of value (byte)
-19
+ Arrays.fill(CHARS, 1523, 1569, (byte) 33 ); // Fill 46 of value (byte)
33
+ Arrays.fill(CHARS, 1569, 1595, (byte) -19 ); // Fill 26 of value
(byte) -19
+ Arrays.fill(CHARS, 1595, 1600, (byte) 33 ); // Fill 5 of value (byte)
33
+ CHARS[1600] = -87;
+ Arrays.fill(CHARS, 1601, 1611, (byte) -19 ); // Fill 10 of value
(byte) -19
+ Arrays.fill(CHARS, 1611, 1619, (byte) -87 ); // Fill 8 of value (byte)
-87
+ Arrays.fill(CHARS, 1619, 1632, (byte) 33 ); // Fill 13 of value (byte)
33
+ Arrays.fill(CHARS, 1632, 1642, (byte) -87 ); // Fill 10 of value
(byte) -87
+ Arrays.fill(CHARS, 1642, 1648, (byte) 33 ); // Fill 6 of value (byte)
33
+ CHARS[1648] = -87;
+ Arrays.fill(CHARS, 1649, 1720, (byte) -19 ); // Fill 71 of value
(byte) -19
+ Arrays.fill(CHARS, 1720, 1722, (byte) 33 ); // Fill 2 of value (byte)
33
+ Arrays.fill(CHARS, 1722, 1727, (byte) -19 ); // Fill 5 of value (byte)
-19
+ CHARS[1727] = 33;
+ Arrays.fill(CHARS, 1728, 1743, (byte) -19 ); // Fill 15 of value
(byte) -19
+ CHARS[1743] = 33;
+ Arrays.fill(CHARS, 1744, 1748, (byte) -19 ); // Fill 4 of value (byte)
-19
+ CHARS[1748] = 33;
+ CHARS[1749] = -19;
+ Arrays.fill(CHARS, 1750, 1765, (byte) -87 ); // Fill 15 of value
(byte) -87
+ Arrays.fill(CHARS, 1765, 1767, (byte) -19 ); // Fill 2 of value (byte)
-19
+ Arrays.fill(CHARS, 1767, 1769, (byte) -87 ); // Fill 2 of value (byte)
-87
+ CHARS[1769] = 33;
+ Arrays.fill(CHARS, 1770, 1774, (byte) -87 ); // Fill 4 of value (byte)
-87
+ Arrays.fill(CHARS, 1774, 1776, (byte) 33 ); // Fill 2 of value (byte)
33
+ Arrays.fill(CHARS, 1776, 1786, (byte) -87 ); // Fill 10 of value
(byte) -87
+ Arrays.fill(CHARS, 1786, 2305, (byte) 33 ); // Fill 519 of value
(byte) 33
+ Arrays.fill(CHARS, 2305, 2308, (byte) -87 ); // Fill 3 of value (byte)
-87
+ CHARS[2308] = 33;
+ Arrays.fill(CHARS, 2309, 2362, (byte) -19 ); // Fill 53 of value
(byte) -19
+ Arrays.fill(CHARS, 2362, 2364, (byte) 33 ); // Fill 2 of value (byte)
33
+ CHARS[2364] = -87;
+ CHARS[2365] = -19;
+ Arrays.fill(CHARS, 2366, 2382, (byte) -87 ); // Fill 16 of value
(byte) -87
+ Arrays.fill(CHARS, 2382, 2385, (byte) 33 ); // Fill 3 of value (byte)
33
+ Arrays.fill(CHARS, 2385, 2389, (byte) -87 ); // Fill 4 of value (byte)
-87
+ Arrays.fill(CHARS, 2389, 2392, (byte) 33 ); // Fill 3 of value (byte)
33
+ Arrays.fill(CHARS, 2392, 2402, (byte) -19 ); // Fill 10 of value
(byte) -19
+ Arrays.fill(CHARS, 2402, 2404, (byte) -87 ); // Fill 2 of value (byte)
-87
+ Arrays.fill(CHARS, 2404, 2406, (byte) 33 ); // Fill 2 of value (byte)
33
+ Arrays.fill(CHARS, 2406, 2416, (byte) -87 ); // Fill 10 of value
(byte) -87
+ Arrays.fill(CHARS, 2416, 2433, (byte) 33 ); // Fill 17 of value (byte)
33
+ Arrays.fill(CHARS, 2433, 2436, (byte) -87 ); // Fill 3 of value (byte)
-87
+ CHARS[2436] = 33;
+ Arrays.fill(CHARS, 2437, 2445, (byte) -19 ); // Fill 8 of value (byte)
-19
+ Arrays.fill(CHARS, 2445, 2447, (byte) 33 ); // Fill 2 of value (byte)
33
+ Arrays.fill(CHARS, 2447, 2449, (byte) -19 ); // Fill 2 of value (byte)
-19
+ Arrays.fill(CHARS, 2449, 2451, (byte) 33 ); // Fill 2 of value (byte)
33
+ Arrays.fill(CHARS, 2451, 2473, (byte) -19 ); // Fill 22 of value
(byte) -19
+ CHARS[2473] = 33;
+ Arrays.fill(CHARS, 2474, 2481, (byte) -19 ); // Fill 7 of value (byte)
-19
+ CHARS[2481] = 33;
+ CHARS[2482] = -19;
+ Arrays.fill(CHARS, 2483, 2486, (byte) 33 ); // Fill 3 of value (byte)
33
+ Arrays.fill(CHARS, 2486, 2490, (byte) -19 ); // Fill 4 of value (byte)
-19
+ Arrays.fill(CHARS, 2490, 2492, (byte) 33 ); // Fill 2 of value (byte)
33
+ CHARS[2492] = -87;
+ CHARS[2493] = 33;
+ Arrays.fill(CHARS, 2494, 2501, (byte) -87 ); // Fill 7 of value (byte)
-87
+ Arrays.fill(CHARS, 2501, 2503, (byte) 33 ); // Fill 2 of value (byte)
33
+ Arrays.fill(CHARS, 2503, 2505, (byte) -87 ); // Fill 2 of value (byte)
-87
+ Arrays.fill(CHARS, 2505, 2507, (byte) 33 ); // Fill 2 of value (byte)
33
+ Arrays.fill(CHARS, 2507, 2510, (byte) -87 ); // Fill 3 of value (byte)
-87
+ Arrays.fill(CHARS, 2510, 2519, (byte) 33 ); // Fill 9 of value (byte)
33
+ CHARS[2519] = -87;
+ Arrays.fill(CHARS, 2520, 2524, (byte) 33 ); // Fill 4 of value (byte)
33
+ Arrays.fill(CHARS, 2524, 2526, (byte) -19 ); // Fill 2 of value (byte)
-19
+ CHARS[2526] = 33;
+ Arrays.fill(CHARS, 2527, 2530, (byte) -19 ); // Fill 3 of value (byte)
-19
+ Arrays.fill(CHARS, 2530, 2532, (byte) -87 ); // Fill 2 of value (byte)
-87
+ Arrays.fill(CHARS, 2532, 2534, (byte) 33 ); // Fill 2 of value (byte)
33
+ Arrays.fill(CHARS, 2534, 2544, (byte) -87 ); // Fill 10 of value
(byte) -87
+ Arrays.fill(CHARS, 2544, 2546, (byte) -19 ); // Fill 2 of value (byte)
-19
+ Arrays.fill(CHARS, 2546, 2562, (byte) 33 ); // Fill 16 of value (byte)
33
+ CHARS[2562] = -87;
+ Arrays.fill(CHARS, 2563, 2565, (byte) 33 ); // Fill 2 of value (byte)
33
+ Arrays.fill(CHARS, 2565, 2571, (byte) -19 ); // Fill 6 of value (byte)
-19
+ Arrays.fill(CHARS, 2571, 2575, (byte) 33 ); // Fill 4 of value (byte)
33
+ Arrays.fill(CHARS, 2575, 2577, (byte) -19 ); // Fill 2 of value (byte)
-19
+ Arrays.fill(CHARS, 2577, 2579, (byte) 33 ); // Fill 2 of value (byte)
33
+ Arrays.fill(CHARS, 2579, 2601, (byte) -19 ); // Fill 22 of value
(byte) -19
+ CHARS[2601] = 33;
+ Arrays.fill(CHARS, 2602, 2609, (byte) -19 ); // Fill 7 of value (byte)
-19
+ CHARS[2609] = 33;
+ Arrays.fill(CHARS, 2610, 2612, (byte) -19 ); // Fill 2 of value (byte)
-19
+ CHARS[2612] = 33;
+ Arrays.fill(CHARS, 2613, 2615, (byte) -19 ); // Fill 2 of value (byte)
-19
+ CHARS[2615] = 33;
+ Arrays.fill(CHARS, 2616, 2618, (byte) -19 ); // Fill 2 of value (byte)
-19
+ Arrays.fill(CHARS, 2618, 2620, (byte) 33 ); // Fill 2 of value (byte)
33
+ CHARS[2620] = -87;
+ CHARS[2621] = 33;
+ Arrays.fill(CHARS, 2622, 2627, (byte) -87 ); // Fill 5 of value (byte)
-87
+ Arrays.fill(CHARS, 2627, 2631, (byte) 33 ); // Fill 4 of value (byte)
33
+ Arrays.fill(CHARS, 2631, 2633, (byte) -87 ); // Fill 2 of value (byte)
-87
+ Arrays.fill(CHARS, 2633, 2635, (byte) 33 ); // Fill 2 of value (byte)
33
+ Arrays.fill(CHARS, 2635, 2638, (byte) -87 ); // Fill 3 of value (byte)
-87
+ Arrays.fill(CHARS, 2638, 2649, (byte) 33 ); // Fill 11 of value (byte)
33
+ Arrays.fill(CHARS, 2649, 2653, (byte) -19 ); // Fill 4 of value (byte)
-19
+ CHARS[2653] = 33;
+ CHARS[2654] = -19;
+ Arrays.fill(CHARS, 2655, 2662, (byte) 33 ); // Fill 7 of value (byte)
33
+ Arrays.fill(CHARS, 2662, 2674, (byte) -87 ); // Fill 12 of value
(byte) -87
+ Arrays.fill(CHARS, 2674, 2677, (byte) -19 ); // Fill 3 of value (byte)
-19
+ Arrays.fill(CHARS, 2677, 2689, (byte) 33 ); // Fill 12 of value (byte)
33
+ Arrays.fill(CHARS, 2689, 2692, (byte) -87 ); // Fill 3 of value (byte)
-87
+ CHARS[2692] = 33;
+ Arrays.fill(CHARS, 2693, 2700, (byte) -19 ); // Fill 7 of value (byte)
-19
+ CHARS[2700] = 33;
+ CHARS[2701] = -19;
+ CHARS[2702] = 33;
+ Arrays.fill(CHARS, 2703, 2706, (byte) -19 ); // Fill 3 of value (byte)
-19
+ CHARS[2706] = 33;
+ Arrays.fill(CHARS, 2707, 2729, (byte) -19 ); // Fill 22 of value
(byte) -19
+ CHARS[2729] = 33;
+ Arrays.fill(CHARS, 2730, 2737, (byte) -19 ); // Fill 7 of value (byte)
-19
+ CHARS[2737] = 33;
+ Arrays.fill(CHARS, 2738, 2740, (byte) -19 ); // Fill 2 of value (byte)
-19
+ CHARS[2740] = 33;
+ Arrays.fill(CHARS, 2741, 2746, (byte) -19 ); // Fill 5 of value (byte)
-19
+ Arrays.fill(CHARS, 2746, 2748, (byte) 33 ); // Fill 2 of value (byte)
33
+ CHARS[2748] = -87;
+ CHARS[2749] = -19;
+ Arrays.fill(CHARS, 2750, 2758, (byte) -87 ); // Fill 8 of value (byte)
-87
+ CHARS[2758] = 33;
+ Arrays.fill(CHARS, 2759, 2762, (byte) -87 ); // Fill 3 of value (byte)
-87
+ CHARS[2762] = 33;
+ Arrays.fill(CHARS, 2763, 2766, (byte) -87 ); // Fill 3 of value (byte)
-87
+ Arrays.fill(CHARS, 2766, 2784, (byte) 33 ); // Fill 18 of value (byte)
33
+ CHARS[2784] = -19;
+ Arrays.fill(CHARS, 2785, 2790, (byte) 33 ); // Fill 5 of value (byte)
33
+ Arrays.fill(CHARS, 2790, 2800, (byte) -87 ); // Fill 10 of value
(byte) -87
+ Arrays.fill(CHARS, 2800, 2817, (byte) 33 ); // Fill 17 of value (byte)
33
+ Arrays.fill(CHARS, 2817, 2820, (byte) -87 ); // Fill 3 of value (byte)
-87
+ CHARS[2820] = 33;
+ Arrays.fill(CHARS, 2821, 2829, (byte) -19 ); // Fill 8 of value (byte)
-19
+ Arrays.fill(CHARS, 2829, 2831, (byte) 33 ); // Fill 2 of value (byte)
33
+ Arrays.fill(CHARS, 2831, 2833, (byte) -19 ); // Fill 2 of value (byte)
-19
+ Arrays.fill(CHARS, 2833, 2835, (byte) 33 ); // Fill 2 of value (byte)
33
+ Arrays.fill(CHARS, 2835, 2857, (byte) -19 ); // Fill 22 of value
(byte) -19
+ CHARS[2857] = 33;
+ Arrays.fill(CHARS, 2858, 2865, (byte) -19 ); // Fill 7 of value (byte)
-19
+ CHARS[2865] = 33;
+ Arrays.fill(CHARS, 2866, 2868, (byte) -19 ); // Fill 2 of value (byte)
-19
+ Arrays.fill(CHARS, 2868, 2870, (byte) 33 ); // Fill 2 of value (byte)
33
+ Arrays.fill(CHARS, 2870, 2874, (byte) -19 ); // Fill 4 of value (byte)
-19
+ Arrays.fill(CHARS, 2874, 2876, (byte) 33 ); // Fill 2 of value (byte)
33
+ CHARS[2876] = -87;
+ CHARS[2877] = -19;
+ Arrays.fill(CHARS, 2878, 2884, (byte) -87 ); // Fill 6 of value (byte)
-87
+ Arrays.fill(CHARS, 2884, 2887, (byte) 33 ); // Fill 3 of value (byte)
33
+ Arrays.fill(CHARS, 2887, 2889, (byte) -87 ); // Fill 2 of value (byte)
-87
+ Arrays.fill(CHARS, 2889, 2891, (byte) 33 ); // Fill 2 of value (byte)
33
+ Arrays.fill(CHARS, 2891, 2894, (byte) -87 ); // Fill 3 of value (byte)
-87
+ Arrays.fill(CHARS, 2894, 2902, (byte) 33 ); // Fill 8 of value (byte)
33
+ Arrays.fill(CHARS, 2902, 2904, (byte) -87 ); // Fill 2 of value (byte)
-87
+ Arrays.fill(CHARS, 2904, 2908, (byte) 33 ); // Fill 4 of value (byte)
33
+ Arrays.fill(CHARS, 2908, 2910, (byte) -19 ); // Fill 2 of value (byte)
-19
+ CHARS[2910] = 33;
+ Arrays.fill(CHARS, 2911, 2914, (byte) -19 ); // Fill 3 of value (byte)
-19
+ Arrays.fill(CHARS, 2914, 2918, (byte) 33 ); // Fill 4 of value (byte)
33
+ Arrays.fill(CHARS, 2918, 2928, (byte) -87 ); // Fill 10 of value
(byte) -87
+ Arrays.fill(CHARS, 2928, 2946, (byte) 33 ); // Fill 18 of value (byte)
33
+ Arrays.fill(CHARS, 2946, 2948, (byte) -87 ); // Fill 2 of value (byte)
-87
+ CHARS[2948] = 33;
+ Arrays.fill(CHARS, 2949, 2955, (byte) -19 ); // Fill 6 of value (byte)
-19
+ Arrays.fill(CHARS, 2955, 2958, (byte) 33 ); // Fill 3 of value (byte)
33
+ Arrays.fill(CHARS, 2958, 2961, (byte) -19 ); // Fill 3 of value (byte)
-19
+ CHARS[2961] = 33;
+ Arrays.fill(CHARS, 2962, 2966, (byte) -19 ); // Fill 4 of value (byte)
-19
+ Arrays.fill(CHARS, 2966, 2969, (byte) 33 ); // Fill 3 of value (byte)
33
+ Arrays.fill(CHARS, 2969, 2971, (byte) -19 ); // Fill 2 of value (byte)
-19
+ CHARS[2971] = 33;
+ CHARS[2972] = -19;
+ CHARS[2973] = 33;
+ Arrays.fill(CHARS, 2974, 2976, (byte) -19 ); // Fill 2 of value (byte)
-19
+ Arrays.fill(CHARS, 2976, 2979, (byte) 33 ); // Fill 3 of value (byte)
33
+ Arrays.fill(CHARS, 2979, 2981, (byte) -19 ); // Fill 2 of value (byte)
-19
+ Arrays.fill(CHARS, 2981, 2984, (byte) 33 ); // Fill 3 of value (byte)
33
+ Arrays.fill(CHARS, 2984, 2987, (byte) -19 ); // Fill 3 of value (byte)
-19
+ Arrays.fill(CHARS, 2987, 2990, (byte) 33 ); // Fill 3 of value (byte)
33
+ Arrays.fill(CHARS, 2990, 2998, (byte) -19 ); // Fill 8 of value (byte)
-19
+ CHARS[2998] = 33;
+ Arrays.fill(CHARS, 2999, 3002, (byte) -19 ); // Fill 3 of value (byte)
-19
+ Arrays.fill(CHARS, 3002, 3006, (byte) 33 ); // Fill 4 of value (byte)
33
+ Arrays.fill(CHARS, 3006, 3011, (byte) -87 ); // Fill 5 of value (byte)
-87
+ Arrays.fill(CHARS, 3011, 3014, (byte) 33 ); // Fill 3 of value (byte)
33
+ Arrays.fill(CHARS, 3014, 3017, (byte) -87 ); // Fill 3 of value (byte)
-87
+ CHARS[3017] = 33;
+ Arrays.fill(CHARS, 3018, 3022, (byte) -87 ); // Fill 4 of value (byte)
-87
+ Arrays.fill(CHARS, 3022, 3031, (byte) 33 ); // Fill 9 of value (byte)
33
+ CHARS[3031] = -87;
+ Arrays.fill(CHARS, 3032, 3047, (byte) 33 ); // Fill 15 of value (byte)
33
+ Arrays.fill(CHARS, 3047, 3056, (byte) -87 ); // Fill 9 of value (byte)
-87
+ Arrays.fill(CHARS, 3056, 3073, (byte) 33 ); // Fill 17 of value (byte)
33
+ Arrays.fill(CHARS, 3073, 3076, (byte) -87 ); // Fill 3 of value (byte)
-87
+ CHARS[3076] = 33;
+ Arrays.fill(CHARS, 3077, 3085, (byte) -19 ); // Fill 8 of value (byte)
-19
+ CHARS[3085] = 33;
+ Arrays.fill(CHARS, 3086, 3089, (byte) -19 ); // Fill 3 of value (byte)
-19
+ CHARS[3089] = 33;
+ Arrays.fill(CHARS, 3090, 3113, (byte) -19 ); // Fill 23 of value
(byte) -19
+ CHARS[3113] = 33;
+ Arrays.fill(CHARS, 3114, 3124, (byte) -19 ); // Fill 10 of value
(byte) -19
+ CHARS[3124] = 33;
+ Arrays.fill(CHARS, 3125, 3130, (byte) -19 ); // Fill 5 of value (byte)
-19
+ Arrays.fill(CHARS, 3130, 3134, (byte) 33 ); // Fill 4 of value (byte)
33
+ Arrays.fill(CHARS, 3134, 3141, (byte) -87 ); // Fill 7 of value (byte)
-87
+ CHARS[3141] = 33;
+ Arrays.fill(CHARS, 3142, 3145, (byte) -87 ); // Fill 3 of value (byte)
-87
+ CHARS[3145] = 33;
+ Arrays.fill(CHARS, 3146, 3150, (byte) -87 ); // Fill 4 of value (byte)
-87
+ Arrays.fill(CHARS, 3150, 3157, (byte) 33 ); // Fill 7 of value (byte)
33
+ Arrays.fill(CHARS, 3157, 3159, (byte) -87 ); // Fill 2 of value (byte)
-87
+ Arrays.fill(CHARS, 3159, 3168, (byte) 33 ); // Fill 9 of value (byte)
33
+ Arrays.fill(CHARS, 3168, 3170, (byte) -19 ); // Fill 2 of value (byte)
-19
+ Arrays.fill(CHARS, 3170, 3174, (byte) 33 ); // Fill 4 of value (byte)
33
+ Arrays.fill(CHARS, 3174, 3184, (byte) -87 ); // Fill 10 of value
(byte) -87
+ Arrays.fill(CHARS, 3184, 3202, (byte) 33 ); // Fill 18 of value (byte)
33
+ Arrays.fill(CHARS, 3202, 3204, (byte) -87 ); // Fill 2 of value (byte)
-87
+ CHARS[3204] = 33;
+ Arrays.fill(CHARS, 3205, 3213, (byte) -19 ); // Fill 8 of value (byte)
-19
+ CHARS[3213] = 33;
+ Arrays.fill(CHARS, 3214, 3217, (byte) -19 ); // Fill 3 of value (byte)
-19
+ CHARS[3217] = 33;
+ Arrays.fill(CHARS, 3218, 3241, (byte) -19 ); // Fill 23 of value
(byte) -19
+ CHARS[3241] = 33;
+ Arrays.fill(CHARS, 3242, 3252, (byte) -19 ); // Fill 10 of value
(byte) -19
+ CHARS[3252] = 33;
+ Arrays.fill(CHARS, 3253, 3258, (byte) -19 ); // Fill 5 of value (byte)
-19
+ Arrays.fill(CHARS, 3258, 3262, (byte) 33 ); // Fill 4 of value (byte)
33
+ Arrays.fill(CHARS, 3262, 3269, (byte) -87 ); // Fill 7 of value (byte)
-87
+ CHARS[3269] = 33;
+ Arrays.fill(CHARS, 3270, 3273, (byte) -87 ); // Fill 3 of value (byte)
-87
+ CHARS[3273] = 33;
+ Arrays.fill(CHARS, 3274, 3278, (byte) -87 ); // Fill 4 of value (byte)
-87
+ Arrays.fill(CHARS, 3278, 3285, (byte) 33 ); // Fill 7 of value (byte)
33
+ Arrays.fill(CHARS, 3285, 3287, (byte) -87 ); // Fill 2 of value (byte)
-87
+ Arrays.fill(CHARS, 3287, 3294, (byte) 33 ); // Fill 7 of value (byte)
33
+ CHARS[3294] = -19;
+ CHARS[3295] = 33;
+ Arrays.fill(CHARS, 3296, 3298, (byte) -19 ); // Fill 2 of value (byte)
-19
+ Arrays.fill(CHARS, 3298, 3302, (byte) 33 ); // Fill 4 of value (byte)
33
+ Arrays.fill(CHARS, 3302, 3312, (byte) -87 ); // Fill 10 of value
(byte) -87
+ Arrays.fill(CHARS, 3312, 3330, (byte) 33 ); // Fill 18 of value (byte)
33
+ Arrays.fill(CHARS, 3330, 3332, (byte) -87 ); // Fill 2 of value (byte)
-87
+ CHARS[3332] = 33;
+ Arrays.fill(CHARS, 3333, 3341, (byte) -19 ); // Fill 8 of value (byte)
-19
+ CHARS[3341] = 33;
+ Arrays.fill(CHARS, 3342, 3345, (byte) -19 ); // Fill 3 of value (byte)
-19
+ CHARS[3345] = 33;
+ Arrays.fill(CHARS, 3346, 3369, (byte) -19 ); // Fill 23 of value
(byte) -19
+ CHARS[3369] = 33;
+ Arrays.fill(CHARS, 3370, 3386, (byte) -19 ); // Fill 16 of value
(byte) -19
+ Arrays.fill(CHARS, 3386, 3390, (byte) 33 ); // Fill 4 of value (byte)
33
+ Arrays.fill(CHARS, 3390, 3396, (byte) -87 ); // Fill 6 of value (byte)
-87
+ Arrays.fill(CHARS, 3396, 3398, (byte) 33 ); // Fill 2 of value (byte)
33
+ Arrays.fill(CHARS, 3398, 3401, (byte) -87 ); // Fill 3 of value (byte)
-87
+ CHARS[3401] = 33;
+ Arrays.fill(CHARS, 3402, 3406, (byte) -87 ); // Fill 4 of value (byte)
-87
+ Arrays.fill(CHARS, 3406, 3415, (byte) 33 ); // Fill 9 of value (byte)
33
+ CHARS[3415] = -87;
+ Arrays.fill(CHARS, 3416, 3424, (byte) 33 ); // Fill 8 of value (byte)
33
+ Arrays.fill(CHARS, 3424, 3426, (byte) -19 ); // Fill 2 of value (byte)
-19
+ Arrays.fill(CHARS, 3426, 3430, (byte) 33 ); // Fill 4 of value (byte)
33
+ Arrays.fill(CHARS, 3430, 3440, (byte) -87 ); // Fill 10 of value
(byte) -87
+ Arrays.fill(CHARS, 3440, 3585, (byte) 33 ); // Fill 145 of value
(byte) 33
+ Arrays.fill(CHARS, 3585, 3631, (byte) -19 ); // Fill 46 of value
(byte) -19
+ CHARS[3631] = 33;
+ CHARS[3632] = -19;
+ CHARS[3633] = -87;
+ Arrays.fill(CHARS, 3634, 3636, (byte) -19 ); // Fill 2 of value (byte)
-19
+ Arrays.fill(CHARS, 3636, 3643, (byte) -87 ); // Fill 7 of value (byte)
-87
+ Arrays.fill(CHARS, 3643, 3648, (byte) 33 ); // Fill 5 of value (byte)
33
+ Arrays.fill(CHARS, 3648, 3654, (byte) -19 ); // Fill 6 of value (byte)
-19
+ Arrays.fill(CHARS, 3654, 3663, (byte) -87 ); // Fill 9 of value (byte)
-87
+ CHARS[3663] = 33;
+ Arrays.fill(CHARS, 3664, 3674, (byte) -87 ); // Fill 10 of value
(byte) -87
+ Arrays.fill(CHARS, 3674, 3713, (byte) 33 ); // Fill 39 of value (byte)
33
+ Arrays.fill(CHARS, 3713, 3715, (byte) -19 ); // Fill 2 of value (byte)
-19
+ CHARS[3715] = 33;
+ CHARS[3716] = -19;
+ Arrays.fill(CHARS, 3717, 3719, (byte) 33 ); // Fill 2 of value (byte)
33
+ Arrays.fill(CHARS, 3719, 3721, (byte) -19 ); // Fill 2 of value (byte)
-19
+ CHARS[3721] = 33;
+ CHARS[3722] = -19;
+ Arrays.fill(CHARS, 3723, 3725, (byte) 33 ); // Fill 2 of value (byte)
33
+ CHARS[3725] = -19;
+ Arrays.fill(CHARS, 3726, 3732, (byte) 33 ); // Fill 6 of value (byte)
33
+ Arrays.fill(CHARS, 3732, 3736, (byte) -19 ); // Fill 4 of value (byte)
-19
+ CHARS[3736] = 33;
+ Arrays.fill(CHARS, 3737, 3744, (byte) -19 ); // Fill 7 of value (byte)
-19
+ CHARS[3744] = 33;
+ Arrays.fill(CHARS, 3745, 3748, (byte) -19 ); // Fill 3 of value (byte)
-19
+ CHARS[3748] = 33;
+ CHARS[3749] = -19;
+ CHARS[3750] = 33;
+ CHARS[3751] = -19;
+ Arrays.fill(CHARS, 3752, 3754, (byte) 33 ); // Fill 2 of value (byte)
33
+ Arrays.fill(CHARS, 3754, 3756, (byte) -19 ); // Fill 2 of value (byte)
-19
+ CHARS[3756] = 33;
+ Arrays.fill(CHARS, 3757, 3759, (byte) -19 ); // Fill 2 of value (byte)
-19
+ CHARS[3759] = 33;
+ CHARS[3760] = -19;
+ CHARS[3761] = -87;
+ Arrays.fill(CHARS, 3762, 3764, (byte) -19 ); // Fill 2 of value (byte)
-19
+ Arrays.fill(CHARS, 3764, 3770, (byte) -87 ); // Fill 6 of value (byte)
-87
+ CHARS[3770] = 33;
+ Arrays.fill(CHARS, 3771, 3773, (byte) -87 ); // Fill 2 of value (byte)
-87
+ CHARS[3773] = -19;
+ Arrays.fill(CHARS, 3774, 3776, (byte) 33 ); // Fill 2 of value (byte)
33
+ Arrays.fill(CHARS, 3776, 3781, (byte) -19 ); // Fill 5 of value (byte)
-19
+ CHARS[3781] = 33;
+ CHARS[3782] = -87;
+ CHARS[3783] = 33;
+ Arrays.fill(CHARS, 3784, 3790, (byte) -87 ); // Fill 6 of value (byte)
-87
+ Arrays.fill(CHARS, 3790, 3792, (byte) 33 ); // Fill 2 of value (byte)
33
+ Arrays.fill(CHARS, 3792, 3802, (byte) -87 ); // Fill 10 of value
(byte) -87
+ Arrays.fill(CHARS, 3802, 3864, (byte) 33 ); // Fill 62 of value (byte)
33
+ Arrays.fill(CHARS, 3864, 3866, (byte) -87 ); // Fill 2 of value (byte)
-87
+ Arrays.fill(CHARS, 3866, 3872, (byte) 33 ); // Fill 6 of value (byte)
33
+ Arrays.fill(CHARS, 3872, 3882, (byte) -87 ); // Fill 10 of value
(byte) -87
+ Arrays.fill(CHARS, 3882, 3893, (byte) 33 ); // Fill 11 of value (byte)
33
+ CHARS[3893] = -87;
+ CHARS[3894] = 33;
+ CHARS[3895] = -87;
+ CHARS[3896] = 33;
+ CHARS[3897] = -87;
+ Arrays.fill(CHARS, 3898, 3902, (byte) 33 ); // Fill 4 of value (byte)
33
+ Arrays.fill(CHARS, 3902, 3904, (byte) -87 ); // Fill 2 of value (byte)
-87
+ Arrays.fill(CHARS, 3904, 3912, (byte) -19 ); // Fill 8 of value (byte)
-19
+ CHARS[3912] = 33;
+ Arrays.fill(CHARS, 3913, 3946, (byte) -19 ); // Fill 33 of value
(byte) -19
+ Arrays.fill(CHARS, 3946, 3953, (byte) 33 ); // Fill 7 of value (byte)
33
+ Arrays.fill(CHARS, 3953, 3973, (byte) -87 ); // Fill 20 of value
(byte) -87
+ CHARS[3973] = 33;
+ Arrays.fill(CHARS, 3974, 3980, (byte) -87 ); // Fill 6 of value (byte)
-87
+ Arrays.fill(CHARS, 3980, 3984, (byte) 33 ); // Fill 4 of value (byte)
33
+ Arrays.fill(CHARS, 3984, 3990, (byte) -87 ); // Fill 6 of value (byte)
-87
+ CHARS[3990] = 33;
+ CHARS[3991] = -87;
+ CHARS[3992] = 33;
+ Arrays.fill(CHARS, 3993, 4014, (byte) -87 ); // Fill 21 of value
(byte) -87
+ Arrays.fill(CHARS, 4014, 4017, (byte) 33 ); // Fill 3 of value (byte)
33
+ Arrays.fill(CHARS, 4017, 4024, (byte) -87 ); // Fill 7 of value (byte)
-87
+ CHARS[4024] = 33;
+ CHARS[4025] = -87;
+ Arrays.fill(CHARS, 4026, 4256, (byte) 33 ); // Fill 230 of value
(byte) 33
+ Arrays.fill(CHARS, 4256, 4294, (byte) -19 ); // Fill 38 of value
(byte) -19
+ Arrays.fill(CHARS, 4294, 4304, (byte) 33 ); // Fill 10 of value (byte)
33
+ Arrays.fill(CHARS, 4304, 4343, (byte) -19 ); // Fill 39 of value
(byte) -19
+ Arrays.fill(CHARS, 4343, 4352, (byte) 33 ); // Fill 9 of value (byte)
33
+ CHARS[4352] = -19;
+ CHARS[4353] = 33;
+ Arrays.fill(CHARS, 4354, 4356, (byte) -19 ); // Fill 2 of value (byte)
-19
+ CHARS[4356] = 33;
+ Arrays.fill(CHARS, 4357, 4360, (byte) -19 ); // Fill 3 of value (byte)
-19
+ CHARS[4360] = 33;
+ CHARS[4361] = -19;
+ CHARS[4362] = 33;
+ Arrays.fill(CHARS, 4363, 4365, (byte) -19 ); // Fill 2 of value (byte)
-19
+ CHARS[4365] = 33;
+ Arrays.fill(CHARS, 4366, 4371, (byte) -19 ); // Fill 5 of value (byte)
-19
+ Arrays.fill(CHARS, 4371, 4412, (byte) 33 ); // Fill 41 of value (byte)
33
+ CHARS[4412] = -19;
+ CHARS[4413] = 33;
+ CHARS[4414] = -19;
+ CHARS[4415] = 33;
+ CHARS[4416] = -19;
+ Arrays.fill(CHARS, 4417, 4428, (byte) 33 ); // Fill 11 of value (byte)
33
+ CHARS[4428] = -19;
+ CHARS[4429] = 33;
+ CHARS[4430] = -19;
+ CHARS[4431] = 33;
+ CHARS[4432] = -19;
+ Arrays.fill(CHARS, 4433, 4436, (byte) 33 ); // Fill 3 of value (byte)
33
+ Arrays.fill(CHARS, 4436, 4438, (byte) -19 ); // Fill 2 of value (byte)
-19
+ Arrays.fill(CHARS, 4438, 4441, (byte) 33 ); // Fill 3 of value (byte)
33
+ CHARS[4441] = -19;
+ Arrays.fill(CHARS, 4442, 4447, (byte) 33 ); // Fill 5 of value (byte)
33
+ Arrays.fill(CHARS, 4447, 4450, (byte) -19 ); // Fill 3 of value (byte)
-19
+ CHARS[4450] = 33;
+ CHARS[4451] = -19;
+ CHARS[4452] = 33;
+ CHARS[4453] = -19;
+ CHARS[4454] = 33;
+ CHARS[4455] = -19;
+ CHARS[4456] = 33;
+ CHARS[4457] = -19;
+ Arrays.fill(CHARS, 4458, 4461, (byte) 33 ); // Fill 3 of value (byte)
33
+ Arrays.fill(CHARS, 4461, 4463, (byte) -19 ); // Fill 2 of value (byte)
-19
+ Arrays.fill(CHARS, 4463, 4466, (byte) 33 ); // Fill 3 of value (byte)
33
+ Arrays.fill(CHARS, 4466, 4468, (byte) -19 ); // Fill 2 of value (byte)
-19
+ CHARS[4468] = 33;
+ CHARS[4469] = -19;
+ Arrays.fill(CHARS, 4470, 4510, (byte) 33 ); // Fill 40 of value (byte)
33
+ CHARS[4510] = -19;
+ Arrays.fill(CHARS, 4511, 4520, (byte) 33 ); // Fill 9 of value (byte)
33
+ CHARS[4520] = -19;
+ Arrays.fill(CHARS, 4521, 4523, (byte) 33 ); // Fill 2 of value (byte)
33
+ CHARS[4523] = -19;
+ Arrays.fill(CHARS, 4524, 4526, (byte) 33 ); // Fill 2 of value (byte)
33
+ Arrays.fill(CHARS, 4526, 4528, (byte) -19 ); // Fill 2 of value (byte)
-19
+ Arrays.fill(CHARS, 4528, 4535, (byte) 33 ); // Fill 7 of value (byte)
33
+ Arrays.fill(CHARS, 4535, 4537, (byte) -19 ); // Fill 2 of value (byte)
-19
+ CHARS[4537] = 33;
+ CHARS[4538] = -19;
+ CHARS[4539] = 33;
+ Arrays.fill(CHARS, 4540, 4547, (byte) -19 ); // Fill 7 of value (byte)
-19
+ Arrays.fill(CHARS, 4547, 4587, (byte) 33 ); // Fill 40 of value (byte)
33
+ CHARS[4587] = -19;
+ Arrays.fill(CHARS, 4588, 4592, (byte) 33 ); // Fill 4 of value (byte)
33
+ CHARS[4592] = -19;
+ Arrays.fill(CHARS, 4593, 4601, (byte) 33 ); // Fill 8 of value (byte)
33
+ CHARS[4601] = -19;
+ Arrays.fill(CHARS, 4602, 7680, (byte) 33 ); // Fill 3078 of value
(byte) 33
+ Arrays.fill(CHARS, 7680, 7836, (byte) -19 ); // Fill 156 of value
(byte) -19
+ Arrays.fill(CHARS, 7836, 7840, (byte) 33 ); // Fill 4 of value (byte)
33
+ Arrays.fill(CHARS, 7840, 7930, (byte) -19 ); // Fill 90 of value
(byte) -19
+ Arrays.fill(CHARS, 7930, 7936, (byte) 33 ); // Fill 6 of value (byte)
33
+ Arrays.fill(CHARS, 7936, 7958, (byte) -19 ); // Fill 22 of value
(byte) -19
+ Arrays.fill(CHARS, 7958, 7960, (byte) 33 ); // Fill 2 of value (byte)
33
+ Arrays.fill(CHARS, 7960, 7966, (byte) -19 ); // Fill 6 of value (byte)
-19
+ Arrays.fill(CHARS, 7966, 7968, (byte) 33 ); // Fill 2 of value (byte)
33
+ Arrays.fill(CHARS, 7968, 8006, (byte) -19 ); // Fill 38 of value
(byte) -19
+ Arrays.fill(CHARS, 8006, 8008, (byte) 33 ); // Fill 2 of value (byte)
33
+ Arrays.fill(CHARS, 8008, 8014, (byte) -19 ); // Fill 6 of value (byte)
-19
+ Arrays.fill(CHARS, 8014, 8016, (byte) 33 ); // Fill 2 of value (byte)
33
+ Arrays.fill(CHARS, 8016, 8024, (byte) -19 ); // Fill 8 of value (byte)
-19
+ CHARS[8024] = 33;
+ CHARS[8025] = -19;
+ CHARS[8026] = 33;
+ CHARS[8027] = -19;
+ CHARS[8028] = 33;
+ CHARS[8029] = -19;
+ CHARS[8030] = 33;
+ Arrays.fill(CHARS, 8031, 8062, (byte) -19 ); // Fill 31 of value
(byte) -19
+ Arrays.fill(CHARS, 8062, 8064, (byte) 33 ); // Fill 2 of value (byte)
33
+ Arrays.fill(CHARS, 8064, 8117, (byte) -19 ); // Fill 53 of value
(byte) -19
+ CHARS[8117] = 33;
+ Arrays.fill(CHARS, 8118, 8125, (byte) -19 ); // Fill 7 of value (byte)
-19
+ CHARS[8125] = 33;
+ CHARS[8126] = -19;
+ Arrays.fill(CHARS, 8127, 8130, (byte) 33 ); // Fill 3 of value (byte)
33
+ Arrays.fill(CHARS, 8130, 8133, (byte) -19 ); // Fill 3 of value (byte)
-19
+ CHARS[8133] = 33;
+ Arrays.fill(CHARS, 8134, 8141, (byte) -19 ); // Fill 7 of value (byte)
-19
+ Arrays.fill(CHARS, 8141, 8144, (byte) 33 ); // Fill 3 of value (byte)
33
+ Arrays.fill(CHARS, 8144, 8148, (byte) -19 ); // Fill 4 of value (byte)
-19
+ Arrays.fill(CHARS, 8148, 8150, (byte) 33 ); // Fill 2 of value (byte)
33
+ Arrays.fill(CHARS, 8150, 8156, (byte) -19 ); // Fill 6 of value (byte)
-19
+ Arrays.fill(CHARS, 8156, 8160, (byte) 33 ); // Fill 4 of value (byte)
33
+ Arrays.fill(CHARS, 8160, 8173, (byte) -19 ); // Fill 13 of value
(byte) -19
+ Arrays.fill(CHARS, 8173, 8178, (byte) 33 ); // Fill 5 of value (byte)
33
+ Arrays.fill(CHARS, 8178, 8181, (byte) -19 ); // Fill 3 of value (byte)
-19
+ CHARS[8181] = 33;
+ Arrays.fill(CHARS, 8182, 8189, (byte) -19 ); // Fill 7 of value (byte)
-19
+ Arrays.fill(CHARS, 8189, 8400, (byte) 33 ); // Fill 211 of value
(byte) 33
+ Arrays.fill(CHARS, 8400, 8413, (byte) -87 ); // Fill 13 of value
(byte) -87
+ Arrays.fill(CHARS, 8413, 8417, (byte) 33 ); // Fill 4 of value (byte)
33
+ CHARS[8417] = -87;
+ Arrays.fill(CHARS, 8418, 8486, (byte) 33 ); // Fill 68 of value (byte)
33
+ CHARS[8486] = -19;
+ Arrays.fill(CHARS, 8487, 8490, (byte) 33 ); // Fill 3 of value (byte)
33
+ Arrays.fill(CHARS, 8490, 8492, (byte) -19 ); // Fill 2 of value (byte)
-19
+ Arrays.fill(CHARS, 8492, 8494, (byte) 33 ); // Fill 2 of value (byte)
33
+ CHARS[8494] = -19;
+ Arrays.fill(CHARS, 8495, 8576, (byte) 33 ); // Fill 81 of value (byte)
33
+ Arrays.fill(CHARS, 8576, 8579, (byte) -19 ); // Fill 3 of value (byte)
-19
+ Arrays.fill(CHARS, 8579, 12293, (byte) 33 ); // Fill 3714 of value
(byte) 33
+ CHARS[12293] = -87;
+ CHARS[12294] = 33;
+ CHARS[12295] = -19;
+ Arrays.fill(CHARS, 12296, 12321, (byte) 33 ); // Fill 25 of value
(byte) 33
+ Arrays.fill(CHARS, 12321, 12330, (byte) -19 ); // Fill 9 of value
(byte) -19
+ Arrays.fill(CHARS, 12330, 12336, (byte) -87 ); // Fill 6 of value
(byte) -87
+ CHARS[12336] = 33;
+ Arrays.fill(CHARS, 12337, 12342, (byte) -87 ); // Fill 5 of value
(byte) -87
+ Arrays.fill(CHARS, 12342, 12353, (byte) 33 ); // Fill 11 of value
(byte) 33
+ Arrays.fill(CHARS, 12353, 12437, (byte) -19 ); // Fill 84 of value
(byte) -19
+ Arrays.fill(CHARS, 12437, 12441, (byte) 33 ); // Fill 4 of value
(byte) 33
+ Arrays.fill(CHARS, 12441, 12443, (byte) -87 ); // Fill 2 of value
(byte) -87
+ Arrays.fill(CHARS, 12443, 12445, (byte) 33 ); // Fill 2 of value
(byte) 33
+ Arrays.fill(CHARS, 12445, 12447, (byte) -87 ); // Fill 2 of value
(byte) -87
+ Arrays.fill(CHARS, 12447, 12449, (byte) 33 ); // Fill 2 of value
(byte) 33
+ Arrays.fill(CHARS, 12449, 12539, (byte) -19 ); // Fill 90 of value
(byte) -19
+ CHARS[12539] = 33;
+ Arrays.fill(CHARS, 12540, 12543, (byte) -87 ); // Fill 3 of value
(byte) -87
+ Arrays.fill(CHARS, 12543, 12549, (byte) 33 ); // Fill 6 of value
(byte) 33
+ Arrays.fill(CHARS, 12549, 12589, (byte) -19 ); // Fill 40 of value
(byte) -19
+ Arrays.fill(CHARS, 12589, 19968, (byte) 33 ); // Fill 7379 of value
(byte) 33
+ Arrays.fill(CHARS, 19968, 40870, (byte) -19 ); // Fill 20902 of value
(byte) -19
+ Arrays.fill(CHARS, 40870, 44032, (byte) 33 ); // Fill 3162 of value
(byte) 33
+ Arrays.fill(CHARS, 44032, 55204, (byte) -19 ); // Fill 11172 of value
(byte) -19
+ Arrays.fill(CHARS, 55204, 55296, (byte) 33 ); // Fill 92 of value
(byte) 33
+ Arrays.fill(CHARS, 57344, 65534, (byte) 33 ); // Fill 8190 of value
(byte) 33
+
+ } // <clinit>()
+
+ //
+ // Public static methods
+ //
+
+ /**
+ * Returns true if the specified character is a supplemental character.
+ *
+ * @param c The character to check.
+ */
+ public static boolean isSupplemental(int c) {
+ return (c >= 0x10000 && c <= 0x10FFFF);
+ }
+
+ /**
+ * Returns true the supplemental character corresponding to the given
+ * surrogates.
+ *
+ * @param h The high surrogate.
+ * @param l The low surrogate.
+ */
+ public static int supplemental(char h, char l) {
+ return (h - 0xD800) * 0x400 + (l - 0xDC00) + 0x10000;
+ }
+
+ /**
+ * Returns the high surrogate of a supplemental character
+ *
+ * @param c The supplemental character to "split".
+ */
+ public static char highSurrogate(int c) {
+ return (char) (((c - 0x00010000) >> 10) + 0xD800);
+ }
+
+ /**
+ * Returns the low surrogate of a supplemental character
+ *
+ * @param c The supplemental character to "split".
+ */
+ public static char lowSurrogate(int c) {
+ return (char) (((c - 0x00010000) & 0x3FF) + 0xDC00);
+ }
+
+ /**
+ * Returns whether the given character is a high surrogate
+ *
+ * @param c The character to check.
+ */
+ public static boolean isHighSurrogate(int c) {
+ return (0xD800 <= c && c <= 0xDBFF);
+ }
+
+ /**
+ * Returns whether the given character is a low surrogate
+ *
+ * @param c The character to check.
+ */
+ public static boolean isLowSurrogate(int c) {
+ return (0xDC00 <= c && c <= 0xDFFF);
+ }
+
+
+ /**
+ * Returns true if the specified character is valid. This method
+ * also checks the surrogate character range from 0x10000 to 0x10FFFF.
+ * <p>
+ * If the program chooses to apply the mask directly to the
+ * <code>CHARS</code> array, then they are responsible for checking
+ * the surrogate character range.
+ *
+ * @param c The character to check.
+ */
+ public static boolean isValid(int c) {
+ return (c < 0x10000 && (CHARS[c] & MASK_VALID) != 0) ||
+ (0x10000 <= c && c <= 0x10FFFF);
+ } // isValid(int):boolean
+
+ /**
+ * Returns true if the specified character is invalid.
+ *
+ * @param c The character to check.
+ */
+ public static boolean isInvalid(int c) {
+ return !isValid(c);
+ } // isInvalid(int):boolean
+
+ /**
+ * Returns true if the specified character can be considered content.
+ *
+ * @param c The character to check.
+ */
+ public static boolean isContent(int c) {
+ return (c < 0x10000 && (CHARS[c] & MASK_CONTENT) != 0) ||
+ (0x10000 <= c && c <= 0x10FFFF);
+ } // isContent(int):boolean
+
+ /**
+ * Returns true if the specified character can be considered markup.
+ * Markup characters include '<', '&', and '%'.
+ *
+ * @param c The character to check.
+ */
+ public static boolean isMarkup(int c) {
+ return c == '<' || c == '&' || c == '%';
+ } // isMarkup(int):boolean
+
+ /**
+ * Returns true if the specified character is a space character
+ * as defined by production [3] in the XML 1.0 specification.
+ *
+ * @param c The character to check.
+ */
+ public static boolean isSpace(int c) {
+ return c <= 0x20 && (CHARS[c] & MASK_SPACE) != 0;
+ } // isSpace(int):boolean
+
+ /**
+ * Returns true if the specified character is a valid name start
+ * character as defined by production [5] in the XML 1.0
+ * specification.
+ *
+ * @param c The character to check.
+ */
+ public static boolean isNameStart(int c) {
+ return c < 0x10000 && (CHARS[c] & MASK_NAME_START) != 0;
+ } // isNameStart(int):boolean
+
+ /**
+ * Returns true if the specified character is a valid name
+ * character as defined by production [4] in the XML 1.0
+ * specification.
+ *
+ * @param c The character to check.
+ */
+ public static boolean isName(int c) {
+ return c < 0x10000 && (CHARS[c] & MASK_NAME) != 0;
+ } // isName(int):boolean
+
+ /**
+ * Returns true if the specified character is a valid NCName start
+ * character as defined by production [4] in Namespaces in XML
+ * recommendation.
+ *
+ * @param c The character to check.
+ */
+ public static boolean isNCNameStart(int c) {
+ return c < 0x10000 && (CHARS[c] & MASK_NCNAME_START) != 0;
+ } // isNCNameStart(int):boolean
+
+ /**
+ * Returns true if the specified character is a valid NCName
+ * character as defined by production [5] in Namespaces in XML
+ * recommendation.
+ *
+ * @param c The character to check.
+ */
+ public static boolean isNCName(int c) {
+ return c < 0x10000 && (CHARS[c] & MASK_NCNAME) != 0;
+ } // isNCName(int):boolean
+
+ /**
+ * Returns true if the specified character is a valid Pubid
+ * character as defined by production [13] in the XML 1.0
+ * specification.
+ *
+ * @param c The character to check.
+ */
+ public static boolean isPubid(int c) {
+ return c < 0x10000 && (CHARS[c] & MASK_PUBID) != 0;
+ } // isPubid(int):boolean
+
+ /*
+ * [5] Name ::= (Letter | '_' | ':') (NameChar)*
+ */
+ /**
+ * Check to see if a string is a valid Name according to [5]
+ * in the XML 1.0 Recommendation
+ *
+ * @param name string to check
+ * @return true if name is a valid Name
+ */
+ public static boolean isValidName(String name) {
+ final int length = name.length();
+ if (length == 0) {
+ return false;
+ }
+ char ch = name.charAt(0);
+ if (!isNameStart(ch)) {
+ return false;
+ }
+ for (int i = 1; i < length; ++i) {
+ ch = name.charAt(i);
+ if (!isName(ch)) {
+ return false;
+ }
+ }
+ return true;
+ } // isValidName(String):boolean
+
+ /*
+ * from the namespace rec
+ * [4] NCName ::= (Letter | '_') (NCNameChar)*
+ */
+ /**
+ * Check to see if a string is a valid NCName according to [4]
+ * from the XML Namespaces 1.0 Recommendation
+ *
+ * @param ncName string to check
+ * @return true if name is a valid NCName
+ */
+ public static boolean isValidNCName(String ncName) {
+ final int length = ncName.length();
+ if (length == 0) {
+ return false;
+ }
+ char ch = ncName.charAt(0);
+ if (!isNCNameStart(ch)) {
+ return false;
+ }
+ for (int i = 1; i < length; ++i) {
+ ch = ncName.charAt(i);
+ if (!isNCName(ch)) {
+ return false;
+ }
+ }
+ return true;
+ } // isValidNCName(String):boolean
+
+ /*
+ * [7] Nmtoken ::= (NameChar)+
+ */
+ /**
+ * Check to see if a string is a valid Nmtoken according to [7]
+ * in the XML 1.0 Recommendation
+ *
+ * @param nmtoken string to check
+ * @return true if nmtoken is a valid Nmtoken
+ */
+ public static boolean isValidNmtoken(String nmtoken) {
+ final int length = nmtoken.length();
+ if (length == 0) {
+ return false;
+ }
+ for (int i = 0; i < length; ++i) {
+ char ch = nmtoken.charAt(i);
+ if (!isName(ch)) {
+ return false;
+ }
+ }
+ return true;
+ } // isValidName(String):boolean
+
+
+
+
+
+ // encodings
+
+ /**
+ * Returns true if the encoding name is a valid IANA encoding.
+ * This method does not verify that there is a decoder available
+ * for this encoding, only that the characters are valid for an
+ * IANA encoding name.
+ *
+ * @param ianaEncoding The IANA encoding name.
+ */
+ public static boolean isValidIANAEncoding(String ianaEncoding) {
+ if (ianaEncoding != null) {
+ int length = ianaEncoding.length();
+ if (length > 0) {
+ char c = ianaEncoding.charAt(0);
+ if ((c >= 'A' && c <= 'Z') || (c >= 'a' && c <= 'z')) {
+ for (int i = 1; i < length; i++) {
+ c = ianaEncoding.charAt(i);
+ if ((c < 'A' || c > 'Z') && (c < 'a' || c > 'z') &&
+ (c < '0' || c > '9') && c != '.' && c != '_' &&
+ c != '-') {
+ return false;
+ }
+ }
+ return true;
+ }
+ }
+ }
+ return false;
+ } // isValidIANAEncoding(String):boolean
+
+ /**
+ * Returns true if the encoding name is a valid Java encoding.
+ * This method does not verify that there is a decoder available
+ * for this encoding, only that the characters are valid for an
+ * Java encoding name.
+ *
+ * @param javaEncoding The Java encoding name.
+ */
+ public static boolean isValidJavaEncoding(String javaEncoding) {
+ if (javaEncoding != null) {
+ int length = javaEncoding.length();
+ if (length > 0) {
+ for (int i = 1; i < length; i++) {
+ char c = javaEncoding.charAt(i);
+ if ((c < 'A' || c > 'Z') && (c < 'a' || c > 'z') &&
+ (c < '0' || c > '9') && c != '.' && c != '_' &&
+ c != '-') {
+ return false;
+ }
+ }
+ return true;
+ }
+ }
+ return false;
+ } // isValidIANAEncoding(String):boolean
+
+ // other methods
+
+ /**
+ * Trims space characters as defined by production [3] in
+ * the XML 1.0 specification from both ends of the given string.
+ *
+ * @param value the string to be trimmed
+ * @return the given string with the space characters trimmed
+ * from both ends
+ */
+ public static String trim(String value) {
+ int start;
+ int end;
+ final int lengthMinusOne = value.length() - 1;
+ for (start = 0; start <= lengthMinusOne; ++start) {
+ if (!isSpace(value.charAt(start))) {
+ break;
+ }
+ }
+ for (end = lengthMinusOne; end >= start; --end) {
+ if (!isSpace(value.charAt(end))) {
+ break;
+ }
+ }
+ if (start == 0 && end == lengthMinusOne) {
+ return value;
+ }
+ if (start > lengthMinusOne) {
+ return "";
+ }
+ return value.substring(start, end + 1);
+ } // trim(String):String
+
+} // class XMLChar
diff --git
a/apache-rat-core/src/main/java/org/apache/rat/report/xml/writer/XmlWriter.java
b/apache-rat-core/src/main/java/org/apache/rat/report/xml/writer/XmlWriter.java
index 128f32f6..3cb591ef 100644
---
a/apache-rat-core/src/main/java/org/apache/rat/report/xml/writer/XmlWriter.java
+++
b/apache-rat-core/src/main/java/org/apache/rat/report/xml/writer/XmlWriter.java
@@ -18,14 +18,22 @@
*/
package org.apache.rat.report.xml.writer;
+import java.io.ByteArrayOutputStream;
+import java.io.Closeable;
import java.io.IOException;
-import java.io.Writer;
import java.util.ArrayDeque;
-import java.util.Arrays;
import java.util.HashSet;
import java.util.Objects;
import java.util.Set;
+import javax.xml.transform.Transformer;
+import javax.xml.transform.TransformerException;
+import javax.xml.transform.dom.DOMSource;
+import javax.xml.transform.stream.StreamResult;
+
+import org.apache.rat.utils.StandardXmlFactory;
+import org.w3c.dom.Document;
+
/**
* <p>
* Lightweight {@link IXmlWriter} implementation.
@@ -39,369 +47,7 @@ import java.util.Set;
*/
@SuppressWarnings({"checkstyle:MagicNumber", "checkstyle:JavadocVariable"})
public final class XmlWriter implements IXmlWriter {
-
- private static final byte NAME_START_MASK = 1 << 1;
- private static final byte NAME_MASK = 1 << 2;
- private static final byte NAME_BODY_CHAR = NAME_MASK;
- private static final byte NAME_START_OR_BODY_CHAR = NAME_MASK |
NAME_START_MASK;
-
- private static final boolean[] ALLOWED_CHARACTERS = new boolean[1 << 16];
-
- static {
- Arrays.fill(ALLOWED_CHARACTERS, false);
- ALLOWED_CHARACTERS[0x9] = true;
- ALLOWED_CHARACTERS[0xA] = true;
- ALLOWED_CHARACTERS[0xD] = true;
- Arrays.fill(ALLOWED_CHARACTERS, 0x20, 0xD7FF, true);
- Arrays.fill(ALLOWED_CHARACTERS, 0xE000, 0xFFFD, true);
- }
-
- private static final byte[] CHARACTER_CODES = new byte[1 << 16];
-
- static {
- // Name ::= (Letter | '_' | ':') (NameChar)*
- CHARACTER_CODES['_'] = NAME_START_OR_BODY_CHAR;
- CHARACTER_CODES[':'] = NAME_START_OR_BODY_CHAR;
- // Letter ::= BaseChar | Ideographic
- // BaseChar
- Arrays.fill(CHARACTER_CODES, 0x0041, 0x005A, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0061, 0x007A, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x00C0, 0x00D6, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x00D8, 0x00F6, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x00F8, 0x00FF, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0100, 0x0131, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0134, 0x013E, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0141, 0x0148, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x014A, 0x017E, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0180, 0x01C3, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x01CD, 0x01F0, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x01F4, 0x01F5, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x01FA, 0x0217, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0250, 0x02A8, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x02BB, 0x02C1, NAME_START_OR_BODY_CHAR);
- CHARACTER_CODES[0x0386] = NAME_START_OR_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x0388, 0x038A, NAME_START_OR_BODY_CHAR);
- CHARACTER_CODES[0x038C] = NAME_START_OR_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x038E, 0x03A1, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x03A3, 0x03CE, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x03D0, 0x03D6, NAME_START_OR_BODY_CHAR);
- CHARACTER_CODES[0x03DA] = NAME_START_OR_BODY_CHAR;
- CHARACTER_CODES[0x03DC] = NAME_START_OR_BODY_CHAR;
- CHARACTER_CODES[0x03DE] = NAME_START_OR_BODY_CHAR;
- CHARACTER_CODES[0x03E0] = NAME_START_OR_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x03E2, 0x03F3, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0401, 0x040C, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x040E, 0x044F, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0451, 0x045C, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x045E, 0x0481, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0490, 0x04C4, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x04C7, 0x04C8, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x04CB, 0x04CC, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x04D0, 0x04EB, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x04EE, 0x04F5, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x04F8, 0x04F9, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0531, 0x0556, NAME_START_OR_BODY_CHAR);
- CHARACTER_CODES[0x0559] = NAME_START_OR_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x0561, 0x0586, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x05D0, 0x05EA, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x05F0, 0x05F2, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0621, 0x063A, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0641, 0x064A, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0671, 0x06B7, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x06BA, 0x06BE, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x06C0, 0x06CE, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x06D0, 0x06D3, NAME_START_OR_BODY_CHAR);
- CHARACTER_CODES[0x06D5] = NAME_START_OR_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x06E5, 0x06E6, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0905, 0x0939, NAME_START_OR_BODY_CHAR);
- CHARACTER_CODES[0x093D] = NAME_START_OR_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x0958, 0x0961, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0985, 0x098C, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x098F, 0x0990, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0993, 0x09A8, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x09AA, 0x09B0, NAME_START_OR_BODY_CHAR);
- CHARACTER_CODES[0x09B2] = NAME_START_OR_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x09B6, 0x09B9, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x09DC, 0x09DD, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x09DF, 0x09E1, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x09F0, 0x09F1, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0A05, 0x0A0A, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0A0F, 0x0A10, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0A13, 0x0A28, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0A2A, 0x0A30, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0A32, 0x0A33, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0A35, 0x0A36, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0A38, 0x0A39, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0A59, 0x0A5C, NAME_START_OR_BODY_CHAR);
- CHARACTER_CODES[0x0A5E] = NAME_START_OR_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x0A72, 0x0A74, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0A85, 0x0A8B, NAME_START_OR_BODY_CHAR);
- CHARACTER_CODES[0x0A8D] = NAME_START_OR_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x0A8F, 0x0A91, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0A93, 0x0AA8, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0AAA, 0x0AB0, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0AB2, 0x0AB3, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0AB5, 0x0AB9, NAME_START_OR_BODY_CHAR);
- CHARACTER_CODES[0x0ABD] = NAME_START_OR_BODY_CHAR;
- CHARACTER_CODES[0x0AE0] = NAME_START_OR_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x0B05, 0x0B0C, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0B0F, 0x0B10, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0B13, 0x0B28, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0B2A, 0x0B30, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0B32, 0x0B33, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0B36, 0x0B39, NAME_START_OR_BODY_CHAR);
- CHARACTER_CODES[0x0B3D] = NAME_START_OR_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x0B5C, 0x0B5D, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0B5F, 0x0B61, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0B85, 0x0B8A, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0B8E, 0x0B90, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0B92, 0x0B95, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0B99, 0x0B9A, NAME_START_OR_BODY_CHAR);
- CHARACTER_CODES[0x0B9C] = NAME_START_OR_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x0B9E, 0x0B9F, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0BA3, 0x0BA4, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0BA8, 0x0BAA, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0BAE, 0x0BB5, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0BB7, 0x0BB9, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0C05, 0x0C0C, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0C0E, 0x0C10, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0C12, 0x0C28, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0C2A, 0x0C33, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0C35, 0x0C39, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0C60, 0x0C61, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0C85, 0x0C8C, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0C8E, 0x0C90, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0C92, 0x0CA8, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0CAA, 0x0CB3, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0CB5, 0x0CB9, NAME_START_OR_BODY_CHAR);
- CHARACTER_CODES[0x0CDE] = NAME_START_OR_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x0CE0, 0x0CE1, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0D05, 0x0D0C, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0D0E, 0x0D10, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0D12, 0x0D28, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0D2A, 0x0D39, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0D60, 0x0D61, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0E01, 0x0E2E, NAME_START_OR_BODY_CHAR);
- CHARACTER_CODES[0x0E30] = NAME_START_OR_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x0E32, 0x0E33, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0E40, 0x0E45, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0E81, 0x0E82, NAME_START_OR_BODY_CHAR);
- CHARACTER_CODES[0x0E84] = NAME_START_OR_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x0E87, 0x0E88, NAME_START_OR_BODY_CHAR);
- CHARACTER_CODES[0x0E8A] = NAME_START_OR_BODY_CHAR;
- CHARACTER_CODES[0x0E8D] = NAME_START_OR_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x0E94, 0x0E97, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0E99, 0x0E9F, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0EA1, 0x0EA3, NAME_START_OR_BODY_CHAR);
- CHARACTER_CODES[0x0EA5] = NAME_START_OR_BODY_CHAR;
- CHARACTER_CODES[0x0EA7] = NAME_START_OR_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x0EAA, 0x0EAB, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0EAD, 0x0EAE, NAME_START_OR_BODY_CHAR);
- CHARACTER_CODES[0x0EB0] = NAME_START_OR_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x0EB2, 0x0EB3, NAME_START_OR_BODY_CHAR);
- CHARACTER_CODES[0x0EBD] = NAME_START_OR_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x0EC0, 0x0EC4, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0F40, 0x0F47, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0F49, 0x0F69, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x10A0, 0x10C5, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x10D0, 0x10F6, NAME_START_OR_BODY_CHAR);
- CHARACTER_CODES[0x1100] = NAME_START_OR_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x1102, 0x1103, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x1105, 0x1107, NAME_START_OR_BODY_CHAR);
- CHARACTER_CODES[0x1109] = NAME_START_OR_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x110B, 0x110C, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x110E, 0x1112, NAME_START_OR_BODY_CHAR);
- CHARACTER_CODES[0x113C] = NAME_START_OR_BODY_CHAR;
- CHARACTER_CODES[0x113E] = NAME_START_OR_BODY_CHAR;
- CHARACTER_CODES[0x1140] = NAME_START_OR_BODY_CHAR;
- CHARACTER_CODES[0x114C] = NAME_START_OR_BODY_CHAR;
- CHARACTER_CODES[0x114E] = NAME_START_OR_BODY_CHAR;
- CHARACTER_CODES[0x1150] = NAME_START_OR_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x1154, 0x1155, NAME_START_OR_BODY_CHAR);
- CHARACTER_CODES[0x1159] = NAME_START_OR_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x115F, 0x1161, NAME_START_OR_BODY_CHAR);
- CHARACTER_CODES[0x1163] = NAME_START_OR_BODY_CHAR;
- CHARACTER_CODES[0x1165] = NAME_START_OR_BODY_CHAR;
- CHARACTER_CODES[0x1167] = NAME_START_OR_BODY_CHAR;
- CHARACTER_CODES[0x1169] = NAME_START_OR_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x116D, 0x116E, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x1172, 0x1173, NAME_START_OR_BODY_CHAR);
- CHARACTER_CODES[0x1175] = NAME_START_OR_BODY_CHAR;
- CHARACTER_CODES[0x119E] = NAME_START_OR_BODY_CHAR;
- CHARACTER_CODES[0x11A8] = NAME_START_OR_BODY_CHAR;
- CHARACTER_CODES[0x11AB] = NAME_START_OR_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x11AE, 0x11AF, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x11B7, 0x11B8, NAME_START_OR_BODY_CHAR);
- CHARACTER_CODES[0x11BA] = NAME_START_OR_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x11BC, 0x11C2, NAME_START_OR_BODY_CHAR);
- CHARACTER_CODES[0x11EB] = NAME_START_OR_BODY_CHAR;
- CHARACTER_CODES[0x11F0] = NAME_START_OR_BODY_CHAR;
- CHARACTER_CODES[0x11F9] = NAME_START_OR_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x1E00, 0x1E9B, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x1EA0, 0x1EF9, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x1F00, 0x1F15, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x1F18, 0x1F1D, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x1F20, 0x1F45, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x1F48, 0x1F4D, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x1F50, 0x1F57, NAME_START_OR_BODY_CHAR);
- CHARACTER_CODES[0x1F59] = NAME_START_OR_BODY_CHAR;
- CHARACTER_CODES[0x1F5B] = NAME_START_OR_BODY_CHAR;
- CHARACTER_CODES[0x1F5D] = NAME_START_OR_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x1F5F, 0x1F7D, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x1F80, 0x1FB4, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x1FB6, 0x1FBC, NAME_START_OR_BODY_CHAR);
- CHARACTER_CODES[0x1FBE] = NAME_START_OR_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x1FC2, 0x1FC4, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x1FC6, 0x1FCC, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x1FD0, 0x1FD3, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x1FD6, 0x1FDB, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x1FE0, 0x1FEC, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x1FF2, 0x1FF4, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x1FF6, 0x1FFC, NAME_START_OR_BODY_CHAR);
- CHARACTER_CODES[0x2126] = NAME_START_OR_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x212A, 0x212B, NAME_START_OR_BODY_CHAR);
- CHARACTER_CODES[0x212E] = NAME_START_OR_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x2180, 0x2182, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x3041, 0x3094, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x30A1, 0x30FA, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x3105, 0x312C, NAME_START_OR_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0xAC00, 0xD7A3, NAME_START_OR_BODY_CHAR);
- // Ideographic
- Arrays.fill(CHARACTER_CODES, 0x4E00, 0x9FA5, NAME_START_OR_BODY_CHAR);
- CHARACTER_CODES[0x3007] = NAME_START_OR_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x3021, 0x3029, NAME_START_OR_BODY_CHAR);
- // NameChar ::= Letter | Digit | '.' | '-' | '_' | ':' | CombiningChar
|
- // Extender
- CHARACTER_CODES['.'] = NAME_BODY_CHAR;
- CHARACTER_CODES['-'] = NAME_BODY_CHAR;
- // CombiningChar
- Arrays.fill(CHARACTER_CODES, 0x0300, 0x0345, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0360, 0x0361, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0483, 0x0486, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0591, 0x05A1, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x05A3, 0x05B9, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x05BB, 0x05BD, NAME_BODY_CHAR);
- CHARACTER_CODES[0x05BF] = NAME_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x05C1, 0x05C2, NAME_BODY_CHAR);
- CHARACTER_CODES[0x05C4] = NAME_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x064B, 0x0652, NAME_BODY_CHAR);
- CHARACTER_CODES[0x0670] = NAME_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x06D6, 0x06DC, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x06DD, 0x06DF, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x06E0, 0x06E4, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x06E7, 0x06E8, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x06EA, 0x06ED, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0901, 0x0903, NAME_BODY_CHAR);
- CHARACTER_CODES[0x093C] = NAME_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x093E, 0x094C, NAME_BODY_CHAR);
- CHARACTER_CODES[0x094D] = NAME_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x0951, 0x0954, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0962, 0x0963, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0981, 0x0983, NAME_BODY_CHAR);
- CHARACTER_CODES[0x09BC] = NAME_BODY_CHAR;
- CHARACTER_CODES[0x09BE] = NAME_BODY_CHAR;
- CHARACTER_CODES[0x09BF] = NAME_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x09C0, 0x09C4, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x09C7, 0x09C8, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x09CB, 0x09CD, NAME_BODY_CHAR);
- CHARACTER_CODES[0x09D7] = NAME_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x09E2, 0x09E3, NAME_BODY_CHAR);
- CHARACTER_CODES[0x0A02] = NAME_BODY_CHAR;
- CHARACTER_CODES[0x0A3C] = NAME_BODY_CHAR;
- CHARACTER_CODES[0x0A3E] = NAME_BODY_CHAR;
- CHARACTER_CODES[0x0A3F] = NAME_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x0A40, 0x0A42, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0A47, 0x0A48, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0A4B, 0x0A4D, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0A70, 0x0A71, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0A81, 0x0A83, NAME_BODY_CHAR);
- CHARACTER_CODES[0x0ABC] = NAME_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x0ABE, 0x0AC5, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0AC7, 0x0AC9, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0ACB, 0x0ACD, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0B01, 0x0B03, NAME_BODY_CHAR);
- CHARACTER_CODES[0x0B3C] = NAME_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x0B3E, 0x0B43, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0B47, 0x0B48, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0B4B, 0x0B4D, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0B56, 0x0B57, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0B82, 0x0B83, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0BBE, 0x0BC2, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0BC6, 0x0BC8, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0BCA, 0x0BCD, NAME_BODY_CHAR);
- CHARACTER_CODES[0x0BD7] = NAME_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x0C01, 0x0C03, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0C3E, 0x0C44, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0C46, 0x0C48, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0C4A, 0x0C4D, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0C55, 0x0C56, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0C82, 0x0C83, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0CBE, 0x0CC4, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0CC6, 0x0CC8, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0CCA, 0x0CCD, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0CD5, 0x0CD6, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0D02, 0x0D03, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0D3E, 0x0D43, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0D46, 0x0D48, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0D4A, 0x0D4D, NAME_BODY_CHAR);
- CHARACTER_CODES[0x0D57] = NAME_BODY_CHAR;
- CHARACTER_CODES[0x0E31] = NAME_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x0E34, 0x0E3A, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0E47, 0x0E4E, NAME_BODY_CHAR);
- CHARACTER_CODES[0x0EB1] = NAME_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x0EB4, 0x0EB9, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0EBB, 0x0EBC, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0EC8, 0x0ECD, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0F18, 0x0F19, NAME_BODY_CHAR);
- CHARACTER_CODES[0x0F35] = NAME_BODY_CHAR;
- CHARACTER_CODES[0x0F37] = NAME_BODY_CHAR;
- CHARACTER_CODES[0x0F39] = NAME_BODY_CHAR;
- CHARACTER_CODES[0x0F3E] = NAME_BODY_CHAR;
- CHARACTER_CODES[0x0F3F] = NAME_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x0F71, 0x0F84, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0F86, 0x0F8B, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0F90, 0x0F95, NAME_BODY_CHAR);
- CHARACTER_CODES[0x0F97] = NAME_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x0F99, 0x0FAD, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0FB1, 0x0FB7, NAME_BODY_CHAR);
- CHARACTER_CODES[0x0FB9] = NAME_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x20D0, 0x20DC, NAME_BODY_CHAR);
- CHARACTER_CODES[0x20E1] = NAME_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x302A, 0x302F, NAME_BODY_CHAR);
- CHARACTER_CODES[0x3099] = NAME_BODY_CHAR;
- CHARACTER_CODES[0x309A] = NAME_BODY_CHAR;
- // Digit
- Arrays.fill(CHARACTER_CODES, 0x0030, 0x0039, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0660, 0x0669, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x06F0, 0x06F9, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0966, 0x096F, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x09E6, 0x09EF, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0A66, 0x0A6F, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0AE6, 0x0AEF, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0B66, 0x0B6F, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0BE7, 0x0BEF, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0C66, 0x0C6F, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0CE6, 0x0CEF, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0D66, 0x0D6F, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0E50, 0x0E59, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0ED0, 0x0ED9, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x0F20, 0x0F29, NAME_BODY_CHAR);
- // Extender
- CHARACTER_CODES[0x00B7] = NAME_BODY_CHAR;
- CHARACTER_CODES[0x02D0] = NAME_BODY_CHAR;
- CHARACTER_CODES[0x02D1] = NAME_BODY_CHAR;
- CHARACTER_CODES[0x0387] = NAME_BODY_CHAR;
- CHARACTER_CODES[0x0640] = NAME_BODY_CHAR;
- CHARACTER_CODES[0x0E46] = NAME_BODY_CHAR;
- CHARACTER_CODES[0x0EC6] = NAME_BODY_CHAR;
- CHARACTER_CODES[0x3005] = NAME_BODY_CHAR;
- Arrays.fill(CHARACTER_CODES, 0x3031, 0x3035, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x309D, 0x309E, NAME_BODY_CHAR);
- Arrays.fill(CHARACTER_CODES, 0x30FC, 0x30FE, NAME_BODY_CHAR);
-
- }
-
- private final Writer writer;
+ private final Appendable appendable;
private final ArrayDeque<CharSequence> elementNames;
private final Set<CharSequence> currentAttributes = new HashSet<>();
@@ -413,11 +59,24 @@ public final class XmlWriter implements IXmlWriter {
* Constructs an XmlWriter with the specified writer for output.
* @param writer the writer to write to.
*/
- public XmlWriter(final Writer writer) {
- this.writer = writer;
+ public XmlWriter(final Appendable writer) {
+ this.appendable = writer;
this.elementNames = new ArrayDeque<>();
}
+ private void validateRootOpen() throws IOException {
+ if (elementsWritten && elementNames.isEmpty()) {
+ throw new OperationNotAllowedException("Root element already
closed. Cannot open new element.");
+ }
+ }
+
+ private void maybeCloseElement() throws IOException {
+ if (inElement) {
+ appendable.append('>');
+ inElement = false;
+ }
+ }
+
/**
* Starts a document by writing a prolog. Calling this method is optional.
When
* writing a document fragment, it should <em>not</em> be called.
@@ -434,7 +93,7 @@ public final class XmlWriter implements IXmlWriter {
if (prologWritten) {
throw new OperationNotAllowedException("Only one prolog allowed");
}
- writer.write("<?xml version='1.0'?>");
+ appendable.append("<?xml version='1.0'?>");
prologWritten = true;
return this;
}
@@ -450,18 +109,14 @@ public final class XmlWriter implements IXmlWriter {
*/
@Override
public IXmlWriter openElement(final CharSequence elementName) throws
IOException {
- if (elementsWritten && elementNames.isEmpty()) {
- throw new OperationNotAllowedException("Root element already
closed. Cannot open new element.");
- }
- if (isInvalidName(elementName)) {
+ validateRootOpen();
+ if (!XMLChar.isValidName(elementName.toString())) {
throw new InvalidXmlException("'" + elementName + "' is not a
valid element name");
}
elementsWritten = true;
- if (inElement) {
- writer.write('>');
- }
- writer.write('<');
- rawWrite(elementName);
+ maybeCloseElement();
+ appendable.append('<');
+ appendable.append(elementName);
inElement = true;
elementNames.push(elementName);
currentAttributes.clear();
@@ -470,14 +125,11 @@ public final class XmlWriter implements IXmlWriter {
@Override
public IXmlWriter comment(final CharSequence text) throws IOException {
- if (inElement) {
- writer.write('>');
- }
- inElement = false;
- writer.write("<!-- ");
- content(text);
- writer.write(" -->");
- return this;
+ maybeCloseElement();
+ appendable.append("<!-- ");
+ writeEscaped(text, false);
+ appendable.append(" -->");
+ return this;
}
/**
@@ -496,12 +148,10 @@ public final class XmlWriter implements IXmlWriter {
@Override
public IXmlWriter attribute(final CharSequence name, final CharSequence
value) throws IOException {
if (elementNames.isEmpty()) {
- if (elementsWritten) {
- throw new OperationNotAllowedException("Root element has
already been closed.");
- }
+ validateRootOpen();
throw new OperationNotAllowedException("Close called before an
element has been opened.");
}
- if (isInvalidName(name)) {
+ if (!XMLChar.isValidName(name.toString())) {
throw new InvalidXmlException("'" + name + "' is not a valid
attribute name.");
}
if (!inElement) {
@@ -510,12 +160,11 @@ public final class XmlWriter implements IXmlWriter {
if (currentAttributes.contains(name)) {
throw new InvalidXmlException("Each attribute can only be written
once");
}
- writer.write(' ');
- rawWrite(name);
- writer.write('=');
- writer.write('\'');
+ appendable.append(' ');
+ appendable.append(name);
+ appendable.append("='");
writeAttributeContent(value);
- writer.write('\'');
+ appendable.append("'");
currentAttributes.add(name);
return this;
}
@@ -526,21 +175,16 @@ public final class XmlWriter implements IXmlWriter {
private void prepareForData() throws IOException {
if (elementNames.isEmpty()) {
- if (elementsWritten) {
- throw new OperationNotAllowedException("Root element has
already been closed.");
- }
+ validateRootOpen();
throw new OperationNotAllowedException("An element must be opened
before content can be written.");
}
- if (inElement) {
- writer.write('>');
- }
+ maybeCloseElement();
}
@Override
public IXmlWriter content(final CharSequence content) throws IOException {
prepareForData();
writeEscaped(content, false);
- inElement = false;
return this;
}
@@ -553,16 +197,16 @@ public final class XmlWriter implements IXmlWriter {
sb.replace(found, found + 3, "{rat:CDATA close}");
}
- writer.write("<![CDATA[ ");
+ appendable.append("<![CDATA[ ");
for (int i = 0; i < sb.length(); i++) {
char c = sb.charAt(i);
- if (isOutOfRange(c)) {
- writer.write(String.format("\\u%X", (int) c));
+ if (!XMLChar.isContent(c)) {
+ appendable.append(String.format("\\u%X", (int) c));
} else {
- writer.write(c);
+ appendable.append(c);
}
}
- writer.write(" ]]>");
+ appendable.append(" ]]>");
inElement = false;
return this;
@@ -573,27 +217,23 @@ public final class XmlWriter implements IXmlWriter {
for (int i = 0; i < length; i++) {
char character = content.charAt(i);
if (character == '&') {
- writer.write("&");
+ appendable.append("&");
} else if (character == '<') {
- writer.write("<");
+ appendable.append("<");
} else if (character == '>') {
- writer.write(">");
+ appendable.append(">");
} else if (isAttributeContent && character == '\'') {
- writer.write("'");
+ appendable.append("'");
} else if (isAttributeContent && character == '\"') {
- writer.write(""");
- } else if (isOutOfRange(character)) {
- writer.write('?');
+ appendable.append(""");
+ } else if (!(XMLChar.isContent(character) ||
XMLChar.isSpace(character))) {
+ appendable.append(String.format("\\u%X", (int) character));
} else {
- writer.write(character);
+ appendable.append(character);
}
}
}
- private boolean isOutOfRange(final char character) {
- return !ALLOWED_CHARACTERS[character];
- }
-
/**
* Closes the last element written.
*
@@ -604,22 +244,17 @@ public final class XmlWriter implements IXmlWriter {
@Override
public IXmlWriter closeElement() throws IOException {
if (elementNames.isEmpty()) {
- if (elementsWritten) {
- throw new OperationNotAllowedException("Root element has
already been closed.");
- }
+ validateRootOpen();
throw new OperationNotAllowedException("Close called before an
element has been opened.");
}
final CharSequence elementName = elementNames.pop();
if (inElement) {
- writer.write('/');
- writer.write('>');
+ appendable.append("/>");
} else {
- writer.write('<');
- writer.write('/');
- rawWrite(elementName);
- writer.write('>');
+ appendable.append("</");
+ appendable.append(elementName);
+ appendable.append('>');
}
- writer.flush();
inElement = false;
return this;
}
@@ -635,26 +270,21 @@ public final class XmlWriter implements IXmlWriter {
public IXmlWriter closeElement(final CharSequence name) throws IOException
{
Objects.requireNonNull(name);
if (elementNames.isEmpty()) {
- if (elementsWritten) {
- throw new OperationNotAllowedException("Root element has
already been closed.");
- }
+ validateRootOpen();
throw new OperationNotAllowedException("Close called before an
element has been opened.");
}
CharSequence elementName = null;
while (!name.equals(elementName)) {
elementName = elementNames.pop();
if (inElement) {
- writer.write('/');
- writer.write('>');
+ appendable.append("/>");
} else {
- writer.write('<');
- writer.write('/');
- rawWrite(elementName);
- writer.write('>');
+ appendable.append("</");
+ appendable.append(elementName);
+ appendable.append('>');
}
inElement = false;
}
- writer.flush();
return this;
}
@@ -675,49 +305,32 @@ public final class XmlWriter implements IXmlWriter {
while (!elementNames.isEmpty()) {
closeElement();
}
- writer.flush();
return this;
}
- private void rawWrite(final CharSequence sequence) throws IOException {
- for (int i = 0; i < sequence.length(); i++) {
- final char charAt = sequence.charAt(i);
- writer.write(charAt);
+ @Override
+ public void close() throws IOException {
+ closeDocument();
+ if (appendable instanceof Closeable closeable) {
+ closeable.close();
}
}
- private boolean isInvalidName(final CharSequence sequence) {
- boolean result = true;
- final int length = sequence.length();
- for (int i = 0; i < length; i++) {
- char character = sequence.charAt(i);
- if (i == 0) {
- if (!isValidNameStart(character)) {
- result = false;
- break;
- }
- } else {
- if (!isValidNameBody(character)) {
- result = false;
- break;
- }
- }
+ public IXmlWriter append(final Document document) throws IOException {
+ validateRootOpen();
+ elementsWritten = true;
+ maybeCloseElement();
+ appendable.append(System.lineSeparator());
+ currentAttributes.clear();
+ try {
+ Transformer transformer = StandardXmlFactory.create();
+ ByteArrayOutputStream baos = new ByteArrayOutputStream();
+ transformer.transform(new DOMSource(document),
+ new StreamResult(baos));
+ appendable.append(baos.toString());
+ } catch (TransformerException e) {
+ throw new IOException(e);
}
- return !result;
- }
-
- private boolean isValidNameStart(final char character) {
- final byte code = CHARACTER_CODES[character];
- return (code & NAME_START_MASK) > 0;
- }
-
- private boolean isValidNameBody(final char character) {
- final byte code = CHARACTER_CODES[character];
- return (code & NAME_MASK) > 0;
- }
-
- @Override
- public void close() throws IOException {
- closeDocument();
+ return this;
}
}
diff --git
a/apache-rat-core/src/main/java/org/apache/rat/utils/StandardXmlFactory.java
b/apache-rat-core/src/main/java/org/apache/rat/utils/StandardXmlFactory.java
new file mode 100644
index 00000000..6ebf2397
--- /dev/null
+++ b/apache-rat-core/src/main/java/org/apache/rat/utils/StandardXmlFactory.java
@@ -0,0 +1,67 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one
+ * or more contributor license agreements. See the NOTICE file
+ * distributed with this work for additional information
+ * regarding copyright ownership. The ASF licenses this file
+ * to you under the Apache License, Version 2.0 (the
+ * "License"); you may not use this file except in compliance
+ * with the License. You may obtain a copy of the License at
+ *
+ * https://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing,
+ * software distributed under the License is distributed on an
+ * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
+ * KIND, either express or implied. See the License for the
+ * specific language governing permissions and limitations
+ * under the License.
+ */
+package org.apache.rat.utils;
+
+import java.io.InputStream;
+
+import javax.xml.XMLConstants;
+import javax.xml.transform.OutputKeys;
+import javax.xml.transform.Transformer;
+import javax.xml.transform.TransformerConfigurationException;
+import javax.xml.transform.TransformerFactory;
+import javax.xml.transform.stream.StreamSource;
+
+/**
+ * Factory to create standard XML objects. The intention of this class is to
resolve in a consistent manner the
+ * XXE errors and similar XML IO errors.
+ */
+public final class StandardXmlFactory {
+
+ private StandardXmlFactory() {
+ // do not instantiate.
+ }
+ /**
+ * Create a transformer with no stylesheet.
+ * @return the transformer.
+ * @throws TransformerConfigurationException on error.
+ */
+ public static Transformer create() throws
TransformerConfigurationException {
+ return create(null);
+ }
+
+ /**
+ * Create a transformer with the specified stylesheet.
+ * @param styleIn the stylesheet to use.
+ * @return the transformer.
+ * @throws TransformerConfigurationException on error.
+ */
+ public static Transformer create(final InputStream styleIn) throws
TransformerConfigurationException {
+ TransformerFactory factory = TransformerFactory.newInstance();
+ factory.setFeature(XMLConstants.FEATURE_SECURE_PROCESSING, true);
+ factory.setAttribute(XMLConstants.ACCESS_EXTERNAL_DTD, "");
+ factory.setAttribute(XMLConstants.ACCESS_EXTERNAL_STYLESHEET, "");
+ Transformer transformer = styleIn == null ? factory.newTransformer() :
factory.newTransformer(new StreamSource(styleIn));
+ transformer.setOutputProperty(OutputKeys.OMIT_XML_DECLARATION, "yes");
+ transformer.setOutputProperty(OutputKeys.METHOD, "xml");
+ transformer.setOutputProperty(OutputKeys.INDENT, "yes");
+ transformer.setOutputProperty(OutputKeys.ENCODING, "UTF-8");
+
transformer.setOutputProperty("{http://xml.apache.org/xslt}indent-amount", "4");
+ return transformer;
+ }
+}
diff --git
a/apache-rat-core/src/test/java/org/apache/rat/report/xml/writer/impl/base/XmlWriterTest.java
b/apache-rat-core/src/test/java/org/apache/rat/report/xml/writer/impl/base/XmlWriterTest.java
index e6e00132..4f5ea5a0 100644
---
a/apache-rat-core/src/test/java/org/apache/rat/report/xml/writer/impl/base/XmlWriterTest.java
+++
b/apache-rat-core/src/test/java/org/apache/rat/report/xml/writer/impl/base/XmlWriterTest.java
@@ -18,24 +18,29 @@
*/
package org.apache.rat.report.xml.writer.impl.base;
+import org.apache.rat.report.xml.writer.IXmlWriter;
import org.apache.rat.report.xml.writer.InvalidXmlException;
import org.apache.rat.report.xml.writer.OperationNotAllowedException;
import org.apache.rat.report.xml.writer.XmlWriter;
+import org.apache.rat.testhelpers.XmlUtils;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;
+import org.w3c.dom.Document;
+import java.io.ByteArrayInputStream;
+import java.io.IOException;
import java.io.StringWriter;
+import java.nio.charset.StandardCharsets;
+import java.util.NoSuchElementException;
import static org.assertj.core.api.Assertions.fail;
import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertFalse;
+import static org.junit.jupiter.api.Assertions.assertThrows;
import static org.junit.jupiter.api.Assertions.assertTrue;
+class XmlWriterTest {
-public class XmlWriterTest {
-
- private static final char[] ZERO_CHAR = {(char)0};
-
private XmlWriter writer;
private StringWriter out;
@@ -46,7 +51,7 @@ public class XmlWriterTest {
}
@Test
- public void returnValues() throws Exception {
+ void returnValues() throws Exception {
assertEquals(
writer, writer.openElement("alpha"), "XmlWriters should always
return themselves");
assertEquals(
@@ -58,7 +63,7 @@ public class XmlWriterTest {
}
@Test
- public void openElement() throws Exception {
+ void openElement() throws Exception {
assertEquals(
writer, writer.openElement("alpha"), "XmlWriters should always
return themselves");
assertEquals("<alpha", out.toString(), "Alpha element started");
@@ -74,7 +79,7 @@ public class XmlWriterTest {
}
@Test
- public void invalidElementName() throws Exception {
+ void invalidElementName() throws Exception {
assertTrue( isValidElementName("alpha"), "All strings ok");
assertTrue(isValidElementName("alpha77"), "Strings and digits ok");
assertFalse(isValidElementName("5alpha77"), "Must no start with
digit");
@@ -102,7 +107,7 @@ public class XmlWriterTest {
}
@Test
- public void callOpenElementAfterLastElementClosed() throws Exception {
+ void callOpenElementAfterLastElementClosed() throws Exception {
assertEquals(
writer, writer.openElement("alpha"), "XmlWriters should always
return themselves");
assertEquals("<alpha", out.toString(), "Alpha element started");
@@ -118,7 +123,7 @@ public class XmlWriterTest {
}
@Test
- public void callCloseElementAfterLastElementClosed() throws Exception {
+ void callCloseElementAfterLastElementClosed() throws Exception {
assertEquals(
writer, writer.openElement("alpha"), "XmlWriters should always
return themselves");
assertEquals("<alpha", out.toString(), "Alpha element started");
@@ -134,7 +139,7 @@ public class XmlWriterTest {
}
@Test
- public void closeFirstElement() throws Exception {
+ void closeFirstElement() throws Exception {
assertEquals(
writer, writer.openElement("alpha"), "XmlWriters should always
return themselves");
assertEquals("<alpha", out.toString(), "Alpha element started");
@@ -144,7 +149,7 @@ public class XmlWriterTest {
}
@Test
- public void closeElementWithContent() throws Exception {
+ void closeElementWithContent() throws Exception {
assertEquals(
writer, writer.openElement("alpha"), "XmlWriters should always
return themselves");
assertEquals("<alpha", out.toString(), "Alpha element started");
@@ -166,7 +171,7 @@ public class XmlWriterTest {
}
@Test
- public void closeElementBeforeFirstElement() throws Exception {
+ void closeElementBeforeFirstElement() throws Exception {
try {
writer.closeElement();
fail("Cannot close elements before the first element has been
closed");
@@ -176,7 +181,7 @@ public class XmlWriterTest {
}
@Test
- public void contentAfterElement() throws Exception {
+ void contentAfterElement() throws Exception {
assertEquals(
writer, writer.openElement("alpha"), "XmlWriters should always
return themselves");
assertEquals("<alpha", out.toString(), "Alpha element started");
@@ -204,7 +209,7 @@ public class XmlWriterTest {
}
@Test
- public void contentAfterLastElement() throws Exception {
+ void contentAfterLastElement() throws Exception {
assertEquals(
writer, writer.openElement("alpha"), "XmlWriters should always
return themselves");
assertEquals("<alpha", out.toString(), "Alpha element started");
@@ -220,7 +225,7 @@ public class XmlWriterTest {
}
@Test
- public void writeContentBeforeFirstElement() throws Exception {
+ void writeContentBeforeFirstElement() throws Exception {
try {
writer.content("Too early");
fail("Cannot close elements before the first element has been
closed");
@@ -230,7 +235,7 @@ public class XmlWriterTest {
}
@Test
- public void contentEscaping() throws Exception {
+ void contentEscaping() throws Exception {
assertEquals(
writer, writer.openElement("alpha"), "XmlWriters should always
return themselves");
assertEquals("<alpha", out.toString(), "Alpha element started");
@@ -247,7 +252,7 @@ public class XmlWriterTest {
}
@Test
- public void attributeAfterLastElement() throws Exception {
+ void attributeAfterLastElement() throws Exception {
assertEquals(
writer, writer.openElement("alpha"), "XmlWriters should always
return themselves");
assertEquals("<alpha", out.toString(), "Alpha element started");
@@ -263,7 +268,7 @@ public class XmlWriterTest {
}
@Test
- public void attributeContentBeforeFirstElement() throws Exception {
+ void attributeContentBeforeFirstElement() throws Exception {
try {
writer.attribute("foo", "bar");
fail("Cannot close elements before the first element has been
closed");
@@ -273,7 +278,7 @@ public class XmlWriterTest {
}
@Test
- public void invalidAttributeName() throws Exception {
+ void invalidAttributeName() throws Exception {
writer.openElement("alpha");
assertTrue(isValidAttributeName("alpha"), "All string ok");
assertTrue(isValidAttributeName("alpha77"), "Strings and digits ok");
@@ -295,7 +300,7 @@ public class XmlWriterTest {
}
@Test
- public void escapeAttributeContent() throws Exception {
+ void escapeAttributeContent() throws Exception {
assertEquals(
writer, writer.openElement("alpha"), "XmlWriters should always
return themselves");
assertEquals("<alpha", out.toString(), "Alpha element started");
@@ -318,7 +323,7 @@ public class XmlWriterTest {
}
@Test
- public void attributeInContent() throws Exception {
+ void attributeInContent() throws Exception {
assertEquals(
writer, writer.openElement("alpha"), "XmlWriters should always
return themselves");
assertEquals("<alpha", out.toString(), "Alpha element started");
@@ -333,18 +338,33 @@ public class XmlWriterTest {
}
@Test
- public void outOfRangeCharacter() throws Exception {
+ void outOfRangeCharacter() throws Exception {
assertEquals(
writer, writer.openElement("alpha"), "XmlWriters should always
return themselves");
assertEquals("<alpha", out.toString(), "Alpha element started");
- assertEquals(
- writer, writer.content(new String(ZERO_CHAR)), "XmlWriters
should always return themselves");
- String out = this.out.toString();
- assertEquals("<alpha>?", out, "Replace illegal characters with
question marks");
+ CharSequence cs = new CharSequence() {
+ @Override
+ public int length() {
+ return 1;
+ }
+
+ @Override
+ public char charAt(int index) {
+ return Character.highSurrogate(0x110000);
+ }
+
+ @Override
+ public CharSequence subSequence(int start, int end) {
+ return null;
+ }
+ };
+
+ assertEquals(writer, writer.content(cs), "XmlWriters should always
return themselves");
+ assertEquals("<alpha>\\uDC00", this.out.toString(), "Replace illegal
characters with \\u encoding");
}
@Test
- public void attributeAfterElementClosed() throws Exception {
+ void attributeAfterElementClosed() throws Exception {
assertEquals(
writer, writer.openElement("alpha"), "XmlWriters should always
return themselves");
assertEquals("<alpha", out.toString(), "Alpha element started");
@@ -363,7 +383,7 @@ public class XmlWriterTest {
}
@Test
- public void closeDocumentBeforeOpen() throws Exception {
+ void closeDocumentBeforeOpen() throws Exception {
try {
writer.closeDocument();
fail("Cannot close document before the first element has been
opened");
@@ -373,7 +393,7 @@ public class XmlWriterTest {
}
@Test
- public void closeDocumentAfterRootElementClosed() throws Exception {
+ void closeDocumentAfterRootElementClosed() throws Exception {
assertEquals(
writer, writer.openElement("alpha"), "XmlWriters should always
return themselves");
assertEquals("<alpha", out.toString(), "Alpha element started");
@@ -388,7 +408,7 @@ public class XmlWriterTest {
}
@Test
- public void closeSimpleDocument() throws Exception {
+ void closeSimpleDocument() throws Exception {
assertEquals(
writer, writer.openElement("alpha"), "XmlWriters should always
return themselves");
assertEquals("<alpha", out.toString(), "Alpha element started");
@@ -401,7 +421,7 @@ public class XmlWriterTest {
}
@Test
- public void closeComplexDocument() throws Exception {
+ void closeComplexDocument() throws Exception {
assertEquals(
writer, writer.openElement("alpha"), "XmlWriters should always
return themselves");
assertEquals("<alpha", out.toString(), "Alpha element started");
@@ -429,14 +449,14 @@ public class XmlWriterTest {
}
@Test
- public void writeProlog() throws Exception {
+ void writeProlog() throws Exception {
assertEquals(
writer, writer.startDocument(), "XmlWriters should always
return themselves");
assertEquals("<?xml version='1.0'?>", out.toString(), "Prolog
written");
}
@Test
- public void writeAfterElement() throws Exception {
+ void writeAfterElement() throws Exception {
assertEquals(
writer, writer.openElement("alpha"), "XmlWriters should always
return themselves");
assertEquals("<alpha", out.toString(), "Alpha element started");
@@ -449,7 +469,7 @@ public class XmlWriterTest {
}
@Test
- public void writePrologTwo() throws Exception {
+ void writePrologTwo() throws Exception {
assertEquals(
writer, writer.startDocument(), "XmlWriters should always
return themselves");
assertEquals("<?xml version='1.0'?>", out.toString(), "Prolog
written");
@@ -462,7 +482,7 @@ public class XmlWriterTest {
}
@Test
- public void duplicateAttributes() throws Exception {
+ void duplicateAttributes() throws Exception {
assertEquals(
writer, writer.openElement("alpha"), "XmlWriters should always
return themselves");
assertEquals("<alpha", out.toString(), "Alpha element started");
@@ -482,4 +502,50 @@ public class XmlWriterTest {
// Each attribute may only be written once
}
}
+
+ @Test
+ void writeCDataBeforeElement() {
+ assertThrows(OperationNotAllowedException.class, () ->
writer.startDocument().cdata("Just cdata").closeDocument());
+ }
+
+ @Test
+ void writeCData() throws Exception {
+ writer.startDocument().openElement("test").cdata("Just
cdata").closeDocument();
+ assertEquals("<?xml version='1.0'?><test><![CDATA[ Just cdata
]]></test>", out.toString());
+ }
+
+ @Test
+ void writeCDataEmbeddedCData() throws Exception {
+ writer.startDocument().openElement("test").cdata("Some <![CDATA[ cdata
]]> text").closeDocument();
+ assertEquals("<?xml version='1.0'?><test><![CDATA[ Some \\u3C![CDATA[
cdata {rat:CDATA close} text ]]></test>", out.toString());
+ }
+
+ @Test
+ void closeElementBeforeOpened() throws IOException {
+ IXmlWriter underTest = writer.startDocument().openElement("test");
+ assertThrows(NoSuchElementException.class, () ->
underTest.closeElement("missing"));
+ }
+
+ @Test
+ void closeElement() throws Exception {
+
writer.startDocument().openElement("root").openElement("hello").openElement("world").content("hello
world").closeElement("hello").openElement("test").closeDocument();
+ assertEquals("<?xml version='1.0'?><root><hello><world>hello
world</world></hello><test/></root>", out.toString());
+ }
+
+ @Test
+ void append() throws Exception {
+ // ensure proper line endings
+ String expected = String.format("<?xml version='1.0'?><base>%n" +
+ "<root>%n" +
+ " <hello>%n" +
+ " <world>hello world</world>%n" +
+ " </hello>%n" +
+ " <test/>%n" +
+ "</root>%n" +
+ "</base>");
+ byte[] rawDocument = "<?xml version='1.0'?><root><hello><world>hello
world</world></hello><test/></root>".getBytes(StandardCharsets.UTF_8);
+ Document document = XmlUtils.toDom(new
ByteArrayInputStream(rawDocument));
+
writer.startDocument().openElement("base").append(document).closeDocument();
+ assertEquals(expected, out.toString());
+ }
}
diff --git
a/apache-rat-tools/src/main/java/org/apache/rat/tools/xsd/XsdGenerator.java
b/apache-rat-tools/src/main/java/org/apache/rat/tools/xsd/XsdGenerator.java
index 6d1529da..a3ceb01c 100644
--- a/apache-rat-tools/src/main/java/org/apache/rat/tools/xsd/XsdGenerator.java
+++ b/apache-rat-tools/src/main/java/org/apache/rat/tools/xsd/XsdGenerator.java
@@ -30,10 +30,8 @@ import java.util.HashMap;
import java.util.List;
import java.util.Map;
-import javax.xml.transform.OutputKeys;
import javax.xml.transform.Transformer;
import javax.xml.transform.TransformerException;
-import javax.xml.transform.TransformerFactory;
import javax.xml.transform.stream.StreamResult;
import javax.xml.transform.stream.StreamSource;
@@ -45,6 +43,7 @@ import org.apache.rat.configuration.MatcherBuilderTracker;
import org.apache.rat.configuration.XMLConfig;
import org.apache.rat.license.SimpleLicense;
import org.apache.rat.tools.xsd.XsdWriter.Type;
+import org.apache.rat.utils.StandardXmlFactory;
/**
* Generates the XSD for a configuration.
@@ -71,16 +70,9 @@ public class XsdGenerator {
public static void main(final String[] args) throws IOException,
TransformerException {
XsdGenerator generator = new XsdGenerator();
- TransformerFactory tf = TransformerFactory.newInstance();
- Transformer transformer;
try (InputStream in = generator.getInputStream();
InputStream styleIn = StyleSheets.XML.getStyleSheet().get()) {
- transformer = tf.newTransformer(new StreamSource(styleIn));
- transformer.setOutputProperty(OutputKeys.OMIT_XML_DECLARATION,
"yes");
- transformer.setOutputProperty(OutputKeys.METHOD, "xml");
- transformer.setOutputProperty(OutputKeys.INDENT, "yes");
- transformer.setOutputProperty(OutputKeys.ENCODING, "UTF-8");
-
transformer.setOutputProperty("{http://xml.apache.org/xslt}indent-amount", "4");
+ Transformer transformer = StandardXmlFactory.create(styleIn);
transformer.transform(new StreamSource(in),
new StreamResult(new OutputStreamWriter(System.out,
StandardCharsets.UTF_8)));
}
@@ -92,11 +84,11 @@ public class XsdGenerator {
* @throws IOException on output errors.
*/
public InputStream getInputStream() throws IOException {
- try (ByteArrayOutputStream baos = new ByteArrayOutputStream();
- Writer writer = new OutputStreamWriter(baos,
StandardCharsets.UTF_8)) {
- write(writer);
- return new ByteArrayInputStream(baos.toByteArray());
+ ByteArrayOutputStream baos = new ByteArrayOutputStream();
+ try (Writer otherWriter = new OutputStreamWriter(baos,
StandardCharsets.UTF_8)) {
+ write(otherWriter);
}
+ return new ByteArrayInputStream(baos.toByteArray());
}
/**
diff --git a/src/conf/checkstyle-suppressions.xml
b/src/conf/checkstyle-suppressions.xml
index 8c654e16..5431cde3 100644
--- a/src/conf/checkstyle-suppressions.xml
+++ b/src/conf/checkstyle-suppressions.xml
@@ -61,4 +61,8 @@
<suppress checks="[a-zA-Z0-9]*"
files="src[/\\]main[/\\]java[/\\]org[/\\]apache[/\\]rat[/\\]anttasks[/\\]BaseAntTask.java"
/>
+ <!-- RAT-539: Suppress checks for imported code -->
+ <suppress checks="[a-zA-Z0-9]*"
+
files="org[/\\]apache[/\\]rat[/\\]report[/\\]xml[/\\]writer[/\\]XMLChar.java" />
+
</suppressions>