8008738: Issue in com.sun.org.apache.xml.internal.serializer.Encodings causes some JCK tests to fail intermittently

Encodings.java sometimes creates EncodingInfo objects whose java names are not recognized by the Charset API. This patch fixes that issue.

Reviewed-by: joehw, alanb
This commit is contained in:
Daniel Fuchs 2013-05-06 18:50:16 +02:00
parent 876bc1c753
commit c76e4b0e46

@ -33,6 +33,14 @@ import java.util.Enumeration;
import java.util.HashMap; import java.util.HashMap;
import java.util.Properties; import java.util.Properties;
import java.util.StringTokenizer; import java.util.StringTokenizer;
import java.io.IOException;
import java.net.MalformedURLException;
import java.nio.charset.Charset;
import java.nio.charset.IllegalCharsetNameException;
import java.nio.charset.UnsupportedCharsetException;
import java.util.Collections;
import java.util.Map;
import java.util.Map.Entry;
import com.sun.org.apache.xalan.internal.utils.SecuritySupport; import com.sun.org.apache.xalan.internal.utils.SecuritySupport;
@ -79,37 +87,18 @@ public final class Encodings extends Object
throws UnsupportedEncodingException throws UnsupportedEncodingException
{ {
for (int i = 0; i < _encodings.length; ++i) final EncodingInfo ei = _encodingInfos.findEncoding(toUpperCaseFast(encoding));
{ if (ei != null) {
if (_encodings[i].name.equalsIgnoreCase(encoding)) try {
{
try
{
return new BufferedWriter(new OutputStreamWriter( return new BufferedWriter(new OutputStreamWriter(
output, output, ei.javaName));
_encodings[i].javaName)); } catch (UnsupportedEncodingException usee) {
}
catch (java.lang.IllegalArgumentException iae) // java 1.1.8
{
// keep trying
}
catch (UnsupportedEncodingException usee)
{
// keep trying // keep trying
} }
} }
}
try
{
return new BufferedWriter(new OutputStreamWriter(output, encoding)); return new BufferedWriter(new OutputStreamWriter(output, encoding));
} }
catch (java.lang.IllegalArgumentException iae) // java 1.1.8
{
throw new UnsupportedEncodingException(encoding);
}
}
/** /**
@ -141,13 +130,25 @@ public final class Encodings extends Object
EncodingInfo ei; EncodingInfo ei;
String normalizedEncoding = toUpperCaseFast(encoding); String normalizedEncoding = toUpperCaseFast(encoding);
ei = (EncodingInfo) _encodingTableKeyJava.get(normalizedEncoding); ei = _encodingInfos.findEncoding(normalizedEncoding);
if (ei == null)
ei = (EncodingInfo) _encodingTableKeyMime.get(normalizedEncoding);
if (ei == null) { if (ei == null) {
// We shouldn't have to do this, but just in case. // We shouldn't have to do this, but just in case.
try {
// This may happen if the caller tries to use
// an encoding that wasn't registered in the
// (java name)->(preferred mime name) mapping file.
// In that case we attempt to load the charset for the
// given encoding, and if that succeeds - we create a new
// EncodingInfo instance - assuming the canonical name
// of the charset can be used as the mime name.
final Charset c = Charset.forName(encoding);
final String name = c.name();
ei = new EncodingInfo(name, name);
_encodingInfos.putEncoding(normalizedEncoding, ei);
} catch (IllegalCharsetNameException | UnsupportedCharsetException x) {
ei = new EncodingInfo(null,null); ei = new EncodingInfo(null,null);
} }
}
return ei; return ei;
} }
@ -269,8 +270,8 @@ public final class Encodings extends Object
*/ */
private static String convertJava2MimeEncoding(String encoding) private static String convertJava2MimeEncoding(String encoding)
{ {
EncodingInfo enc = final EncodingInfo enc =
(EncodingInfo) _encodingTableKeyJava.get(encoding.toUpperCase()); _encodingInfos.getEncodingFromJavaKey(toUpperCaseFast(encoding));
if (null != enc) if (null != enc)
return enc.name; return enc.name;
return encoding; return encoding;
@ -285,38 +286,37 @@ public final class Encodings extends Object
*/ */
public static String convertMime2JavaEncoding(String encoding) public static String convertMime2JavaEncoding(String encoding)
{ {
final EncodingInfo info = _encodingInfos.findEncoding(toUpperCaseFast(encoding));
for (int i = 0; i < _encodings.length; ++i) return info != null ? info.javaName : encoding;
{
if (_encodings[i].name.equalsIgnoreCase(encoding))
{
return _encodings[i].javaName;
}
} }
return encoding; // Using an inner static class here prevent initialization races
// where the hash maps could be used before they were populated.
//
private final static class EncodingInfos {
// These maps are final and not modified after initialization.
private final Map<String, EncodingInfo> _encodingTableKeyJava = new HashMap<>();
private final Map<String, EncodingInfo> _encodingTableKeyMime = new HashMap<>();
// This map will be added to after initialization: make sure it's
// thread-safe. This map should not be used frequently - only in cases
// where the mapping requested was not declared in the Encodings.properties
// file.
private final Map<String, EncodingInfo> _encodingDynamicTable =
Collections.synchronizedMap(new HashMap<String, EncodingInfo>());
private EncodingInfos() {
loadEncodingInfo();
} }
/** // Opens the file/resource containing java charset name -> preferred mime
* Load a list of all the supported encodings. // name mapping and returns it as an InputStream.
* private InputStream openEncodingsFileStream() throws MalformedURLException, IOException {
* System property "encodings" formatted using URL syntax may define an
* external encodings list. Thanks to Sergey Ushakov for the code
* contribution!
*/
private static EncodingInfo[] loadEncodingInfo()
{
try
{
String urlString = null; String urlString = null;
InputStream is = null; InputStream is = null;
try try {
{
urlString = SecuritySupport.getSystemProperty(ENCODINGS_PROP, ""); urlString = SecuritySupport.getSystemProperty(ENCODINGS_PROP, "");
} } catch (SecurityException e) {
catch (SecurityException e)
{
} }
if (urlString != null && urlString.length() > 0) { if (urlString != null && urlString.length() > 0) {
@ -327,11 +327,17 @@ public final class Encodings extends Object
if (is == null) { if (is == null) {
is = SecuritySupport.getResourceAsStream(ENCODINGS_FILE); is = SecuritySupport.getResourceAsStream(ENCODINGS_FILE);
} }
return is;
}
// Loads the Properties resource containing the mapping:
// java charset name -> preferred mime name
// and returns it.
private Properties loadProperties() throws MalformedURLException, IOException {
Properties props = new Properties(); Properties props = new Properties();
try (InputStream is = openEncodingsFileStream()) {
if (is != null) { if (is != null) {
props.load(is); props.load(is);
is.close();
} else { } else {
// Seems to be no real need to force failure here, let the // Seems to be no real need to force failure here, let the
// system do its best... The issue is not really very critical, // system do its best... The issue is not really very critical,
@ -340,74 +346,172 @@ public final class Encodings extends Object
// But maybe report/log the resource problem? // But maybe report/log the resource problem?
// Any standard ways to report/log errors (in static context)? // Any standard ways to report/log errors (in static context)?
} }
int totalEntries = props.size();
int totalMimeNames = 0;
Enumeration keys = props.keys();
for (int i = 0; i < totalEntries; ++i)
{
String javaName = (String) keys.nextElement();
String val = props.getProperty(javaName);
totalMimeNames++;
int pos = val.indexOf(' ');
for (int j = 0; j < pos; ++j)
if (val.charAt(j) == ',')
totalMimeNames++;
} }
EncodingInfo[] ret = new EncodingInfo[totalMimeNames]; return props;
int j = 0; }
keys = props.keys();
for (int i = 0; i < totalEntries; ++i) // Parses the mime list associated to a java charset name.
{ // The first mime name in the list is supposed to be the preferred
String javaName = (String) keys.nextElement(); // mime name.
String val = props.getProperty(javaName); private String[] parseMimeTypes(String val) {
int pos = val.indexOf(' '); int pos = val.indexOf(' ');
String mimeName;
//int lastPrintable; //int lastPrintable;
if (pos < 0) if (pos < 0) {
{
// Maybe report/log this problem? // Maybe report/log this problem?
// "Last printable character not defined for encoding " + // "Last printable character not defined for encoding " +
// mimeName + " (" + val + ")" ... // mimeName + " (" + val + ")" ...
mimeName = val; return new String[] { val };
//lastPrintable = 0x00FF; //lastPrintable = 0x00FF;
} }
else
{
//lastPrintable = //lastPrintable =
// Integer.decode(val.substring(pos).trim()).intValue(); // Integer.decode(val.substring(pos).trim()).intValue();
StringTokenizer st = StringTokenizer st =
new StringTokenizer(val.substring(0, pos), ","); new StringTokenizer(val.substring(0, pos), ",");
for (boolean first = true; String[] values = new String[st.countTokens()];
st.hasMoreTokens(); for (int i=0; st.hasMoreTokens(); i++) {
first = false) values[i] = st.nextToken();
{ }
mimeName = st.nextToken(); return values;
ret[j] = }
new EncodingInfo(mimeName, javaName);
_encodingTableKeyMime.put( // This method here attempts to find the canonical charset name for the
mimeName.toUpperCase(), // the given name - which is supposed to be either a java name or a mime
ret[j]); // name.
if (first) // For that, it attempts to load the charset using the given name, and
_encodingTableKeyJava.put( // then returns the charset's canonical name.
javaName.toUpperCase(), // If the charset could not be loaded from the given name,
ret[j]); // the method returns null.
j++; private String findCharsetNameFor(String name) {
try {
return Charset.forName(name).name();
} catch (Exception x) {
return null;
} }
} }
// This method here attempts to find the canonical charset name for the
// the set javaName+mimeNames - which are supposed to all refer to the
// same charset.
// For that it attempts to load the charset using the javaName, and if
// not found, attempts again using each of the mime names in turn.
// If the charset could be loaded from the javaName, then the javaName
// itself is returned as charset name. Otherwise, each of the mime names
// is tried in turn, until a charset can be loaded from one of the names,
// and the loaded charset's canonical name is returned.
// If no charset can be loaded from either the javaName or one of the
// mime names, then null is returned.
//
// Note that the returned name is the 'java' name that will be used in
// instances of EncodingInfo.
// This is important because EncodingInfo uses that 'java name' later on
// in calls to String.getBytes(javaName).
// As it happens, sometimes only one element of the set mime names/javaName
// is known by Charset: sometimes only one of the mime names is known,
// sometime only the javaName is known, sometimes all are known.
//
// By using this method here, we fix the problem where one of the mime
// names is known but the javaName is unknown, by associating the charset
// loaded from one of the mime names with the unrecognized javaName.
//
// When none of the mime names or javaName are known - there's not much we can
// do... It can mean that this encoding is not supported for this
// OS. If such a charset is ever use it will result in having all characters
// escaped.
//
private String findCharsetNameFor(String javaName, String[] mimes) {
String cs = findCharsetNameFor(javaName);
if (cs != null) return javaName;
for (String m : mimes) {
cs = findCharsetNameFor(m);
if (cs != null) break;
} }
return ret; return cs;
} }
catch (java.net.MalformedURLException mue)
{ /**
* Loads a list of all the supported encodings.
*
* System property "encodings" formatted using URL syntax may define an
* external encodings list. Thanks to Sergey Ushakov for the code
* contribution!
*/
private void loadEncodingInfo() {
try {
// load (java name)->(preferred mime name) mapping.
final Properties props = loadProperties();
// create instances of EncodingInfo from the loaded mapping
Enumeration keys = props.keys();
Map<String, EncodingInfo> canonicals = new HashMap<>();
while (keys.hasMoreElements()) {
final String javaName = (String) keys.nextElement();
final String[] mimes = parseMimeTypes(props.getProperty(javaName));
final String charsetName = findCharsetNameFor(javaName, mimes);
if (charsetName != null) {
final String kj = toUpperCaseFast(javaName);
final String kc = toUpperCaseFast(charsetName);
for (int i = 0; i < mimes.length; ++i) {
final String mimeName = mimes[i];
final String km = toUpperCaseFast(mimeName);
EncodingInfo info = new EncodingInfo(mimeName, charsetName);
_encodingTableKeyMime.put(km, info);
if (!canonicals.containsKey(kc)) {
// canonicals will map the charset name to
// the info containing the prefered mime name
// (the preferred mime name is the first mime
// name in the list).
canonicals.put(kc, info);
_encodingTableKeyJava.put(kc, info);
}
_encodingTableKeyJava.put(kj, info);
}
} else {
// None of the java or mime names on the line were
// recognized => this charset is not supported?
}
}
// Fix up the _encodingTableKeyJava so that the info mapped to
// the java name contains the preferred mime name.
// (a given java name can correspond to several mime name,
// but we want the _encodingTableKeyJava to point to the
// preferred mime name).
for (Entry<String, EncodingInfo> e : _encodingTableKeyJava.entrySet()) {
e.setValue(canonicals.get(toUpperCaseFast(e.getValue().javaName)));
}
} catch (java.net.MalformedURLException mue) {
throw new com.sun.org.apache.xml.internal.serializer.utils.WrappedRuntimeException(mue); throw new com.sun.org.apache.xml.internal.serializer.utils.WrappedRuntimeException(mue);
} } catch (java.io.IOException ioe) {
catch (java.io.IOException ioe)
{
throw new com.sun.org.apache.xml.internal.serializer.utils.WrappedRuntimeException(ioe); throw new com.sun.org.apache.xml.internal.serializer.utils.WrappedRuntimeException(ioe);
} }
} }
EncodingInfo findEncoding(String normalizedEncoding) {
EncodingInfo info = _encodingTableKeyJava.get(normalizedEncoding);
if (info == null) {
info = _encodingTableKeyMime.get(normalizedEncoding);
}
if (info == null) {
info = _encodingDynamicTable.get(normalizedEncoding);
}
return info;
}
EncodingInfo getEncodingFromMimeKey(String normalizedMimeName) {
return _encodingTableKeyMime.get(normalizedMimeName);
}
EncodingInfo getEncodingFromJavaKey(String normalizedJavaName) {
return _encodingTableKeyJava.get(normalizedJavaName);
}
void putEncoding(String key, EncodingInfo info) {
_encodingDynamicTable.put(key, info);
}
}
/** /**
* Return true if the character is the high member of a surrogate pair. * Return true if the character is the high member of a surrogate pair.
* <p> * <p>
@ -457,7 +561,6 @@ public final class Encodings extends Object
return codePoint; return codePoint;
} }
private static final HashMap _encodingTableKeyJava = new HashMap(); private final static EncodingInfos _encodingInfos = new EncodingInfos();
private static final HashMap _encodingTableKeyMime = new HashMap();
private static final EncodingInfo[] _encodings = loadEncodingInfo();
} }