8167179: Make XSL generated namespace prefixes local to transformation process
Reviewed-by: joehw
This commit is contained in:
parent
d544a61489
commit
4acf90edaa
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2007, 2015, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 2007, 2016, Oracle and/or its affiliates. All rights reserved.
|
||||||
*/
|
*/
|
||||||
/*
|
/*
|
||||||
* Licensed to the Apache Software Foundation (ASF) under one or more
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
||||||
@ -31,6 +31,7 @@ import com.sun.org.apache.bcel.internal.generic.GETFIELD;
|
|||||||
import com.sun.org.apache.bcel.internal.generic.GETSTATIC;
|
import com.sun.org.apache.bcel.internal.generic.GETSTATIC;
|
||||||
import com.sun.org.apache.bcel.internal.generic.INVOKEINTERFACE;
|
import com.sun.org.apache.bcel.internal.generic.INVOKEINTERFACE;
|
||||||
import com.sun.org.apache.bcel.internal.generic.INVOKESPECIAL;
|
import com.sun.org.apache.bcel.internal.generic.INVOKESPECIAL;
|
||||||
|
import com.sun.org.apache.bcel.internal.generic.INVOKESTATIC;
|
||||||
import com.sun.org.apache.bcel.internal.generic.INVOKEVIRTUAL;
|
import com.sun.org.apache.bcel.internal.generic.INVOKEVIRTUAL;
|
||||||
import com.sun.org.apache.bcel.internal.generic.ISTORE;
|
import com.sun.org.apache.bcel.internal.generic.ISTORE;
|
||||||
import com.sun.org.apache.bcel.internal.generic.InstructionHandle;
|
import com.sun.org.apache.bcel.internal.generic.InstructionHandle;
|
||||||
@ -1252,6 +1253,10 @@ public final class Stylesheet extends SyntaxTreeNode {
|
|||||||
classGen.getConstantPool());
|
classGen.getConstantPool());
|
||||||
transf.addException("com.sun.org.apache.xalan.internal.xsltc.TransletException");
|
transf.addException("com.sun.org.apache.xalan.internal.xsltc.TransletException");
|
||||||
|
|
||||||
|
// call resetPrefixIndex at the beginning of transform
|
||||||
|
final int check = cpg.addMethodref(BASIS_LIBRARY_CLASS, "resetPrefixIndex", "()V");
|
||||||
|
il.append(new INVOKESTATIC(check));
|
||||||
|
|
||||||
// Define and initialize current with the root node
|
// Define and initialize current with the root node
|
||||||
final LocalVariableGen current =
|
final LocalVariableGen current =
|
||||||
transf.addLocalVariable("current",
|
transf.addLocalVariable("current",
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2007, 2015, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 2007, 2016, Oracle and/or its affiliates. All rights reserved.
|
||||||
*/
|
*/
|
||||||
/*
|
/*
|
||||||
* Licensed to the Apache Software Foundation (ASF) under one or more
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
||||||
@ -31,6 +31,7 @@ import java.text.MessageFormat;
|
|||||||
import java.text.NumberFormat;
|
import java.text.NumberFormat;
|
||||||
import java.util.Locale;
|
import java.util.Locale;
|
||||||
import java.util.ResourceBundle;
|
import java.util.ResourceBundle;
|
||||||
|
import java.util.concurrent.atomic.AtomicInteger;
|
||||||
import javax.xml.transform.dom.DOMSource;
|
import javax.xml.transform.dom.DOMSource;
|
||||||
|
|
||||||
import com.sun.org.apache.xalan.internal.xsltc.DOM;
|
import com.sun.org.apache.xalan.internal.xsltc.DOM;
|
||||||
@ -1533,16 +1534,25 @@ public final class BasisLibrary {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This function is used in the execution of xsl:element
|
* These functions are used in the execution of xsl:element to generate
|
||||||
|
* and reset namespace prefix index local to current transformation process
|
||||||
*/
|
*/
|
||||||
private static int prefixIndex = 0;
|
|
||||||
|
|
||||||
public static String generatePrefix() {
|
public static String generatePrefix() {
|
||||||
synchronized (BasisLibrary.class) {
|
return ("ns" + threadLocalPrefixIndex.get().getAndIncrement());
|
||||||
return ("ns" + prefixIndex++);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static void resetPrefixIndex() {
|
||||||
|
threadLocalPrefixIndex.get().set(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static final ThreadLocal<AtomicInteger> threadLocalPrefixIndex =
|
||||||
|
new ThreadLocal<AtomicInteger>() {
|
||||||
|
@Override
|
||||||
|
protected AtomicInteger initialValue() {
|
||||||
|
return new AtomicInteger();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
public static final String RUN_TIME_INTERNAL_ERR =
|
public static final String RUN_TIME_INTERNAL_ERR =
|
||||||
"RUN_TIME_INTERNAL_ERR";
|
"RUN_TIME_INTERNAL_ERR";
|
||||||
public static final String RUN_TIME_COPY_ERR =
|
public static final String RUN_TIME_COPY_ERR =
|
||||||
|
@ -0,0 +1,193 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2016, Oracle and/or its affiliates. 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package transform;
|
||||||
|
|
||||||
|
import java.io.StringReader;
|
||||||
|
import java.io.StringWriter;
|
||||||
|
import java.util.concurrent.CyclicBarrier;
|
||||||
|
import java.util.concurrent.ExecutorService;
|
||||||
|
import java.util.concurrent.Executors;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
import java.util.concurrent.atomic.AtomicBoolean;
|
||||||
|
|
||||||
|
import javax.xml.transform.Source;
|
||||||
|
import javax.xml.transform.Templates;
|
||||||
|
import javax.xml.transform.Transformer;
|
||||||
|
import javax.xml.transform.TransformerFactory;
|
||||||
|
import javax.xml.transform.stream.StreamResult;
|
||||||
|
import javax.xml.transform.stream.StreamSource;
|
||||||
|
|
||||||
|
import org.testng.annotations.Test;
|
||||||
|
import static org.testng.Assert.assertTrue;
|
||||||
|
import static jaxp.library.JAXPTestUtilities.runWithAllPerm;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* @test
|
||||||
|
* @bug 8167179
|
||||||
|
* @library /javax/xml/jaxp/libs
|
||||||
|
* @run testng/othervm -DrunSecMngr=true transform.NamespacePrefixTest
|
||||||
|
* @run testng/othervm transform.NamespacePrefixTest
|
||||||
|
* @summary This class tests the generation of namespace prefixes
|
||||||
|
*/
|
||||||
|
public class NamespacePrefixTest {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testReuseTemplates() throws Exception {
|
||||||
|
final TransformerFactory tf = TransformerFactory.newInstance();
|
||||||
|
final Source xslsrc = new StreamSource(new StringReader(XSL));
|
||||||
|
final Templates tmpl = tf.newTemplates(xslsrc);
|
||||||
|
for (int i = 0; i < TRANSF_COUNT; i++) {
|
||||||
|
checkResult(doTransformation(tmpl.newTransformer()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testReuseTransformer() throws Exception {
|
||||||
|
final TransformerFactory tf = TransformerFactory.newInstance();
|
||||||
|
final Source xslsrc = new StreamSource(new StringReader(XSL));
|
||||||
|
final Transformer t = tf.newTransformer(xslsrc);
|
||||||
|
for (int i = 0; i < TRANSF_COUNT; i++) {
|
||||||
|
checkResult(doTransformation(t));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testConcurrentTransformations() throws Exception {
|
||||||
|
final TransformerFactory tf = TransformerFactory.newInstance();
|
||||||
|
final Source xslsrc = new StreamSource(new StringReader(XSL));
|
||||||
|
final Templates tmpl = tf.newTemplates(xslsrc);
|
||||||
|
concurrentTestPassed.set(true);
|
||||||
|
|
||||||
|
// Execute multiple TestWorker tasks
|
||||||
|
for (int id = 0; id < THREADS_COUNT; id++) {
|
||||||
|
EXECUTOR.execute(new TransformerThread(tmpl.newTransformer(), id));
|
||||||
|
}
|
||||||
|
// Initiate shutdown of previously submitted task
|
||||||
|
runWithAllPerm(EXECUTOR::shutdown);
|
||||||
|
// Wait for termination of submitted tasks
|
||||||
|
if (!EXECUTOR.awaitTermination(THREADS_COUNT, TimeUnit.SECONDS)) {
|
||||||
|
// If not all tasks terminates during the time out force them to shutdown
|
||||||
|
runWithAllPerm(EXECUTOR::shutdownNow);
|
||||||
|
}
|
||||||
|
// Check if all transformation threads generated the correct namespace prefix
|
||||||
|
assertTrue(concurrentTestPassed.get());
|
||||||
|
}
|
||||||
|
|
||||||
|
// Do one transformation with the provided transformer
|
||||||
|
private static String doTransformation(Transformer t) throws Exception {
|
||||||
|
StringWriter resWriter = new StringWriter();
|
||||||
|
Source xmlSrc = new StreamSource(new StringReader(XML));
|
||||||
|
t.transform(xmlSrc, new StreamResult(resWriter));
|
||||||
|
return resWriter.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if the transformation result string contains the
|
||||||
|
// element with the exact namespace prefix generated.
|
||||||
|
private static void checkResult(String result) {
|
||||||
|
// Check prefix of 'Element2' element, it should always be the same
|
||||||
|
assertTrue(result.contains(EXPECTED_CONTENT));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if the transformation result string contains the element with
|
||||||
|
// the exact namespace prefix generated by current thread.
|
||||||
|
// If the expected prefix is not found and there was no failures observed by
|
||||||
|
// other test threads then mark concurrent test as failed.
|
||||||
|
private static void checkThreadResult(String result, int id) {
|
||||||
|
boolean res = result.contains(EXPECTED_CONTENT);
|
||||||
|
System.out.printf("%d: transformation result: %s%n", id, res ? "Pass" : "Fail");
|
||||||
|
if (!res) {
|
||||||
|
System.out.printf("%d result:%s%n", id, result);
|
||||||
|
}
|
||||||
|
concurrentTestPassed.compareAndSet(true, res);
|
||||||
|
}
|
||||||
|
|
||||||
|
// TransformerThread task that does the transformation similar
|
||||||
|
// to testReuseTransformer test method
|
||||||
|
private class TransformerThread implements Runnable {
|
||||||
|
|
||||||
|
private final Transformer transformer;
|
||||||
|
private final int id;
|
||||||
|
|
||||||
|
TransformerThread(Transformer transformer, int id) {
|
||||||
|
this.transformer = transformer;
|
||||||
|
this.id = id;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
try {
|
||||||
|
System.out.printf("%d: waiting for barrier%n", id);
|
||||||
|
//Synchronize startup of all tasks
|
||||||
|
BARRIER.await();
|
||||||
|
System.out.printf("%d: starting transformation%n", id);
|
||||||
|
checkThreadResult(doTransformation(transformer), id);
|
||||||
|
} catch (Exception ex) {
|
||||||
|
throw new RuntimeException("TransformerThread " + id + " failed", ex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Number of subsequent transformations
|
||||||
|
private static final int TRANSF_COUNT = 10;
|
||||||
|
|
||||||
|
// Number of transformer threads running concurently
|
||||||
|
private static final int THREADS_COUNT = 10;
|
||||||
|
|
||||||
|
// Variable for storing the concurrent transformation test result. It is
|
||||||
|
// updated by transformer threads
|
||||||
|
private static final AtomicBoolean concurrentTestPassed = new AtomicBoolean(true);
|
||||||
|
|
||||||
|
// Cyclic barrier for threads startup synchronization
|
||||||
|
private static final CyclicBarrier BARRIER = new CyclicBarrier(THREADS_COUNT);
|
||||||
|
|
||||||
|
// Thread pool
|
||||||
|
private static final ExecutorService EXECUTOR = Executors.newCachedThreadPool();
|
||||||
|
|
||||||
|
// XSL that transforms XML and produces unique namespace prefixes for each element
|
||||||
|
private final static String XSL = "<xsl:stylesheet version=\"1.0\" xmlns:xsl=\"http://www.w3.org/1999/XSL/Transform\">\n"
|
||||||
|
+ " <xsl:template match=\"node()|@*\" priority=\"1\">\n"
|
||||||
|
+ " <xsl:copy>\n"
|
||||||
|
+ " <xsl:apply-templates select=\"node()|@*\"/>\n"
|
||||||
|
+ " </xsl:copy>\n"
|
||||||
|
+ " </xsl:template>\n"
|
||||||
|
+ " <xsl:template match=\"*\" priority=\"2\">\n"
|
||||||
|
+ " <xsl:element name=\"{name()}\" namespace=\"{namespace-uri()}\">\n"
|
||||||
|
+ " <xsl:apply-templates select=\"node()|@*\"/>\n"
|
||||||
|
+ " </xsl:element>\n"
|
||||||
|
+ " </xsl:template>\n"
|
||||||
|
+ "</xsl:stylesheet>";
|
||||||
|
|
||||||
|
// Simple XML content with root and two child elements
|
||||||
|
private final static String XML = "<TestRoot xmlns=\"test.xmlns\">\n"
|
||||||
|
+ " <Element1 xmlns=\"test.xmlns\">\n"
|
||||||
|
+ " </Element1>\n"
|
||||||
|
+ " <Element2 xmlns=\"test.xmlns\">\n"
|
||||||
|
+ " </Element2>\n"
|
||||||
|
+ "</TestRoot>";
|
||||||
|
|
||||||
|
// With thread local namespace prefix index each transformation result should
|
||||||
|
// be the same and contain the same prefix for Element2
|
||||||
|
private final static String EXPECTED_CONTENT = "</ns2:Element2>";
|
||||||
|
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user