--- /dev/null Thu Jan 01 00:00:00 1970 +0000
+++ b/src/java.naming/share/classes/javax/naming/NameImpl.java Tue Sep 12 19:03:39 2017 +0200
@@ -0,0 +1,735 @@
+/*
+ * Copyright (c) 1999, 2011, 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. Oracle designates this
+ * particular file as subject to the "Classpath" exception as provided
+ * by Oracle in the LICENSE file that accompanied this code.
+ *
+ * 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 javax.naming;
+
+import java.util.Locale;
+import java.util.Vector;
+import java.util.Enumeration;
+import java.util.Properties;
+import java.util.NoSuchElementException;
+
+/**
+ * The implementation class for CompoundName and CompositeName.
+ * This class is package private.
+ *
+ * @author Rosanna Lee
+ * @author Scott Seligman
+ * @author Aravindan Ranganathan
+ * @since 1.3
+ */
+
+class NameImpl {
+ private static final byte LEFT_TO_RIGHT = 1;
+ private static final byte RIGHT_TO_LEFT = 2;
+ private static final byte FLAT = 0;
+
+ private Vector<String> components;
+
+ private byte syntaxDirection = LEFT_TO_RIGHT;
+ private String syntaxSeparator = "/";
+ private String syntaxSeparator2 = null;
+ private boolean syntaxCaseInsensitive = false;
+ private boolean syntaxTrimBlanks = false;
+ private String syntaxEscape = "\\";
+ private String syntaxBeginQuote1 = "\"";
+ private String syntaxEndQuote1 = "\"";
+ private String syntaxBeginQuote2 = "'";
+ private String syntaxEndQuote2 = "'";
+ private String syntaxAvaSeparator = null;
+ private String syntaxTypevalSeparator = null;
+
+ // escapingStyle gives the method used at creation time for
+ // quoting or escaping characters in the name. It is set to the
+ // first style of quote or escape encountered if and when the name
+ // is parsed.
+ private static final int STYLE_NONE = 0;
+ private static final int STYLE_QUOTE1 = 1;
+ private static final int STYLE_QUOTE2 = 2;
+ private static final int STYLE_ESCAPE = 3;
+ private int escapingStyle = STYLE_NONE;
+
+ // Returns true if "match" is not null, and n contains "match" at
+ // position i.
+ private final boolean isA(String n, int i, String match) {
+ return (match != null && n.startsWith(match, i));
+ }
+
+ private final boolean isMeta(String n, int i) {
+ return (isA(n, i, syntaxEscape) ||
+ isA(n, i, syntaxBeginQuote1) ||
+ isA(n, i, syntaxBeginQuote2) ||
+ isSeparator(n, i));
+ }
+
+ private final boolean isSeparator(String n, int i) {
+ return (isA(n, i, syntaxSeparator) ||
+ isA(n, i, syntaxSeparator2));
+ }
+
+ private final int skipSeparator(String name, int i) {
+ if (isA(name, i, syntaxSeparator)) {
+ i += syntaxSeparator.length();
+ } else if (isA(name, i, syntaxSeparator2)) {
+ i += syntaxSeparator2.length();
+ }
+ return (i);
+ }
+
+ private final int extractComp(String name, int i, int len, Vector<String> comps)
+ throws InvalidNameException {
+ String beginQuote;
+ String endQuote;
+ boolean start = true;
+ boolean one = false;
+ StringBuilder answer = new StringBuilder(len);
+
+ while (i < len) {
+ // handle quoted strings
+ if (start && ((one = isA(name, i, syntaxBeginQuote1)) ||
+ isA(name, i, syntaxBeginQuote2))) {
+
+ // record choice of quote chars being used
+ beginQuote = one ? syntaxBeginQuote1 : syntaxBeginQuote2;
+ endQuote = one ? syntaxEndQuote1 : syntaxEndQuote2;
+ if (escapingStyle == STYLE_NONE) {
+ escapingStyle = one ? STYLE_QUOTE1 : STYLE_QUOTE2;
+ }
+
+ // consume string until matching quote
+ for (i += beginQuote.length();
+ ((i < len) && !name.startsWith(endQuote, i));
+ i++) {
+ // skip escape character if it is escaping ending quote
+ // otherwise leave as is.
+ if (isA(name, i, syntaxEscape) &&
+ isA(name, i + syntaxEscape.length(), endQuote)) {
+ i += syntaxEscape.length();
+ }
+ answer.append(name.charAt(i)); // copy char
+ }
+
+ // no ending quote found
+ if (i >= len)
+ throw
+ new InvalidNameException(name + ": no close quote");
+// new Exception("no close quote");
+
+ i += endQuote.length();
+
+ // verify that end-quote occurs at separator or end of string
+ if (i == len || isSeparator(name, i)) {
+ break;
+ }
+// throw (new Exception(
+ throw (new InvalidNameException(name +
+ ": close quote appears before end of component"));
+
+ } else if (isSeparator(name, i)) {
+ break;
+
+ } else if (isA(name, i, syntaxEscape)) {
+ if (isMeta(name, i + syntaxEscape.length())) {
+ // if escape precedes meta, consume escape and let
+ // meta through
+ i += syntaxEscape.length();
+ if (escapingStyle == STYLE_NONE) {
+ escapingStyle = STYLE_ESCAPE;
+ }
+ } else if (i + syntaxEscape.length() >= len) {
+ throw (new InvalidNameException(name +
+ ": unescaped " + syntaxEscape + " at end of component"));
+ }
+ } else if (isA(name, i, syntaxTypevalSeparator) &&
+ ((one = isA(name, i+syntaxTypevalSeparator.length(), syntaxBeginQuote1)) ||
+ isA(name, i+syntaxTypevalSeparator.length(), syntaxBeginQuote2))) {
+ // Handle quote occurring after typeval separator
+ beginQuote = one ? syntaxBeginQuote1 : syntaxBeginQuote2;
+ endQuote = one ? syntaxEndQuote1 : syntaxEndQuote2;
+
+ i += syntaxTypevalSeparator.length();
+ answer.append(syntaxTypevalSeparator).append(beginQuote); // add back
+
+ // consume string until matching quote
+ for (i += beginQuote.length();
+ ((i < len) && !name.startsWith(endQuote, i));
+ i++) {
+ // skip escape character if it is escaping ending quote
+ // otherwise leave as is.
+ if (isA(name, i, syntaxEscape) &&
+ isA(name, i + syntaxEscape.length(), endQuote)) {
+ i += syntaxEscape.length();
+ }
+ answer.append(name.charAt(i)); // copy char
+ }
+
+ // no ending quote found
+ if (i >= len)
+ throw
+ new InvalidNameException(name + ": typeval no close quote");
+
+ i += endQuote.length();
+ answer.append(endQuote); // add back
+
+ // verify that end-quote occurs at separator or end of string
+ if (i == len || isSeparator(name, i)) {
+ break;
+ }
+ throw (new InvalidNameException(name.substring(i) +
+ ": typeval close quote appears before end of component"));
+ }
+
+ answer.append(name.charAt(i++));
+ start = false;
+ }
+
+ if (syntaxDirection == RIGHT_TO_LEFT)
+ comps.insertElementAt(answer.toString(), 0);
+ else
+ comps.addElement(answer.toString());
+ return i;
+ }
+
+ private static boolean getBoolean(Properties p, String name) {
+ return toBoolean(p.getProperty(name));
+ }
+
+ private static boolean toBoolean(String name) {
+ return ((name != null) &&
+ name.toLowerCase(Locale.ENGLISH).equals("true"));
+ }
+
+ private final void recordNamingConvention(Properties p) {
+ String syntaxDirectionStr =
+ p.getProperty("jndi.syntax.direction", "flat");
+ if (syntaxDirectionStr.equals("left_to_right")) {
+ syntaxDirection = LEFT_TO_RIGHT;
+ } else if (syntaxDirectionStr.equals("right_to_left")) {
+ syntaxDirection = RIGHT_TO_LEFT;
+ } else if (syntaxDirectionStr.equals("flat")) {
+ syntaxDirection = FLAT;
+ } else {
+ throw new IllegalArgumentException(syntaxDirectionStr +
+ " is not a valid value for the jndi.syntax.direction property");
+ }
+
+ if (syntaxDirection != FLAT) {
+ syntaxSeparator = p.getProperty("jndi.syntax.separator");
+ syntaxSeparator2 = p.getProperty("jndi.syntax.separator2");
+ if (syntaxSeparator == null) {
+ throw new IllegalArgumentException(
+ "jndi.syntax.separator property required for non-flat syntax");
+ }
+ } else {
+ syntaxSeparator = null;
+ }
+ syntaxEscape = p.getProperty("jndi.syntax.escape");
+
+ syntaxCaseInsensitive = getBoolean(p, "jndi.syntax.ignorecase");
+ syntaxTrimBlanks = getBoolean(p, "jndi.syntax.trimblanks");
+
+ syntaxBeginQuote1 = p.getProperty("jndi.syntax.beginquote");
+ syntaxEndQuote1 = p.getProperty("jndi.syntax.endquote");
+ if (syntaxEndQuote1 == null && syntaxBeginQuote1 != null)
+ syntaxEndQuote1 = syntaxBeginQuote1;
+ else if (syntaxBeginQuote1 == null && syntaxEndQuote1 != null)
+ syntaxBeginQuote1 = syntaxEndQuote1;
+ syntaxBeginQuote2 = p.getProperty("jndi.syntax.beginquote2");
+ syntaxEndQuote2 = p.getProperty("jndi.syntax.endquote2");
+ if (syntaxEndQuote2 == null && syntaxBeginQuote2 != null)
+ syntaxEndQuote2 = syntaxBeginQuote2;
+ else if (syntaxBeginQuote2 == null && syntaxEndQuote2 != null)
+ syntaxBeginQuote2 = syntaxEndQuote2;
+
+ syntaxAvaSeparator = p.getProperty("jndi.syntax.separator.ava");
+ syntaxTypevalSeparator =
+ p.getProperty("jndi.syntax.separator.typeval");
+ }
+
+ NameImpl(Properties syntax) {
+ if (syntax != null) {
+ recordNamingConvention(syntax);
+ }
+ components = new Vector<>();
+ }
+
+ NameImpl(Properties syntax, String n) throws InvalidNameException {
+ this(syntax);
+
+ boolean rToL = (syntaxDirection == RIGHT_TO_LEFT);
+ boolean compsAllEmpty = true;
+ int len = n.length();
+
+ for (int i = 0; i < len; ) {
+ i = extractComp(n, i, len, components);
+
+ String comp = rToL
+ ? components.firstElement()
+ : components.lastElement();
+ if (comp.length() >= 1) {
+ compsAllEmpty = false;
+ }
+
+ if (i < len) {
+ i = skipSeparator(n, i);
+ if ((i == len) && !compsAllEmpty) {
+ // Trailing separator found. Add an empty component.
+ if (rToL) {
+ components.insertElementAt("", 0);
+ } else {
+ components.addElement("");
+ }
+ }
+ }
+ }
+ }
+
+ NameImpl(Properties syntax, Enumeration<String> comps) {
+ this(syntax);
+
+ // %% comps could shrink in the middle.
+ while (comps.hasMoreElements())
+ components.addElement(comps.nextElement());
+ }
+/*
+ // Determines whether this component needs any escaping.
+ private final boolean escapingNeeded(String comp) {
+ int len = comp.length();
+ for (int i = 0; i < len; i++) {
+ if (i == 0) {
+ if (isA(comp, 0, syntaxBeginQuote1) ||
+ isA(comp, 0, syntaxBeginQuote2)) {
+ return (true);
+ }
+ }
+ if (isSeparator(comp, i)) {
+ return (true);
+ }
+ if (isA(comp, i, syntaxEscape)) {
+ i += syntaxEscape.length();
+ if (i >= len || isMeta(comp, i)) {
+ return (true);
+ }
+ }
+ }
+ return (false);
+ }
+*/
+ private final String stringifyComp(String comp) {
+ int len = comp.length();
+ boolean escapeSeparator = false, escapeSeparator2 = false;
+ String beginQuote = null, endQuote = null;
+ StringBuffer strbuf = new StringBuffer(len);
+
+ // determine whether there are any separators; if so escape
+ // or quote them
+ if (syntaxSeparator != null &&
+ comp.indexOf(syntaxSeparator) >= 0) {
+ if (syntaxBeginQuote1 != null) {
+ beginQuote = syntaxBeginQuote1;
+ endQuote = syntaxEndQuote1;
+ } else if (syntaxBeginQuote2 != null) {
+ beginQuote = syntaxBeginQuote2;
+ endQuote = syntaxEndQuote2;
+ } else if (syntaxEscape != null)
+ escapeSeparator = true;
+ }
+ if (syntaxSeparator2 != null &&
+ comp.indexOf(syntaxSeparator2) >= 0) {
+ if (syntaxBeginQuote1 != null) {
+ if (beginQuote == null) {
+ beginQuote = syntaxBeginQuote1;
+ endQuote = syntaxEndQuote1;
+ }
+ } else if (syntaxBeginQuote2 != null) {
+ if (beginQuote == null) {
+ beginQuote = syntaxBeginQuote2;
+ endQuote = syntaxEndQuote2;
+ }
+ } else if (syntaxEscape != null)
+ escapeSeparator2 = true;
+ }
+
+ // if quoting component,
+ if (beginQuote != null) {
+
+ // start string off with opening quote
+ strbuf = strbuf.append(beginQuote);
+
+ // component is being quoted, so we only need to worry about
+ // escaping end quotes that occur in component
+ for (int i = 0; i < len; ) {
+ if (comp.startsWith(endQuote, i)) {
+ // end-quotes must be escaped when inside a quoted string
+ strbuf.append(syntaxEscape).append(endQuote);
+ i += endQuote.length();
+ } else {
+ // no special treatment required
+ strbuf.append(comp.charAt(i++));
+ }
+ }
+
+ // end with closing quote
+ strbuf.append(endQuote);
+
+ } else {
+
+ // When component is not quoted, add escape for:
+ // 1. leading quote
+ // 2. an escape preceding any meta char
+ // 3. an escape at the end of a component
+ // 4. separator
+
+ // go through characters in component and escape where necessary
+ boolean start = true;
+ for (int i = 0; i < len; ) {
+ // leading quote must be escaped
+ if (start && isA(comp, i, syntaxBeginQuote1)) {
+ strbuf.append(syntaxEscape).append(syntaxBeginQuote1);
+ i += syntaxBeginQuote1.length();
+ } else if (start && isA(comp, i, syntaxBeginQuote2)) {
+ strbuf.append(syntaxEscape).append(syntaxBeginQuote2);
+ i += syntaxBeginQuote2.length();
+ } else
+
+ // Escape an escape preceding meta characters, or at end.
+ // Other escapes pass through.
+ if (isA(comp, i, syntaxEscape)) {
+ if (i + syntaxEscape.length() >= len) {
+ // escape an ending escape
+ strbuf.append(syntaxEscape);
+ } else if (isMeta(comp, i + syntaxEscape.length())) {
+ // escape meta strings
+ strbuf.append(syntaxEscape);
+ }
+ strbuf.append(syntaxEscape);
+ i += syntaxEscape.length();
+ } else
+
+ // escape unescaped separator
+ if (escapeSeparator && comp.startsWith(syntaxSeparator, i)) {
+ // escape separator
+ strbuf.append(syntaxEscape).append(syntaxSeparator);
+ i += syntaxSeparator.length();
+ } else if (escapeSeparator2 &&
+ comp.startsWith(syntaxSeparator2, i)) {
+ // escape separator2
+ strbuf.append(syntaxEscape).append(syntaxSeparator2);
+ i += syntaxSeparator2.length();
+ } else {
+ // no special treatment required
+ strbuf.append(comp.charAt(i++));
+ }
+ start = false;
+ }
+ }
+ return (strbuf.toString());
+ }
+
+ public String toString() {
+ StringBuffer answer = new StringBuffer();
+ String comp;
+ boolean compsAllEmpty = true;
+ int size = components.size();
+
+ for (int i = 0; i < size; i++) {
+ if (syntaxDirection == RIGHT_TO_LEFT) {
+ comp =
+ stringifyComp(components.elementAt(size - 1 - i));
+ } else {
+ comp = stringifyComp(components.elementAt(i));
+ }
+ if ((i != 0) && (syntaxSeparator != null))
+ answer.append(syntaxSeparator);
+ if (comp.length() >= 1)
+ compsAllEmpty = false;
+ answer = answer.append(comp);
+ }
+ if (compsAllEmpty && (size >= 1) && (syntaxSeparator != null))
+ answer = answer.append(syntaxSeparator);
+ return (answer.toString());
+ }
+
+ public boolean equals(Object obj) {
+ if ((obj != null) && (obj instanceof NameImpl)) {
+ NameImpl target = (NameImpl)obj;
+ if (target.size() == this.size()) {
+ Enumeration<String> mycomps = getAll();
+ Enumeration<String> comps = target.getAll();
+ while (mycomps.hasMoreElements()) {
+ // %% comps could shrink in the middle.
+ String my = mycomps.nextElement();
+ String his = comps.nextElement();
+ if (syntaxTrimBlanks) {
+ my = my.trim();
+ his = his.trim();
+ }
+ if (syntaxCaseInsensitive) {
+ if (!(my.equalsIgnoreCase(his)))
+ return false;
+ } else {
+ if (!(my.equals(his)))
+ return false;
+ }
+ }
+ return true;
+ }
+ }
+ return false;
+ }
+
+ /**
+ * Compares obj to this NameImpl to determine ordering.
+ * Takes into account syntactic properties such as
+ * elimination of blanks, case-ignore, etc, if relevant.
+ *
+ * Note: using syntax of this NameImpl and ignoring
+ * that of comparison target.
+ */
+ public int compareTo(NameImpl obj) {
+ if (this == obj) {
+ return 0;
+ }
+
+ int len1 = size();
+ int len2 = obj.size();
+ int n = Math.min(len1, len2);
+
+ int index1 = 0, index2 = 0;
+
+ while (n-- != 0) {
+ String comp1 = get(index1++);
+ String comp2 = obj.get(index2++);
+
+ // normalize according to syntax
+ if (syntaxTrimBlanks) {
+ comp1 = comp1.trim();
+ comp2 = comp2.trim();
+ }
+
+ int local;
+ if (syntaxCaseInsensitive) {
+ local = comp1.compareToIgnoreCase(comp2);
+ } else {
+ local = comp1.compareTo(comp2);
+ }
+
+ if (local != 0) {
+ return local;
+ }
+ }
+
+ return len1 - len2;
+ }
+
+ public int size() {
+ return (components.size());
+ }
+
+ public Enumeration<String> getAll() {
+ return components.elements();
+ }
+
+ public String get(int posn) {
+ return components.elementAt(posn);
+ }
+
+ public Enumeration<String> getPrefix(int posn) {
+ if (posn < 0 || posn > size()) {
+ throw new ArrayIndexOutOfBoundsException(posn);
+ }
+ return new NameImplEnumerator(components, 0, posn);
+ }
+
+ public Enumeration<String> getSuffix(int posn) {
+ int cnt = size();
+ if (posn < 0 || posn > cnt) {
+ throw new ArrayIndexOutOfBoundsException(posn);
+ }
+ return new NameImplEnumerator(components, posn, cnt);
+ }
+
+ public boolean isEmpty() {
+ return (components.isEmpty());
+ }
+
+ public boolean startsWith(int posn, Enumeration<String> prefix) {
+ if (posn < 0 || posn > size()) {
+ return false;
+ }
+ try {
+ Enumeration<String> mycomps = getPrefix(posn);
+ while (mycomps.hasMoreElements()) {
+ String my = mycomps.nextElement();
+ String his = prefix.nextElement();
+ if (syntaxTrimBlanks) {
+ my = my.trim();
+ his = his.trim();
+ }
+ if (syntaxCaseInsensitive) {
+ if (!(my.equalsIgnoreCase(his)))
+ return false;
+ } else {
+ if (!(my.equals(his)))
+ return false;
+ }
+ }
+ } catch (NoSuchElementException e) {
+ return false;
+ }
+ return true;
+ }
+
+ public boolean endsWith(int posn, Enumeration<String> suffix) {
+ // posn is number of elements in suffix
+ // startIndex is the starting position in this name
+ // at which to start the comparison. It is calculated by
+ // subtracting 'posn' from size()
+ int startIndex = size() - posn;
+ if (startIndex < 0 || startIndex > size()) {
+ return false;
+ }
+ try {
+ Enumeration<String> mycomps = getSuffix(startIndex);
+ while (mycomps.hasMoreElements()) {
+ String my = mycomps.nextElement();
+ String his = suffix.nextElement();
+ if (syntaxTrimBlanks) {
+ my = my.trim();
+ his = his.trim();
+ }
+ if (syntaxCaseInsensitive) {
+ if (!(my.equalsIgnoreCase(his)))
+ return false;
+ } else {
+ if (!(my.equals(his)))
+ return false;
+ }
+ }
+ } catch (NoSuchElementException e) {
+ return false;
+ }
+ return true;
+ }
+
+ public boolean addAll(Enumeration<String> comps) throws InvalidNameException {
+ boolean added = false;
+ while (comps.hasMoreElements()) {
+ try {
+ String comp = comps.nextElement();
+ if (size() > 0 && syntaxDirection == FLAT) {
+ throw new InvalidNameException(
+ "A flat name can only have a single component");
+ }
+ components.addElement(comp);
+ added = true;
+ } catch (NoSuchElementException e) {
+ break; // "comps" has shrunk.
+ }
+ }
+ return added;
+ }
+
+ public boolean addAll(int posn, Enumeration<String> comps)
+ throws InvalidNameException {
+ boolean added = false;
+ for (int i = posn; comps.hasMoreElements(); i++) {
+ try {
+ String comp = comps.nextElement();
+ if (size() > 0 && syntaxDirection == FLAT) {
+ throw new InvalidNameException(
+ "A flat name can only have a single component");
+ }
+ components.insertElementAt(comp, i);
+ added = true;
+ } catch (NoSuchElementException e) {
+ break; // "comps" has shrunk.
+ }
+ }
+ return added;
+ }
+
+ public void add(String comp) throws InvalidNameException {
+ if (size() > 0 && syntaxDirection == FLAT) {
+ throw new InvalidNameException(
+ "A flat name can only have a single component");
+ }
+ components.addElement(comp);
+ }
+
+ public void add(int posn, String comp) throws InvalidNameException {
+ if (size() > 0 && syntaxDirection == FLAT) {
+ throw new InvalidNameException(
+ "A flat name can only zero or one component");
+ }
+ components.insertElementAt(comp, posn);
+ }
+
+ public Object remove(int posn) {
+ Object r = components.elementAt(posn);
+ components.removeElementAt(posn);
+ return r;
+ }
+
+ public int hashCode() {
+ int hash = 0;
+ for (Enumeration<String> e = getAll(); e.hasMoreElements();) {
+ String comp = e.nextElement();
+ if (syntaxTrimBlanks) {
+ comp = comp.trim();
+ }
+ if (syntaxCaseInsensitive) {
+ comp = comp.toLowerCase(Locale.ENGLISH);
+ }
+
+ hash += comp.hashCode();
+ }
+ return hash;
+ }
+}
+
+final
+class NameImplEnumerator implements Enumeration<String> {
+ Vector<String> vector;
+ int count;
+ int limit;
+
+ NameImplEnumerator(Vector<String> v, int start, int lim) {
+ vector = v;
+ count = start;
+ limit = lim;
+ }
+
+ public boolean hasMoreElements() {
+ return count < limit;
+ }
+
+ public String nextElement() {
+ if (count < limit) {
+ return vector.elementAt(count++);
+ }
+ throw new NoSuchElementException("NameImplEnumerator");
+ }
+}