Open Source Repository

Home /beanshell/bsh-2.0b4 | Repository Home



bsh/ExternalNameSpace.java
package  bsh;

import java.util.*;

/**
  A namespace which maintains an external map of values held in variables in
  its scope.  This mechanism provides a standard collections based interface
  to the namespace as well as a convenient way to export and view values of
  the namespace without the ordinary BeanShell wrappers.   
  </p>

  Variables are maintained internally in the normal fashion to support
  meta-information (such as variable type and visibility modifiers), but
  exported and imported in a synchronized way.  Variables are exported each
  time they are written by BeanShell.  Imported variables from the map appear
  in the BeanShell namespace as untyped variables with no modifiers and
  shadow any previously defined variables in the scope. 
  <p/>

  Note: this class is inherentely dependent on Java 1.2 (for Map), however
  it is not used directly by the core as other than type NameSpace, so no
  dependency is introduced.
*/
/*
  Implementation notes:

  It would seem that we should have been accomplished this by overriding the
  getImportedVar() method of NameSpace, which behaves in a similar way
  for fields of classes and objects.  However we need more control here to
  be able to bump up the precedence and remove items that have been removed
  via the map.  So we override getVariableImp().  We should reevaluate this
  at some point.  All of NameSpace is a mess.

  The primary abstraction here is that we override createVariable() to
  create LHS Variables bound to the map for this namespace.

  Methods:

  bsh methods are not currently exported to the
  external namespace.  All that would be required to add this is to override
  setMethod() and provide a friendlier view than vector (currently used) for
  overloaded forms (perhaps a map by method SignatureKey).

*/
public class ExternalNameSpace extends NameSpace
{
  private Map externalMap;

    public ExternalNameSpace() 
  {
    thisnull, "External Map Namespace"null );
  }

  /**
  */
    public ExternalNameSpaceNameSpace parent, String name, Map externalMap 
  {
    superparent, name );

    if externalMap == null )
      externalMap = new HashMap();
      
    this.externalMap = externalMap;

  }

  /**
    Get the map view of this namespace.
  */
  public Map getMap() { return externalMap; }

  /**
    Set the external Map which to which this namespace synchronizes.
    The previous external map is detached from this namespace.  Previous
    map values are retained in the external map, but are removed from the
    BeanShell namespace.
  */
  public void setMapMap map 
  
    // Detach any existing namespace to preserve it, then clear this
    // namespace and set the new one
    this.externalMap = null
    clear();
    this.externalMap = map ; 
  }

  /**
  */
  public void unsetVariableString name )
  {
    super.unsetVariablename );
    externalMap.removename );
  }

  /**
  */
  public String [] getVariableNames() 
  {
    // union of the names in the enclosing namespace and external map
    Set nameSet = new HashSet();
    String [] nsNames = super.getVariableNames();
    nameSet.addAllArrays.asListnsNames ) );
    nameSet.addAllexternalMap.keySet() );
    return (String [])nameSet.toArraynew String[0] );
  }

  /**
  */
  /*
    Notes: This implementation of getVariableImpl handles the following
    cases:
    1) var in map not in local scope - var was added through map
    2) var in map and in local scope - var was added through namespace
    3) var not in map but in local scope - var was removed via map
    4) var not in map and not in local scope - non-existent var

    Note: It would seem that we could simply override getImportedVar()
    in NameSpace, rather than this higher level method.  However we need
    more control here to change the import precedence and remove variables
    if they are removed via the extenal map.
  */
    protected Variable getVariableImplString name, boolean recurse )
    throws UtilEvalError
  {
    // check the external map for the variable name
    Object value =  externalMap.getname );

    Variable var;
    if value == null 
    {
      // The var is not in external map and it should therefore not be
      // found in local scope (it may have been removed via the map).  
      // Clear it prophalactically.
      super.unsetVariablename )

      // Search parent for var if applicable.
      var = super.getVariableImplname, recurse );
    else
    {
      // Var in external map may be found in local scope with type and
      // modifier info.
      Variable localVar = super.getVariableImplname, false );

      // If not in local scope then it was added via the external map,
      // we'll wrap it and pass it along.  Else we'll use the one we
      // found.
      if localVar == null 
        var = createVariablename, null/*type*/, value, null/*mods*/ );
      else
        var = localVar;
    }

    return var;
    }

  public Variable createVariable(
    String name, Class type, Object value, Modifiers mods )
  {
    LHS lhs = new LHSexternalMap, name );
    // Is this race condition worth worrying about?
    // value will appear in map before it's really in the interpreter
    try {
      lhs.assignvalue, false/*strict*/ );
    catch UtilEvalError e) {
      throw new InterpreterErrore.toString() );
    }
    return new Variablename, type, lhs );
  }

  /**
    Clear all variables, methods, and imports from this namespace and clear
    all values from the external map (via Map clear()).
  */
  public void clear() 
  {
    super.clear();
    externalMap.clear();
  }

}