mirror of git://gcc.gnu.org/git/gcc.git
				
				
				
			
		
			
				
	
	
		
			524 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			Java
		
	
	
	
			
		
		
	
	
			524 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			Java
		
	
	
	
/* CompoundName.java --
 | 
						|
   Copyright (C) 2001, 2004, 2005  Free Software Foundation, Inc.
 | 
						|
 | 
						|
This file is part of GNU Classpath.
 | 
						|
 | 
						|
GNU Classpath is free software; you can redistribute it and/or modify
 | 
						|
it under the terms of the GNU General Public License as published by
 | 
						|
the Free Software Foundation; either version 2, or (at your option)
 | 
						|
any later version.
 | 
						|
 | 
						|
GNU Classpath 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 for more details.
 | 
						|
 | 
						|
You should have received a copy of the GNU General Public License
 | 
						|
along with GNU Classpath; see the file COPYING.  If not, write to the
 | 
						|
Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
 | 
						|
02110-1301 USA.
 | 
						|
 | 
						|
Linking this library statically or dynamically with other modules is
 | 
						|
making a combined work based on this library.  Thus, the terms and
 | 
						|
conditions of the GNU General Public License cover the whole
 | 
						|
combination.
 | 
						|
 | 
						|
As a special exception, the copyright holders of this library give you
 | 
						|
permission to link this library with independent modules to produce an
 | 
						|
executable, regardless of the license terms of these independent
 | 
						|
modules, and to copy and distribute the resulting executable under
 | 
						|
terms of your choice, provided that you also meet, for each linked
 | 
						|
independent module, the terms and conditions of the license of that
 | 
						|
module.  An independent module is a module which is not derived from
 | 
						|
or based on this library.  If you modify this library, you may extend
 | 
						|
this exception to your version of the library, but you are not
 | 
						|
obligated to do so.  If you do not wish to do so, delete this
 | 
						|
exception statement from your version. */
 | 
						|
 | 
						|
 | 
						|
package javax.naming;
 | 
						|
 | 
						|
import gnu.java.lang.CPStringBuilder;
 | 
						|
 | 
						|
import java.io.IOException;
 | 
						|
import java.io.ObjectInputStream;
 | 
						|
import java.io.ObjectOutputStream;
 | 
						|
import java.io.Serializable;
 | 
						|
import java.util.Enumeration;
 | 
						|
import java.util.NoSuchElementException;
 | 
						|
import java.util.Properties;
 | 
						|
import java.util.Vector;
 | 
						|
 | 
						|
/**
 | 
						|
 * Represents hierarchical names from the single namespace. For instance,
 | 
						|
 * the path /home/audriusa/classpath/file.txt is the compound name, using
 | 
						|
 * the filesystem namespace.
 | 
						|
 *
 | 
						|
 * @author Tom Tromey (tromey@redhat.com)
 | 
						|
 * @date May 16, 2001
 | 
						|
 *
 | 
						|
 * FIXME: this class is underspecified.  For instance, the `flat'
 | 
						|
 * direction is never described.  If it means that the CompoundName
 | 
						|
 * can only have a single element, then the Enumeration-based
 | 
						|
 * constructor ought to throw InvalidNameException.
 | 
						|
 *
 | 
						|
 * @since 1.3
 | 
						|
 */
 | 
						|
