1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159
|
/*
* Copyright (c) 2014 SAP SE. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
/**
* @test
* @bug 8034087 8027359
* @summary XML parser may overwrite element content if that content falls onto the border of an entity scanner buffer
* @run main FragmentScannerBufferLimitTest
*/
import java.io.*;
import javax.xml.parsers.*;
import javax.xml.transform.*;
import org.w3c.dom.*;
import org.xml.sax.*;
/**
* Test for overwriting of XML element content by the XML parser when reading over buffer
* limits.
*
* We create a simple XML document of the form:
*
* <?xml version=\"1.1\"?>
* <ROOT>
* <FILLER>ffffffff...fffff</FILLER>
* <TEST>content</TEST><TEST2>content2</TEST2>
* <FILLER>ffffffff...fffffffff</FILLER>
* </ROOT>
*
* What's important here is, that the test content is at the border of an entity scanner
* buffer (XMLEntityScanner uses 8192 byte buffers). That's why there are filler elements
* of sufficient length that ensure there is a buffer break inside the test content
* and there is enough to read to require another buffer read after the content has been
* read.
*
* With the faulty implementation, the test content gets overwritten with characters
* read from the next buffer, i.e. 'f's.
*
* @author steffen.schreiber@sap.com
*/
public class FragmentScannerBufferLimitTest {
static int errCount = 0;
/**
* Check the test content.
*/
public static void main(String[] args) throws ParserConfigurationException,
SAXException, IOException, TransformerConfigurationException,
TransformerException, TransformerFactoryConfigurationError {
String testString = "<TEST>content</TEST><TEST2>content2</TEST2>";
for (int i = 0; i < testString.length(); i++) {
test(createDocument(testString.toString(), i), ""+ i);
}
if (errCount == 0) {
System.out.println("OK");
}
else {
System.out.println("ERROR");
throw new RuntimeException("Parsing error: element content has been overwritten");
}
}
/**
* Create the test XML document.
* @param testString the test content string
* @param bufferLimitPosition the position in the string where the buffer should break
* @return the document
*/
private static String createDocument(String testString, int bufferLimitPosition) throws UnsupportedEncodingException {
StringBuilder result = new StringBuilder();
result.append("<?xml version=\"1.1\"?>");
result.append("<ROOT>");
int fillerLength = 8192 - bufferLimitPosition;
createFiller(result, fillerLength);
result.append(testString);
createFiller(result, 9000);
result.append("</ROOT>");
return result.toString();
}
/**
* Create the filler element of the given length.
* @param buffer the output buffer
* @param length the required length of the element, including the element tags
*/
private static void createFiller(StringBuilder buffer, int length) {
buffer.append("<FILLER>");
int fillLength = length - "<FILLER></FILLER>".length();
for (int i=0; i<fillLength; i++) {
buffer.append('f');
}
buffer.append("</FILLER>");
}
private static void test(String document, String testName) throws SAXException, IOException, ParserConfigurationException {
DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
DocumentBuilder builder = factory.newDocumentBuilder();
Document doc = builder.parse(new ByteArrayInputStream(document.getBytes("UTF-8")));
// check that there is the root node
NodeList roots = doc.getElementsByTagName("ROOT");
assert roots.getLength() == 1;
Node root = roots.item(0);
// check that root has children "FILLER" and "TEST"
NodeList children = root.getChildNodes();
assert children.getLength() == 4;
assert children.item(0).getNodeName().equals("FILLER");
assert children.item(1).getNodeName().equals("TEST");
assert children.item(2).getNodeName().equals("TEST2");
assert children.item(3).getNodeName().equals("FILLER");
// check that the test node has content "content"
checkContent(children.item(1).getTextContent(), "content", document);
checkContent(children.item(2).getTextContent(), "content2", document);
}
private static void checkContent(String found, String expected, String document) {
if (! (found.equals(expected))) {
errCount++;
int bufferStart = "<?xml version=\"1.1\"?><ROOT>".length() +1;
int bufferStart2 = bufferStart + 8192;
System.err.println("\nError:: expected \"" + expected
+ "\", but found \"" + found + "\"!");
System.err.println("Buffer was (probably): [ ... "
+ document.substring(bufferStart2 - 20, bufferStart2) + "] ["
+ document.substring(bufferStart2, bufferStart2 + 30) + " ... ]");
}
}
}
|