public class CompoundName implements Name, Cloneable, Serializable
 | 
						|
{
 | 
						|
  private static final long serialVersionUID = 3513100557083972036L;
 | 
						|
 | 
						|
  private CompoundName (Properties syntax)
 | 
						|
  {
 | 
						|
    elts = new Vector<String> ();
 | 
						|
    mySyntax = syntax;
 | 
						|
    initializeSyntax ();
 | 
						|
  }
 | 
						|
 | 
						|
  protected CompoundName (Enumeration<String> comps, Properties syntax)
 | 
						|
  {
 | 
						|
    elts = new Vector<String> ();
 | 
						|
    mySyntax = syntax;
 | 
						|
    initializeSyntax ();
 | 
						|
    try
 | 
						|
      {
 | 
						|
        while (comps.hasMoreElements ())
 | 
						|
          elts.add (comps.nextElement ());
 | 
						|
      }
 | 
						|
    catch (NoSuchElementException ignore)
 | 
						|
      {
 | 
						|
      }
 | 
						|
  }
 | 
						|
 | 
						|
  public CompoundName (String n, Properties syntax)
 | 
						|
    throws InvalidNameException
 | 
						|
  {
 | 
						|
    elts = new Vector<String> ();
 | 
						|
    mySyntax = syntax;
 | 
						|
    initializeSyntax ();
 | 
						|
 | 
						|
    StringBuilder new_element = new StringBuilder ();
 | 
						|
    int i = 0;
 | 
						|
    // QUOTE==null means no quoting right now.  When it is set it is
 | 
						|
    // the value of the closing quote.
 | 
						|
    String quote = null;
 | 
						|
    while (i < n.length ())
 | 
						|
      {
 | 
						|
        String special = isSpecial (n, i);
 | 
						|
 | 
						|
        if (special == escape && escape != null)
 | 
						|
          {
 | 
						|
            if (n.length () == i + special.length ())
 | 
						|
              {
 | 
						|
                // A trailing escape is treated as itself.
 | 
						|
                new_element.append (special);
 | 
						|
                i += special.length ();
 | 
						|
              }
 | 
						|
            else
 | 
						|
              {
 | 
						|
                String eSpecial = isSpecial (n, i + special.length ());
 | 
						|
                if (eSpecial != null)
 | 
						|
                  {
 | 
						|
                    // Treat the escape as an escape.
 | 
						|
                    new_element.append (eSpecial);
 | 
						|
                    i += special.length () + eSpecial.length ();
 | 
						|
                  }
 | 
						|
                else
 | 
						|
                  {
 | 
						|
                    // Treat the escape as itself.
 | 
						|
                    new_element.append (special);
 | 
						|
                    i += special.length ();
 | 
						|
                  }
 | 
						|
                continue;
 | 
						|
              }
 | 
						|
          }
 | 
						|
        else if (quote != null)
 | 
						|
          {
 | 
						|
            // It is safe to use == here.
 | 
						|
            if (quote == special)
 | 
						|
              {
 | 
						|
                // Quotes must surround a complete component.
 | 
						|
                if (i + quote.length () < n.length ()
 | 
						|
                    && ! n.startsWith (separator, i + quote.length ()))
 | 
						|
                  throw new InvalidNameException ("close quote before end of component");
 | 
						|
                elts.add (new_element.toString ());
 | 
						|
                new_element.setLength (0);
 | 
						|
                i += quote.length ();
 | 
						|
                quote = null;
 | 
						|
                continue;
 | 
						|
              }
 | 
						|
            // Otherwise, fall through.
 | 
						|
          }
 | 
						|
        // Quotes are only special at the start of a component.
 | 
						|
        else if (new_element.length () == 0
 | 
						|
                 && special == beginQuote
 | 
						|
                 && beginQuote != null)
 | 
						|
          {
 | 
						|
            quote = endQuote;
 | 
						|
            i += special.length ();
 | 
						|
            continue;
 | 
						|
          }
 | 
						|
        else if (new_element.length () == 0
 | 
						|
                 && special == beginQuote2
 | 
						|
                 && beginQuote2 != null)
 | 
						|
          {
 | 
						|
            quote = endQuote2;
 | 
						|
            i += special.length ();
 | 
						|
            continue;
 | 
						|
          }
 | 
						|
        else if (direction != FLAT && special == separator)
 | 
						|
          {
 | 
						|
            elts.add (new_element.toString ());
 | 
						|
            new_element.setLength (0);
 | 
						|
            i += special.length ();
 | 
						|
            continue;
 | 
						|
          }
 | 
						|
 | 
						|
        // Nothing in particular, so try the next character.
 | 
						|
        new_element.append (n.charAt (i));
 | 
						|
        ++i;
 | 
						|
      }
 | 
						|
 | 
						|
    if (new_element.length () != 0)
 | 
						|
      elts.add (new_element.toString ());
 | 
						|
 | 
						|
    if (direction == RIGHT_TO_LEFT)
 | 
						|
      {
 | 
						|
        // Reverse the order of the elements.
 | 
						|
        int len = elts.size ();
 | 
						|
        for (i = 0; i < len / 2; ++i)
 | 
						|
          {
 | 
						|
            String t = elts.set (i, elts.get (len - i - 1));
 | 
						|
            elts.set (len - i - 1, t);
 | 
						|
          }
 | 
						|
      }
 | 
						|
 | 
						|
    // Error checking.
 | 
						|
    if (quote != null)
 | 
						|
      throw new InvalidNameException ("unterminated quote");
 | 
						|
  }
 | 
						|
 | 
						|
  public Name add (int posn, String comp) throws InvalidNameException
 | 
						|
  {
 | 
						|
    elts.add (posn, comp);
 | 
						|
    return this;
 | 
						|
  }
 | 
						|
 | 
						|
  public Name add (String comp) throws InvalidNameException
 | 
						|
  {
 | 
						|
    elts.add (comp);
 | 
						|
    return this;
 | 
						|
  }
 | 
						|
 | 
						|
  public Name addAll (int posn, Name n) throws InvalidNameException
 | 
						|
  {
 | 
						|
    Enumeration<String> e = n.getAll ();
 | 
						|
    try
 | 
						|
      {
 | 
						|
        while (e.hasMoreElements ())
 | 
						|
          {
 | 
						|
            elts.add (posn, e.nextElement ());
 | 
						|
            ++posn;
 | 
						|
          }
 | 
						|
      }
 | 
						|
    catch (NoSuchElementException ignore)
 | 
						|
      {
 | 
						|
      }
 | 
						|
    return this;
 | 
						|
  }
 | 
						|
 | 
						|
  public Name addAll (Name suffix) throws InvalidNameException
 | 
						|
  {
 | 
						|
    Enumeration<String> e = suffix.getAll ();
 | 
						|
    try
 | 
						|
      {
 | 
						|
        while (e.hasMoreElements ())
 | 
						|
          elts.add (e.nextElement ());
 | 
						|
      }
 | 
						|
    catch (NoSuchElementException ignore)
 | 
						|
      {
 | 
						|
      }
 | 
						|
    return this;
 | 
						|
  }
 | 
						|
 | 
						|
  public Object clone ()
 | 
						|
  {
 | 
						|
    return new CompoundName (elts.elements (), mySyntax);
 | 
						|
  }
 | 
						|
 | 
						|
  public int compareTo (Object obj)
 | 
						|
  {
 | 
						|
    if (! (obj instanceof CompoundName))
 | 
						|
      throw new ClassCastException ("CompoundName.compareTo() expected CompoundName");
 | 
						|
    CompoundName cn = (CompoundName) obj;
 | 
						|
    int last = Math.min (cn.elts.size (), elts.size ());
 | 
						|
    for (int i = 0; i < last; ++i)
 | 
						|
      {
 | 
						|
        String f = canonicalize (elts.get (i));
 | 
						|
        int comp = f.compareTo (canonicalize (cn.elts.get (i)));
 | 
						|
        if (comp != 0)
 | 
						|
          return comp;
 | 
						|
      }
 | 
						|
    return elts.size () - cn.elts.size ();
 | 
						|
  }
 | 
						|
 | 
						|
  public boolean endsWith (Name n)
 | 
						|
  {
 | 
						|
    if (! (n instanceof CompoundName))
 | 
						|
      return false;
 | 
						|
    CompoundName cn = (CompoundName) n;
 | 
						|
    if (cn.elts.size () > elts.size ())
 | 
						|
      return false;
 | 
						|
    int delta = elts.size () - cn.elts.size ();
 | 
						|
    for (int i = 0; i < cn.elts.size (); ++i)
 | 
						|
      {
 | 
						|
        String f = canonicalize (elts.get (delta + i));
 | 
						|
        if (! f.equals (canonicalize (cn.elts.get (i))))
 | 
						|
          return false;
 | 
						|
      }
 | 
						|
    return true;
 | 
						|
  }
 | 
						|
 | 
						|
  public boolean equals (Object obj)
 | 
						|
  {
 | 
						|
    if (! (obj instanceof CompoundName))
 | 
						|
      return false;
 | 
						|
    return compareTo (obj) == 0;
 | 
						|
  }
 | 
						|
 | 
						|
  public String get (int posn)
 | 
						|
  {
 | 
						|
    return elts.get (posn);
 | 
						|
  }
 | 
						|
 | 
						|
  public Enumeration<String> getAll ()
 | 
						|
  {
 | 
						|
    return elts.elements ();
 | 
						|
  }
 | 
						|
 | 
						|
  public Name getPrefix (int posn)
 | 
						|
  {
 | 
						|
    CompoundName cn = new CompoundName (mySyntax);
 | 
						|
    for (int i = 0; i < posn; ++i)
 | 
						|
      cn.elts.add (elts.get (i));
 | 
						|
    return cn;
 | 
						|
  }
 | 
						|
 | 
						|
  public Name getSuffix (int posn)
 | 
						|
  {
 | 
						|
    if (posn > elts.size ())
 | 
						|
      throw new ArrayIndexOutOfBoundsException (posn);
 | 
						|
    CompoundName cn = new CompoundName (mySyntax);
 | 
						|
    for (int i = posn; i < elts.size (); ++i)
 | 
						|
      cn.elts.add (elts.get (i));
 | 
						|
    return cn;
 | 
						|
  }
 | 
						|
 | 
						|
  public int hashCode ()
 | 
						|
  {
 | 
						|
    int h = 0;
 | 
						|
    for (int i = 0; i < elts.size (); ++i)
 | 
						|
      h += canonicalize (elts.get (i)).hashCode ();
 | 
						|
    return h;
 | 
						|
  }
 | 
						|
 | 
						|
  public boolean isEmpty ()
 | 
						|
  {
 | 
						|
    return elts.isEmpty ();
 | 
						|
  }
 | 
						|
 | 
						|
  public Object remove (int posn) throws InvalidNameException
 | 
						|
  {
 | 
						|
    return elts.remove (posn);
 | 
						|
  }
 | 
						|
 | 
						|
  public int size ()
 | 
						|
  {
 | 
						|
    return elts.size ();
 | 
						|
  }
 | 
						|
 | 
						|
  public boolean startsWith (Name n)
 | 
						|
  {
 | 
						|
    if (! (n instanceof CompoundName))
 | 
						|
      return false;
 | 
						|
    CompoundName cn = (CompoundName) n;
 | 
						|
    if (cn.elts.size () > elts.size ())
 | 
						|
      return false;
 | 
						|
    for (int i = 0; i < cn.elts.size (); ++i)
 | 
						|
      {
 | 
						|
        String f = canonicalize (elts.get (i));
 | 
						|
        if (! f.equals (canonicalize (cn.elts.get (i))))
 | 
						|
          return false;
 | 
						|
      }
 | 
						|
    return true;
 | 
						|
  }
 | 
						|
 | 
						|
  // If ELEMENT starts with some meta-sequence at OFFSET, then return
 | 
						|
  // the string representing the meta-sequence.  Otherwise return
 | 
						|
  // null.
 | 
						|
  private String isSpecial (String element, int offset)
 | 
						|
  {
 | 
						|
    String special = null;
 | 
						|
    if (separator != null && element.startsWith (separator, offset))
 | 
						|
      special = separator;
 | 
						|
    else if (escape != null && element.startsWith (escape, offset))
 | 
						|
      special = escape;
 | 
						|
    else if (beginQuote != null && element.startsWith (beginQuote, offset))
 | 
						|
      special = beginQuote;
 | 
						|
    else if (endQuote != null && element.startsWith (endQuote, offset))
 | 
						|
      special = endQuote;
 | 
						|
    else if (beginQuote2 != null
 | 
						|
             && element.startsWith (beginQuote2, offset))
 | 
						|
      special = beginQuote2;
 | 
						|
    else if (endQuote2 != null && element.startsWith (endQuote2, offset))
 | 
						|
      special = endQuote2;
 | 
						|
 | 
						|
    return special;
 | 
						|
  }
 | 
						|
 | 
						|
  public String toString ()
 | 
						|
  {
 | 
						|
    CPStringBuilder result = new CPStringBuilder ();
 | 
						|
    int size = elts.size ();
 | 
						|
    for (int i = 0; i < size; ++i)
 | 
						|
      {
 | 
						|
        // Find the appropriate element.  FIXME: not clear what FLAT
 | 
						|
        // means.
 | 
						|
        int offset = (direction == RIGHT_TO_LEFT) ? (size - i - 1) : i;
 | 
						|
        String element = elts.get (offset);
 | 
						|
        if (i > 0
 | 
						|
            || (i == size - 1 && element.equals ("")))
 | 
						|
          result.append (separator);
 | 
						|
 | 
						|
        int k = 0;
 | 
						|
        while (k < element.length ())
 | 
						|
          {
 | 
						|
            String special = isSpecial (element, k);
 | 
						|
            if (special != null)
 | 
						|
              {
 | 
						|
                result.append (escape);
 | 
						|
                result.append (special);
 | 
						|
                k += special.length ();
 | 
						|
              }
 | 
						|
            else
 | 
						|
              {
 | 
						|
                result.append (element.charAt (k));
 | 
						|
                ++k;
 | 
						|
              }
 | 
						|
          }
 | 
						|
      }
 | 
						|
 | 
						|
    return result.toString ();
 | 
						|
  }
 | 
						|
 | 
						|
  // This canonicalizes a String, based on the syntax, for comparison
 | 
						|
  // or other similar purposes.
 | 
						|
  private String canonicalize (String element)
 | 
						|
  {
 | 
						|
    String ret = element;
 | 
						|
 | 
						|
    if (ignoreCase)
 | 
						|
      ret = ret.toLowerCase ();
 | 
						|
 | 
						|
    if (trimBlanks)
 | 
						|
      {
 | 
						|
        int first = 0;
 | 
						|
        while (first < ret.length ()
 | 
						|
               && Character.isWhitespace (ret.charAt (first)))
 | 
						|
          ++first;
 | 
						|
 | 
						|
        int last = ret.length () - 1;
 | 
						|
        while (last >= first
 | 
						|
               && Character.isWhitespace (ret.charAt (last)))
 | 
						|
          --last;
 | 
						|
 | 
						|
        ret = ret.substring (first, last);
 | 
						|
      }
 | 
						|
 | 
						|
    return ret;
 | 
						|
  }
 | 
						|
 | 
						|
  // This initializes all the syntax variables.  This seems easier
 | 
						|
  // than re-querying the properties every time.  We're allowed to do
 | 
						|
  // this because the spec says that subclasses should consider the
 | 
						|
  // syntax as being read-only.
 | 
						|
  private void initializeSyntax ()
 | 
						|
  {
 | 
						|
    String t = mySyntax.getProperty ("jndi.syntax.direction", "flat");
 | 
						|
    if (t.equals ("right_to_left"))
 | 
						|
      this.direction = RIGHT_TO_LEFT;
 | 
						|
    else if (t.equals ("left_to_right"))
 | 
						|
      this.direction = LEFT_TO_RIGHT;
 | 
						|
    else
 | 
						|
      {
 | 
						|
        // If we don't recognize it, default to flat.
 | 
						|
        this.direction = FLAT;
 | 
						|
      }
 | 
						|
 | 
						|
    // This is required unless the direction is FLAT.  Unfortunately
 | 
						|
    // there is no way to report this error.
 | 
						|
    this.separator = mySyntax.getProperty ("jndi.syntax.separator", "");
 | 
						|
 | 
						|
    this.ignoreCase
 | 
						|
      = Boolean.valueOf (mySyntax.getProperty ("jndi.syntax.ignorecase",
 | 
						|
                                               "false")).booleanValue ();
 | 
						|
    this.escape = mySyntax.getProperty ("jndi.syntax.escape", null);
 | 
						|
    this.beginQuote = mySyntax.getProperty ("jndi.syntax.beginquote", null);
 | 
						|
    this.endQuote = mySyntax.getProperty ("jndi.syntax.endquote",
 | 
						|
                                          this.beginQuote);
 | 
						|
    this.beginQuote2 = mySyntax.getProperty ("jndi.syntax.beginquote2",
 | 
						|
                                             null);
 | 
						|
    this.endQuote2 = mySyntax.getProperty ("jndi.syntax.endquote2",
 | 
						|
                                           this.beginQuote2);
 | 
						|
    this.trimBlanks
 | 
						|
      = Boolean.valueOf (mySyntax.getProperty ("jndi.syntax.trimblanks",
 | 
						|
                                               "false")).booleanValue ();
 | 
						|
  }
 | 
						|
 | 
						|
  private void readObject(ObjectInputStream s)
 | 
						|
    throws IOException, ClassNotFoundException
 | 
						|
  {
 | 
						|
    mySyntax = (Properties) s.readObject();
 | 
						|
    int count = s.readInt();
 | 
						|
    elts = new Vector<String>(count);
 | 
						|
    for (int i = 0; i < count; i++)
 | 
						|
      elts.addElement((String) s.readObject());
 | 
						|
  }
 | 
						|
 | 
						|
  private void writeObject(ObjectOutputStream s)
 | 
						|
    throws IOException
 | 
						|
  {
 | 
						|
    s.writeObject(mySyntax);
 | 
						|
    s.writeInt(elts.size());
 | 
						|
    for (int i = 0; i < elts.size(); i++)
 | 
						|
        s.writeObject(elts.elementAt(i));
 | 
						|
  }
 | 
						|
 | 
						|
  // The spec specifies this but does not document it in any way (it
 | 
						|
  // is a package-private class).  It is useless as far as I can tell.
 | 
						|
  // So we ignore it.
 | 
						|
  // protected transient NameImpl impl;
 | 
						|
  protected transient Properties mySyntax;
 | 
						|
 | 
						|
  // The actual elements.
 | 
						|
  private transient Vector<String> elts;
 | 
						|
 | 
						|
  // The following are all used for syntax.
 | 
						|
  private transient int direction;
 | 
						|
  private transient String separator;
 | 
						|
  private transient boolean ignoreCase;
 | 
						|
  private transient String escape;
 | 
						|
  private transient String beginQuote;
 | 
						|
  private transient String endQuote;
 | 
						|
  private transient String beginQuote2;
 | 
						|
  private transient String endQuote2;
 | 
						|
  private transient boolean trimBlanks;
 | 
						|
  // We didn't need these for parsing, so they are gone.
 | 
						|
  // private transient String avaSeparator;
 | 
						|
  // private transient String typevalSeparator;
 | 
						|
 | 
						|
  private static final int RIGHT_TO_LEFT = -1;
 | 
						|
  private static final int LEFT_TO_RIGHT = 1;
 | 
						|
  private static final int FLAT = 0;
 | 
						|
}
 |