Package helma.objectmodel.db

Source Code of helma.objectmodel.db.Node

/*
* Helma License Notice
*
* The contents of this file are subject to the Helma License
* Version 2.0 (the "License"). You may not use this file except in
* compliance with the License. A copy of the License is available at
* http://adele.helma.org/download/helma/license.txt
*
* Copyright 1998-2003 Helma Software. All Rights Reserved.
*
* $RCSfile$
* $Author: hannes $
* $Revision: 9994 $
* $Date: 2009-11-25 10:33:28 +0100 (Mit, 25. Nov 2009) $
*/

package helma.objectmodel.db;

import helma.framework.IPathElement;
import helma.framework.core.RequestEvaluator;
import helma.framework.core.Application;
import helma.objectmodel.ConcurrencyException;
import helma.objectmodel.INode;
import helma.objectmodel.IProperty;
import helma.objectmodel.TransientNode;
import helma.util.EmptyEnumeration;

import java.util.*;

/**
* An implementation of INode that can be stored in the internal database or
* an external relational database.
*/
public final class Node implements INode {

    // The handle to the node's parent
    protected NodeHandle parentHandle;

    // Ordered list of subnodes of this node
    private SubnodeList subnodes;

    // Named subnodes (properties) of this node
    private Hashtable propMap;

    protected long created;
    protected long lastmodified;
    private String id;
    private String name;

    // is this node's main identity as a named property or an
    // anonymous node in a subnode collection?
    protected boolean anonymous = false;

    // the serialization version this object was read from (see readObject())
    protected short version = 0;
    private String prototype;
    private NodeHandle handle;
    private INode cacheNode;
    final WrappedNodeManager nmgr;
    DbMapping dbmap;
    Key primaryKey = null;
    String subnodeRelation = null;
    long lastNameCheck = 0;
    long lastParentSet = 0;
    private volatile Transactor lock;
    private volatile int state;
    private static long idgen = 0;

    /**
     * Creates an empty, uninitialized Node with the given create and modify time.
     * This is used for null-node references in the node cache.
     * @param timestamp
     */
    protected Node(long timestamp) {
        created = lastmodified = timestamp;
        this.nmgr = null;
    }

    /**
     * Creates an empty, uninitialized Node. The init() method must be called on the
     * Node before it can do anything useful.
     */
    protected Node(WrappedNodeManager nmgr) {
        if (nmgr == null) {
            throw new NullPointerException("nmgr");
        }
        this.nmgr = nmgr;
        created = lastmodified = System.currentTimeMillis();
    }

    /**
     * Creates a new Node with the given name. Used by NodeManager for creating "root nodes"
     * outside of a Transaction context, which is why we can immediately mark it as CLEAN.
     * Also used by embedded database to re-create an existing Node.
     */
    public Node(String name, String id, String prototype, WrappedNodeManager nmgr) {
        if (nmgr == null) {
            throw new NullPointerException("nmgr");
        }
        this.nmgr = nmgr;
        if (prototype == null) {
            prototype = "HopObject";
        }
        init(nmgr.getDbMapping(prototype), id, name, prototype, null);
    }

    /**
     * Constructor used to create a Node with a given name from a embedded database.
     */
    public Node(String name, String id, String prototype, WrappedNodeManager nmgr,
                long created, long lastmodified) {
        this(name, id, prototype, nmgr);
        this.created = created;
        this.lastmodified = lastmodified;
    }

    /**
     * Constructor used for virtual nodes.
     */
    public Node(Node home, String propname, WrappedNodeManager nmgr, String prototype) {
        if (nmgr == null) {
            throw new NullPointerException("nmgr");
        }
        this.nmgr = nmgr;
        setParent(home);
        // generate a key for the virtual node that can't be mistaken for a Database Key
        primaryKey = new SyntheticKey(home.getKey(), propname);
        this.id = primaryKey.getID();
        this.name = propname;
        this.prototype = prototype;
        this.anonymous = false;

        // set the collection's state according to the home node's state
        if (home.state == NEW || home.state == TRANSIENT) {
            this.state = TRANSIENT;
        } else {
            this.state = VIRTUAL;
        }
    }

    /**
     * Creates a new Node with the given name. This is used for ordinary transient nodes.
     */
    public Node(String name, String prototype, WrappedNodeManager nmgr) {
        if (nmgr == null) {
            throw new NullPointerException("nmgr");
        }
        this.nmgr = nmgr;
        this.prototype = prototype;
        dbmap = nmgr.getDbMapping(prototype);

        // the id is only generated when the node is actually checked into db,
        // or when it's explicitly requested.
        id = null;
        this.name = (name == null) ? "" : name;
        created = lastmodified = System.currentTimeMillis();
        state = TRANSIENT;

        if (prototype != null && dbmap != null) {
            String protoProperty = dbmap.columnNameToProperty(dbmap.getPrototypeField());
            if (protoProperty != null) {
                setString(protoProperty, dbmap.getExtensionId());
            }
        }
    }

    /**
     * Initializer used for nodes being instanced from an embedded or relational database.
     */
    public synchronized void init(DbMapping dbm, String id, String name,
                                  String prototype, Hashtable propMap) {
        this.dbmap = dbm;
        this.prototype = prototype;
        this.id = id;
        this.name = name;
        // If name was not set from resultset, create a synthetical name now.
        if ((name == null) || (name.length() == 0)) {
            this.name = prototype + " " + id;
        }

        this.propMap = propMap;

        // set lastmodified and created timestamps and mark as clean
        created = lastmodified = System.currentTimeMillis();

        if (state != CLEAN) {
            markAs(CLEAN);
        }
    }

    /**
     * used by Xml deserialization
     */
    public synchronized void setPropMap(Hashtable propMap) {
        this.propMap = propMap;
    }

    /**
     * Get the write lock on this node, throwing a ConcurrencyException if the
     * lock is already held by another thread.
     */
    synchronized void checkWriteLock() {
        if (state == TRANSIENT) {
            return; // no need to lock transient node
        }

        Transactor tx = Transactor.getInstanceOrFail();

        if (!tx.isActive()) {
            throw new helma.framework.TimeoutException();
        }

        if (state == INVALID) {
            nmgr.logEvent("Got Invalid Node: " + this);
            Thread.dumpStack();
            throw new ConcurrencyException("Node " + this +
                                           " was invalidated by another thread.");
        }

        if ((lock != null) && (lock != tx) && lock.isAlive() && lock.isActive()) {
            // nmgr.logEvent("Concurrency conflict for " + this + ", lock held by " + lock);
            throw new ConcurrencyException("Tried to modify " + this +
                                           " from two threads at the same time.");
        }

        tx.visitDirtyNode(this);
        lock = tx;
    }

    /**
     * Clear the write lock on this node.
     */
    synchronized void clearWriteLock() {
        lock = null;
    }

    /**
     *  Set this node's state, registering it with the transactor if necessary.
     */
    void markAs(int s) {
        if (s == state || state == INVALID || state == VIRTUAL || state == TRANSIENT) {
            return;
        }

        state = s;

        Transactor tx = Transactor.getInstance();
        if (tx != null) {
            if (s == CLEAN) {
                clearWriteLock();
                tx.dropDirtyNode(this);
            } else {
                tx.visitDirtyNode(this);

                if (s == NEW) {
                    clearWriteLock();
                    tx.visitCleanNode(this);
                }
            }
        }
    }

    /**
     * Register this node as parent node with the transactor so that
     * setLastSubnodeChange is called when the transaction completes.
     */
    void registerSubnodeChange() {
        // we do not fetch subnodes for nodes that haven't been persisted yet or are in
        // the process of being persistified - except if "manual" subnoderelation is set.
        if ((state == TRANSIENT || state == NEW) && subnodeRelation == null) {
            return;
        } else {
            Transactor tx = Transactor.getInstance();
            if (tx != null) {
                tx.visitParentNode(this);
            }
        }
    }

    /**
     * Notify the node's parent that its child collection needs to be reloaded
     * in case the changed property has an affect on collection order or content.
     *
     * @param propname the name of the property being changed
     */
    void notifyPropertyChange(String propname) {
        Node parent = (parentHandle == null) ? null : (Node) getParent();

        if ((parent != null) && (parent.getDbMapping() != null)) {
            // check if this node is already registered with the old name; if so, remove it.
            // then set parent's property to this node for the new name value
            DbMapping parentmap = parent.getDbMapping();
            Relation subrel = parentmap.getSubnodeRelation();
            String dbcolumn = dbmap.propertyToColumnName(propname);
            if (subrel == null || dbcolumn == null)
                return;

            if (subrel.order != null && subrel.order.indexOf(dbcolumn) > -1) {
                parent.registerSubnodeChange();
            }
        }
    }

    /**
     * Called by the transactor on registered parent nodes to mark the
     * child index as changed
     */
    public void markSubnodesChanged() {
        if (subnodes != null) {
            subnodes.markAsChanged();
        }
    }

    /**
     *  Gets this node's stateas defined in the INode interface
     *
     * @return this node's state
     */
    public int getState() {
        return state;
    }

    /**
     * Sets this node's state as defined in the INode interface
     *
     * @param s this node's new state
     */
    public void setState(int s) {
        state = s;
    }

    /**
     *  Mark node as invalid so it is re-fetched from the database
     */
    public void invalidate() {
        // This doesn't make sense for transient nodes
        if ((state == TRANSIENT) || (state == NEW)) {
            return;
        }

        checkWriteLock();
        nmgr.evictNode(this);
    }

    /**
     *  Check for a child mapping and evict the object specified by key from the cache
     */
    public void invalidateNode(String key) {
        // This doesn't make sense for transient nodes
        if ((state == TRANSIENT) || (state == NEW)) {
            return;
        }

        Relation rel = getDbMapping().getSubnodeRelation();

        if (rel != null) {
            if (rel.usesPrimaryKey()) {
                nmgr.evictNodeByKey(new DbKey(getDbMapping().getSubnodeMapping(), key));
            } else {
                nmgr.evictNodeByKey(new SyntheticKey(getKey(), key));
            }
        }
    }

    /**
     *  Get the ID of this Node. This is the primary database key and used as part of the
     *  key for the internal node cache.
     */
    public String getID() {
        // if we are transient, we generate an id on demand. It's possible that we'll never need
        // it, but if we do it's important to keep the one we have.
        if (state == TRANSIENT && id == null) {
            id = generateTransientID();
        }
        return id;
    }

    /**
     * Returns true if this node is accessed by id from its aprent, false if it
     * is accessed by name
     */
    public boolean isAnonymous() {
        return anonymous;
    }

    /**
     * Return this node' name, which may or may not have some meaning
     */
    public String getName() {
        return name;
    }

    /**
     * Get something to identify this node within a URL. This is the ID for anonymous nodes
     * and a property value for named properties.
     */
    public String getElementName() {
        // check element name - this is either the Node's id or name.
        long lastmod = lastmodified;

        if (dbmap != null) {
            lastmod = Math.max(lastmod, dbmap.getLastTypeChange());
        }

        if ((parentHandle != null) && (lastNameCheck <= lastmod)) {
            try {
                Node p = parentHandle.getNode(nmgr);
                DbMapping parentmap = p.getDbMapping();
                Relation prel = parentmap.getSubnodeRelation();

                if (prel != null) {
                    if (prel.groupby != null) {
                        setName(getString("groupname"));
                        anonymous = false;
                    } else if (prel.accessName != null) {
                        String propname = dbmap.columnNameToProperty(prel.accessName);
                        String propvalue = getString(propname);

                        if ((propvalue != null) && (propvalue.length() > 0)) {
                            setName(propvalue);
                            anonymous = false;
                        } else if (!anonymous && p.isParentOf(this)) {
                            anonymous = true;
                        }
                    } else if (!anonymous && p.isParentOf(this)) {
                        anonymous = true;
                    }
                } else if (!anonymous && p.isParentOf(this)) {
                    anonymous = true;
                }
            } catch (Exception ignore) {
                // FIXME: add proper NullPointer checks in try statement
                // just fall back to default method
            }

            lastNameCheck = System.currentTimeMillis();
        }

        return (anonymous || (name == null) || (name.length() == 0)) ? id : name;
    }

    /**
     * Get the node's path
     */
    public String getPath() {
        String divider = null;
        StringBuffer b = new StringBuffer();
        INode p = this;
        int loopWatch = 0;

        while (p != null && p.getParent() != null) {
            if (divider != null) {
                b.insert(0, divider);
            } else {
                divider = "/";
            }

            b.insert(0, p.getElementName());
            p = p.getParent();

            loopWatch++;

            if (loopWatch > 10) {
                b.insert(0, "...");

                break;
            }
        }
        return b.toString();
    }

    /**
     * Return the node's prototype name.
     */
    public String getPrototype() {
        // if prototype is null, it's a vanilla HopObject.
        if (prototype == null) {
            return "HopObject";
        }

        return prototype;
    }

    /**
     * Set the node's prototype name.
     */
    public void setPrototype(String proto) {
        this.prototype = proto;
        // Note: we mustn't set the DbMapping according to the prototype,
        // because some nodes have custom dbmappings, e.g. the groupby
        // dbmappings created in DbMapping.initGroupbyMapping().
    }

    /**
     * Set the node's {@link DbMapping}.
     */
    public void setDbMapping(DbMapping dbmap) {
        this.dbmap = dbmap;
    }

    /**
     * Get the node's {@link DbMapping}.
     */
    public DbMapping getDbMapping() {
        return dbmap;
    }

    /**
     * Get the node's key.
     */
    public Key getKey() {
        if (primaryKey == null && state == TRANSIENT) {
            throw new RuntimeException("getKey called on transient Node: " + this);
        }

        if ((dbmap == null) && (prototype != null) && (nmgr != null)) {
            dbmap = nmgr.getDbMapping(prototype);
        }

        if (primaryKey == null) {
            primaryKey = new DbKey(dbmap, id);
        }

        return primaryKey;
    }

    /**
     * Get the node's handle.
     */
    public NodeHandle getHandle() {
        if (handle == null) {
            handle = new NodeHandle(this);
        }
        return handle;
    }

    /**
     * Set an explicit select clause for the node's subnodes
     */
    public synchronized void setSubnodeRelation(String rel) {
        if ((rel == null && this.subnodeRelation == null) ||
                (rel != null && rel.equalsIgnoreCase(this.subnodeRelation))) {
            return;
        }

        checkWriteLock();
        this.subnodeRelation = rel;

        DbMapping smap = (dbmap == null) ? null : dbmap.getSubnodeMapping();

        if (subnodes != null && smap != null && smap.isRelational()) {
            subnodes = null;
        }
    }

    /**
     * Get the node's explicit subnode select clause if one was set, or null
     */
    public synchronized String getSubnodeRelation() {
        return subnodeRelation;
    }

    /**
     * Set the node's name.
     */
    public void setName(String name) {
        if ((name == null) || (name.length() == 0)) {
            // use id as name
            this.name = id;
        } else if (name.indexOf('/') > -1) {
            // "/" is used as delimiter, so it's not a legal char
            return;
        } else {
            this.name = name;
        }
    }

    /**
     * Set this node's parent node.
     */
    public void setParent(Node parent) {
        parentHandle = (parent == null) ? null : parent.getHandle();
    }

    /**
     *  Set this node's parent node to the node referred to by the NodeHandle.
     */
    public void setParentHandle(NodeHandle parent) {
        parentHandle = parent;
    }

    /**
     * Get parent, retrieving it if necessary.
     */
    public INode getParent() {
        // check what's specified in the type.properties for this node.
        ParentInfo[] parentInfo = null;

        if (isRelational() && lastParentSet <= Math.max(dbmap.getLastTypeChange(), lastmodified)) {
            parentInfo = dbmap.getParentInfo();
        }

        // check if current parent candidate matches presciption,
        // if not, try to get one that does.
        if (nmgr.isRootNode(this)) {
            parentHandle = null;
            lastParentSet =  System.currentTimeMillis();
            return null;
        } else if (parentInfo != null) {

            Node parentFallback = null;

            for (int i = 0; i < parentInfo.length; i++) {

                ParentInfo pinfo = parentInfo[i];
                Node parentNode = null;

                // see if there is an explicit relation defined for this parent info
                // we only try to fetch a node if an explicit relation is specified for the prop name
                Relation rel = dbmap.propertyToRelation(pinfo.propname);
                if ((rel != null) && (rel.isReference() || rel.isComplexReference())) {
                    parentNode = (Node) getNode(pinfo.propname);
                }

                // the parent of this node is the app's root node...
                if ((parentNode == null) && pinfo.isroot) {
                    parentNode = nmgr.getRootNode();
                }

                // if we found a parent node, check if we ought to use a virtual or groupby node as parent
                if (parentNode != null) {
                    // see if dbmapping specifies anonymity for this node
                    if (pinfo.virtualname != null) {
                        Node pn2 = (Node) parentNode.getNode(pinfo.virtualname);
                        if (pn2 == null) {
                            getApp().logError("Error: Can't retrieve parent node " +
                                                   pinfo + " for " + this);
                        } else if (pinfo.collectionname != null) {
                            pn2 = (Node) pn2.getNode(pinfo.collectionname);
                        } else if (pn2.equals(this)) {
                            // a special case we want to support: virtualname is actually
                            // a reference to this node, not a collection containing this node.
                            parentHandle = parentNode.getHandle();
                            name = pinfo.virtualname;
                            anonymous = false;
                            return parentNode;
                        }

                        parentNode = pn2;
                    }

                    DbMapping dbm = (parentNode == null) ? null : parentNode.getDbMapping();

                    try {
                        if ((dbm != null) && (dbm.getSubnodeGroupby() != null)) {
                            // check for groupby
                            rel = dbmap.columnNameToRelation(dbm.getSubnodeGroupby());
                            parentNode = (Node) parentNode.getChildElement(getString(rel.propName));
                        }

                        // check if parent actually contains this node. If it does,
                        // accept it immediately, otherwise, keep it as fallback in case
                        // no other parent matches. See http://helma.org/bugs/show_bug.cgi?id=593
                        if (parentNode != null) {
                            if (parentNode.isParentOf(this)) {
                                parentHandle = parentNode.getHandle();
                                lastParentSet = System.currentTimeMillis();
                                return parentNode;
                            } else if (parentFallback == null) {
                                parentFallback = parentNode;
                            }
                        }
                    } catch (Exception x) {
                        getApp().logError("Error retrieving parent node " +
                                                   pinfo + " for " + this, x);
                    }
                }
            }
            lastParentSet = System.currentTimeMillis();
            // if we came till here and we didn't find a parent.
            // set parent to null unless we have a fallback.
            if (parentFallback != null) {
                parentHandle = parentFallback.getHandle();
                return parentFallback;
            } else {
                parentHandle = null;
                if (state != TRANSIENT) {
                    getApp().logEvent("*** Couldn't resolve parent for " + this +
                            " - please check _parent info in type.properties!");
                }
                return null;
            }
        }

        return parentHandle == null ? null : parentHandle.getNode(nmgr);
    }

    /**
     * Get parent, using cached info if it exists.
     */
    public Node getCachedParent() {
        if (parentHandle == null) {
            return null;
        }

        return parentHandle.getNode(nmgr);
    }

    /**
     *  INode-related
     */
    public INode addNode(INode elem) {
        return addNode(elem, -1);
    }

    /**
     * Add a node to this Node's subnodes, making the added node persistent if it
     * hasn't been before and this Node is already persistent.
     *
     * @param elem the node to add to this Nodes subnode-list
     * @param where the index-position where this node has to be added
     *
     * @return the added node itselve
     */
    public INode addNode(INode elem, int where) {
        Node node = null;

        if (elem instanceof Node) {
            node = (Node) elem;
        } else {
            throw new RuntimeException("Can't add fixed-transient node to a persistent node");
        }

        // only lock nodes if parent node is not transient
        if (state != TRANSIENT) {
            // only lock parent if it has to be modified for a change in subnodes
            if (!ignoreSubnodeChange()) {
                checkWriteLock();
            }
            node.checkWriteLock();
        }

        Relation subrel = dbmap == null ? null : dbmap.getSubnodeRelation();
        // if subnodes are defined via relation, make sure its constraints are enforced.
        if (subrel != null && (subrel.countConstraints() < 2 || state != TRANSIENT)) {
            subrel.setConstraints(this, node);
        }

        // if the new node is marked as TRANSIENT and this node is not, mark new node as NEW
        if (state != TRANSIENT && node.state == TRANSIENT) {
            node.makePersistable();
        }

        // only mark this node as modified if subnodes are not in relational db
        // pointing to this node.
        if (!ignoreSubnodeChange() && (state == CLEAN || state == DELETED)) {
            markAs(MODIFIED);
        }

        // TODO this is a rather minimal fix for bug http://helma.org/bugs/show_bug.cgi?id=554
        // - eventually we want to get rid of this code as a whole.
        if (state != TRANSIENT && (node.state == CLEAN || node.state == DELETED)) {
            node.markAs(MODIFIED);
        }

        loadNodes();

        if (subrel != null && subrel.groupby != null) {
            // check if this node has a group-by subnode-relation
            Node groupbyNode = getGroupbySubnode(node, true);
            if (groupbyNode != null) {
                groupbyNode.addNode(node);
                return node;
            }
        }

        NodeHandle nhandle = node.getHandle();

        if (subnodes != null && subnodes.contains(nhandle)) {
            // Node is already subnode of this - just move to new position
            synchronized (subnodes) {
                subnodes.remove(nhandle);
                // check if index is out of bounds when adding
                if (where < 0 || where > subnodes.size()) {
                    subnodes.add(nhandle);
                } else {
                    subnodes.add(where, nhandle);
                }
            }
        } else {
            // create subnode list if necessary
            if (subnodes == null) {
                subnodes = createSubnodeList();
            }

            // check if subnode accessname is set. If so, check if another node
            // uses the same access name, throwing an exception if so.
            if (dbmap != null && node.dbmap != null) {
                Relation prel = dbmap.getSubnodeRelation();

                if (prel != null && prel.accessName != null) {
                    Relation localrel = node.dbmap.columnNameToRelation(prel.accessName);

                    // if no relation from db column to prop name is found,
                    // assume that both are equal
                    String propname = (localrel == null) ? prel.accessName
                                                         : localrel.propName;
                    String prop = node.getString(propname);

                    if (prop != null && prop.length() > 0) {
                        INode old = (INode) getChildElement(prop);

                        if (old != null && old != node) {
                            // A node with this name already exists. This is a
                            // programming error, throw an exception.
                            throw new RuntimeException("An object named \"" + prop +
                                "\" is already contained in the collection.");
                        }

                        if (state != TRANSIENT) {
                            Transactor tx = Transactor.getInstanceOrFail();
                            SyntheticKey key = new SyntheticKey(this.getKey(), prop);
                            tx.visitCleanNode(key, node);
                            nmgr.registerNode(node, key);
                        }
                    }
                }
            }

            // actually add the new child to the subnode list
            synchronized (subnodes) {
                // check if index is out of bounds when adding
                if (where < 0 || where > subnodes.size()) {
                    subnodes.add(nhandle);
                } else {
                    subnodes.add(where, nhandle);
                }
            }

            if (node != this && !nmgr.isRootNode(node)) {
                // avoid calling getParent() because it would return bogus results
                // for the not-anymore transient node
                Node nparent = (node.parentHandle == null) ? null
                                                           : node.parentHandle.getNode(nmgr);

                // if the node doesn't have a parent yet, or it has one but it's
                // transient while we are persistent, make this the nodes new parent.
                if ((nparent == null) ||
                        ((state != TRANSIENT) && (nparent.getState() == TRANSIENT))) {
                    node.setParent(this);
                    node.anonymous = true;
                }
            }
        }

        lastmodified = System.currentTimeMillis();
        // we want the element name to be recomputed on the child node
        node.lastNameCheck = 0;
        registerSubnodeChange();

        return node;
    }

    /**
     *
     *
     * @return ...
     */
    public INode createNode() {
        // create new node at end of subnode array
        return createNode(null, -1);
    }

    /**
     *
     *
     * @param where ...
     *
     * @return ...
     */
    public INode createNode(int where) {
        return createNode(null, where);
    }

    /**
     *
     *
     * @param nm ...
     *
     * @return ...
     */
    public INode createNode(String nm) {
        // parameter where is  ignored if nm != null so we try to avoid calling numberOfNodes()
        return createNode(nm, -1);
    }

    /**
     *
     *
     * @param nm ...
     * @param where ...
     *
     * @return ...
     */
    public INode createNode(String nm, int where) {
        // checkWriteLock();

        boolean anon = false;

        if ((nm == null) || "".equals(nm.trim())) {
            anon = true;
        }

        String proto = null;

        // try to get proper prototype for new node
        if (dbmap != null) {
            DbMapping childmap = anon ?
                dbmap.getSubnodeMapping() :
                dbmap.getPropertyMapping(nm);
            if (childmap != null) {
                proto = childmap.getTypeName();
            }
        }

        Node n = new Node(nm, proto, nmgr);

        if (anon) {
            addNode(n, where);
        } else {
            setNode(nm, n);
        }

        return n;
    }


    /**
     * This implements the getChildElement() method of the IPathElement interface
     */
    public IPathElement getChildElement(String name) {
        if (dbmap != null) {
            // if a dbmapping is provided, check what it tells us about
            // getting this specific child element
            Relation rel = dbmap.getExactPropertyRelation(name);

            if (rel != null && !rel.isPrimitive()) {
                return getNode(name);
            }

            rel = dbmap.getSubnodeRelation();

            if ((rel != null) && (rel.groupby != null || rel.accessName != null)) {
                if (state != TRANSIENT && rel.otherType != null && rel.otherType.isRelational()) {
                    return nmgr.getNode(this, name, rel);
                } else {
                    // Do what we have to do: loop through subnodes and
                    // check if any one matches
                    String propname = rel.groupby != null ? "groupname" : rel.accessName;
                    INode node = null;
                    Enumeration e = getSubnodes();
                    while (e.hasMoreElements()) {
                        Node n = (Node) e.nextElement();
                        if (name.equalsIgnoreCase(n.getString(propname))) {
                            node = n;
                            break;
                        }
                    }
                    // set DbMapping for embedded db group nodes
                    if (node != null && rel.groupby != null) {
                         node.setDbMapping(dbmap.getGroupbyMapping());
                    }
                    return node;
                }
            }

            return getSubnode(name);
        } else {
            // no dbmapping - just try child collection first, then named property.
            INode child = getSubnode(name);

            if (child == null) {
                child = getNode(name);
            }

            return child;
        }
    }

    /**
     * This implements the getParentElement() method of the IPathElement interface
     */
    public IPathElement getParentElement() {
        return getParent();
    }

    /**
     * Get a named child node with the given id.
     */
    public INode getSubnode(String subid) {
        if (subid == null || subid.length() == 0) {
            return null;
        }

        Node retval = null;
        loadNodes();
        if (subnodes == null || subnodes.size() == 0) {
            return null;
        }

        NodeHandle nhandle = null;
        int l = subnodes.size();

        for (int i = 0; i < l; i++) {
            try {
                NodeHandle shandle = subnodes.get(i);
                if (subid.equals(shandle.getID())) {
                    nhandle = shandle;
                    break;
                }
            } catch (Exception x) {
                break;
            }
        }

        if (nhandle != null) {
            retval = nhandle.getNode(nmgr);
        }

        // This would be a better way to do it, without loading the subnodes,
        // but it currently isn't supported by NodeManager.
        //    if (dbmap != null && dbmap.getSubnodeRelation () != null)
        //         retval = nmgr.getNode (this, subid, dbmap.getSubnodeRelation ());

        if (retval != null && retval.parentHandle == null && !nmgr.isRootNode(retval)) {
            retval.setParent(this);
            retval.anonymous = true;
        }

        return retval;
    }

    /**
     * Get a node at a given position. This causes the subnode list to be loaded in case
     * it isn't up to date.
     * @param index the subnode index
     * @return the node at the given index
     */
    public INode getSubnodeAt(int index) {
        loadNodes();
        if (subnodes == null) {
            return null;
        }
        return subnodes.getNode(index);
    }

    /**
     * Get or create a group name for a given content node.
     *
     * @param node the content node
     * @param create whether the node should be created if it doesn't exist
     * @return the group node, or null
     */
    protected Node getGroupbySubnode(Node node, boolean create) {
        if (node.dbmap != null && node.dbmap.isGroup()) {
            return null;
        }
       
        if (dbmap != null) {
            Relation subrel = dbmap.getSubnodeRelation();

            if (subrel != null && subrel.groupby != null) {
                // use actual child mapping to resolve group property name,
                // otherwise the subnode mapping defined for the collection.
                DbMapping childmap = node.dbmap == null ? subrel.otherType : node.dbmap;
                Relation grouprel = childmap.columnNameToRelation(subrel.groupby);
                // If group name can't be resolved to a property name use the group name itself
                String groupprop = (grouprel != null) ? grouprel.propName : subrel.groupby;
                String groupname = node.getString(groupprop);
                Node groupbyNode = (Node) getChildElement(groupname);

                // if group-by node doesn't exist, we'll create it
                if (groupbyNode == null) {
                    groupbyNode = getGroupbySubnode(groupname, create);
                    // mark subnodes as changed as we have a new group node
                    if (create && groupbyNode != null) {
                        Transactor.getInstance().visitParentNode(this);
                    }
                } else {
                    groupbyNode.setDbMapping(dbmap.getGroupbyMapping());
                }

                return groupbyNode;
            }
        }
        return null;
    }

    /**
     * Get or create a group name for a given group name.
     *
     * @param groupname the group name
     * @param create whether the node should be created if it doesn't exist
     * @return the group node, or null
     */
    protected Node getGroupbySubnode(String groupname, boolean create) {
        if (groupname == null) {
            throw new IllegalArgumentException("Can't create group by null");
        }

        boolean persistent = state != TRANSIENT;

        loadNodes();

        if (subnodes == null) {
            subnodes = new SubnodeList(this);
        }

        if (create || subnodes.contains(new NodeHandle(new SyntheticKey(getKey(), groupname)))) {
            try {
                DbMapping groupbyMapping = dbmap.getGroupbyMapping();
                boolean relational = groupbyMapping.getSubnodeMapping().isRelational();

                if (relational || create) {
                    Node node;
                    if (relational && persistent) {
                        node = new Node(this, groupname, nmgr, null);
                    } else {
                        node = new Node(groupname, null, nmgr);
                        node.setParent(this);
                    }

                    // set "groupname" property to value of groupby field
                    node.setString("groupname", groupname);
                    // Set the dbmapping on the group node
                    node.setDbMapping(groupbyMapping);
                    node.setPrototype(groupbyMapping.getTypeName());

                    // if we're relational and persistent, make new node persistable
                    if (!relational && persistent) {
                        node.makePersistable();
                        node.checkWriteLock();
                    }

                    // if we created a new node, check if we need to add it to subnodes
                    if (create) {
                        NodeHandle handle = node.getHandle();
                        if (!subnodes.contains(handle))
                            subnodes.add(handle);
                    }

                    // If we created the group node, we register it with the
                    // nodemanager. Otherwise, we just evict whatever was there before
                    if (persistent) {
                        if (create) {
                            // register group node with transactor
                            Transactor tx = Transactor.getInstanceOrFail();
                            tx.visitCleanNode(node);
                            nmgr.registerNode(node);
                        } else {
                            nmgr.evictKey(node.getKey());
                        }
                    }

                    return node;
                }
            } catch (Exception noluck) {
                nmgr.nmgr.app.logError("Error creating group-by node for " + groupname, noluck);
            }
        }

        return null;
    }

    /**
     *
     *
     * @return ...
     */
    public boolean remove() {
        INode parent = getParent();
        if (parent != null) {
            try {
                parent.removeNode(this);
            } catch (Exception x) {
                // couldn't remove from parent. Log and continue
                getApp().logError("Couldn't remove node from parent: " + x);
            }
        }
        deepRemoveNode();
        return true;
    }

    /**
     *
     *
     * @param node ...
     */
    public void removeNode(INode node) {
        Node n = (Node) node;
        releaseNode(n);
    }

    /**
     * "Locally" remove a subnode from the subnodes table.
     * The logical stuff necessary for keeping data consistent is done in
     * {@link #removeNode(INode)}.
     */
    protected void releaseNode(Node node) {

        Node groupNode = getGroupbySubnode(node, false);
        if (groupNode != null) {
            groupNode.releaseNode(node);
            return;
        }

        INode parent = node.getParent();

        checkWriteLock();
        node.checkWriteLock();

        // load subnodes in case they haven't been loaded.
        // this is to prevent subsequent access to reload the
        // index which would potentially still contain the removed child
        loadNodes();

        if (subnodes != null) {
            boolean removed = false;
            synchronized (subnodes) {
                removed = subnodes.remove(node.getHandle());
            }
            if (dbmap != null && dbmap.isGroup() && subnodes.size() == 0) {
                // clean up ourself if we're an empty group node
                remove();
            } else if (removed) {
                registerSubnodeChange();
            }
        }


        // check if subnodes are also accessed as properties. If so, also unset the property
        if (dbmap != null && node.dbmap != null) {
            Relation prel = dbmap.getSubnodeRelation();

            if (prel != null) {
                if (prel.accessName != null) {
                    Relation localrel = node.dbmap.columnNameToRelation(prel.accessName);

                    // if no relation from db column to prop name is found, assume that both are equal
                    String propname = (localrel == null) ? prel.accessName : localrel.propName;
                    String prop = node.getString(propname);

                    if (prop != null) {
                        if (getNode(prop) == node) {
                            unset(prop);
                        }
                        // let the node cache know this key's not for this node anymore.
                        if (state != TRANSIENT) {
                            nmgr.evictKey(new SyntheticKey(getKey(), prop));
                        }
                    }
                } else if (prel.groupby != null) {
                    String prop = node.getString("groupname");
                    if (prop != null && state != TRANSIENT) {
                        nmgr.evictKey(new SyntheticKey(getKey(), prop));
                    }

                }
                // TODO: We should unset constraints to actually remove subnodes here,
                // but omit it by convention and to keep backwards compatible.
                // if (prel.countConstraints() > 1) {
                //    prel.unsetConstraints(this, node);
                // }
            }
        }

        if (parent == this) {
            // node.markAs(MODIFIED);
            node.setParentHandle(null);
        }

        // If subnodes are relational no need to mark this node as modified
        if (ignoreSubnodeChange()) {
            return;
        }

        lastmodified = System.currentTimeMillis();

        if (state == CLEAN) {
            markAs(MODIFIED);
        }
    }

    /**
     * Delete the node from the db. This mainly tries to notify all nodes referring to this that
     * it's going away. For nodes from the embedded db it also does a cascading delete, since
     * it can tell which nodes are actual children and which are just linked in.
     */
    protected void deepRemoveNode() {

        // tell all nodes that are properties of n that they are no longer used as such
        if (propMap != null) {
            for (Enumeration en = propMap.elements(); en.hasMoreElements();) {
                Property p = (Property) en.nextElement();

                if ((p != null) && (p.getType() == Property.NODE)) {
                    Node n = (Node) p.getNodeValue();
                    if (n != null && !n.isRelational() && n.getParent() == this) {
                        n.deepRemoveNode();
                    }
                }
            }
        }

        // cascading delete of all subnodes. This is never done for relational subnodes, because
        // the parent info is not 100% accurate for them.
        if (subnodes != null) {
            Vector v = new Vector();

            // remove modifies the Vector we are enumerating, so we are extra careful.
            for (Enumeration en = getSubnodes(); en.hasMoreElements();) {
                v.add(en.nextElement());
            }

            int m = v.size();

            for (int i = 0; i < m; i++) {
                // getParent() is heuristical/implicit for relational nodes, so we don't base
                // a cascading delete on that criterium for relational nodes.
                Node n = (Node) v.get(i);

                if (!n.isRelational() && n.getParent() == this) {
                    n.deepRemoveNode();
                }
            }
        }

        // mark the node as deleted and evict its primary key
        setParent(null);
        if (primaryKey != null || state != TRANSIENT) {
            nmgr.evictKey(getKey());
        }
        markAs(DELETED);
    }

    /**
     * Check if the given node is contained in this node's child list.
     * If it is contained return its index in the list, otherwise return -1.
     *
     * @param n a node
     *
     * @return the node's index position in the child list, or -1
     */
    public int contains(INode n) {
        if (n == null) {
            return -1;
        }

        loadNodes();

        if (subnodes == null) {
            return -1;
        }

        // if the node contains relational groupby subnodes, the subnodes vector
        // contains the names instead of ids.
        if (!(n instanceof Node)) {
            return -1;
        }

        Node node = (Node) n;

        return subnodes.indexOf(node.getHandle());
    }

    /**
     * Check if the given node is contained in this node's child list. This
     * is similar to <code>contains(INode)</code> but does not load the
     * child index for relational nodes.
     *
     * @param n a node
     * @return true if the given node is contained in this node's child list
     */
    public boolean isParentOf(Node n) {
        if (dbmap != null) {
            Relation subrel = dbmap.getSubnodeRelation();
            // if we're dealing with relational child nodes use
            // Relation.checkConstraints to avoid loading the child index.
            // Note that we only do that if no filter is set, since
            // Relation.checkConstraints() would always return false
            // if there was a filter property.
            if (subrel != null && subrel.otherType != null
                               && subrel.otherType.isRelational()
                               && subrel.filter == null) {
                // first check if types are stored in same table
                if (!subrel.otherType.isStorageCompatible(n.getDbMapping())) {
                    return false;
                }
                // if they are, check if constraints are met
                return subrel.checkConstraints(this, n);
            }
        }
        // just fall back to contains() for non-relational nodes
        return contains(n) > -1;
    }

    /**
     * Count the subnodes of this node. If they're stored in a relational data source, we
     * may actually load their IDs in order to do this.
     */
    public int numberOfNodes() {
        loadNodes();
        return (subnodes == null) ? 0 : subnodes.size();
    }

    /**
     * Make sure the subnode index is loaded for subnodes stored in a relational data source.
     *  Depending on the subnode.loadmode specified in the type.properties, we'll load just the
     *  ID index or the actual nodes.
     */
    public void loadNodes() {
        // Don't do this for transient nodes which don't have an explicit subnode relation set
        if ((state == TRANSIENT || state == NEW) && subnodeRelation == null) {
            return;
        }

        DbMapping subMap = (dbmap == null) ? null : dbmap.getSubnodeMapping();

        if (subMap != null && subMap.isRelational()) {
            // check if subnodes need to be reloaded
            synchronized (this) {
                if (subnodes == null) {
                    createSubnodeList();
                }
                subnodes.update();
            }
        }
    }

    /**
     * Create an empty subnode list.
     * @return List an empty List of the type used by this Node
     */
    public SubnodeList createSubnodeList() {
        Relation subrel = dbmap == null ? null : dbmap.getSubnodeRelation();
        subnodes = subrel == null || !subrel.lazyLoading ?
                new SubnodeList(this) : new SegmentedSubnodeList(this);
        return subnodes;
    }

    /**
     * Compute a serial number indicating the last change in subnode collection
     * @return a serial number that increases with each subnode change
     */
    long getLastSubnodeChange() {
        // TODO check if we should compute this on demand
        if (subnodes == null) {
            createSubnodeList();
        }
        return subnodes.getLastSubnodeChange();
    }

    /**
     *
     *
     * @param startIndex ...
     * @param length ...
     *
     * @throws Exception ...
     */
    public void prefetchChildren(int startIndex, int length) {
        if (startIndex < 0) {
            return;
        }

        loadNodes();

        if (subnodes == null || startIndex >= subnodes.size()) {
            return;
        }

        subnodes.prefetch(startIndex, length);
    }

    /**
     * Enumerate through the subnodes of this node.
     * @return an enumeration of this node's subnodes
     */
    public Enumeration getSubnodes() {
        loadNodes();
        return getLoadedSubnodes();
    }

    private Enumeration getLoadedSubnodes() {
        final SubnodeList list = subnodes;
        if (list == null) {
            return new EmptyEnumeration();
        }

        return new Enumeration() {
            int pos = 0;

            public boolean hasMoreElements() {
                return pos < list.size();
            }

            public Object nextElement() {
                // prefetch in batches of 100
                // if (pos % 100 == 0)
                //     list.prefetch(pos, 100);
                return list.getNode(pos++);
            }
        };
    }

    /**
     * Return this Node's subnode list
     *
     * @return the subnode list
     */
    public SubnodeList getSubnodeList() {
        return subnodes;
    }

   /**
    * Return true if a change in subnodes can be ignored because it is
    * stored in the subnodes themselves.
    */
    private boolean ignoreSubnodeChange() {
        Relation rel = (dbmap == null) ? null : dbmap.getSubnodeRelation();

        return ((rel != null) && (rel.otherType != null) && rel.otherType.isRelational());
    }

    /**
     *  Get all properties of this node.
     */
    public Enumeration properties() {
        if ((dbmap != null) && dbmap.isRelational()) {
            // return the properties defined in type.properties, if there are any
            return dbmap.getPropertyEnumeration();
        }

        Relation prel = (dbmap == null) ? null : dbmap.getSubnodeRelation();

        if (state != TRANSIENT && prel != null && prel.hasAccessName() &&
                prel.otherType != null && prel.otherType.isRelational()) {
            // return names of objects from a relational db table
            return nmgr.getPropertyNames(this, prel).elements();
        } else if (propMap != null) {
            // return the actually explicitly stored properties
            return propMap.keys();
        }

        // sorry, no properties for this Node
        return new EmptyEnumeration();
    }

    /**
     *
     *
     * @return ...
     */
    public Hashtable getPropMap() {
        return propMap;
    }

    /**
     *
     *
     * @param propname ...
     *
     * @return ...
     */
    public IProperty get(String propname) {
        return getProperty(propname);
    }

    /**
     *
     *
     * @return ...
     */
    public String getParentInfo() {
        return "anonymous:" + anonymous + ",parentHandle" + parentHandle + ",parent:" +
               getParent();
    }

    /**
     *
     *
     * @param propname ...
     *
     * @return ...
     */
    protected Property getProperty(String propname) {
        if (propname == null) {
            return null;
        }

        Relation rel = dbmap == null ?
                null : dbmap.getExactPropertyRelation(propname);

        // 1) check if the property is contained in the propMap
        Property prop = propMap == null ? null : (Property) propMap.get(propname);

        if (prop != null) {
            if (rel != null) {
                // Is a relational node stored by id but things it's a string or int. Fix it.
                if (rel.otherType != null && prop.getType() != Property.NODE) {
                    prop.convertToNodeReference(rel);
                }
                if (rel.isVirtual()) {
                    // property was found in propMap and is a collection - this is
                    // a collection holding non-relational objects. set DbMapping and
                    // NodeManager
                    Node n = (Node) prop.getNodeValue();
                    if (n != null) {
                        // do set DbMapping for embedded db collection nodes
                        n.setDbMapping(rel.getVirtualMapping());
                    }
                }
            }
            return prop;
        } else if (state == TRANSIENT && rel != null && rel.isVirtual()) {
            // When we get a collection from a transient node for the first time, or when
            // we get a collection whose content objects are stored in the embedded
            // XML data storage, we just want to create and set a generic node without
            // consulting the NodeManager about it.
            Node n = new Node(propname, rel.getPrototype(), nmgr);
            n.setDbMapping(rel.getVirtualMapping());
            n.setParent(this);
            setNode(propname, n);
            return (Property) propMap.get(propname);
        }

        // 2) check if this is a create-on-demand node property
        if (rel != null && (rel.isVirtual() || rel.isComplexReference())) {
            if (state != TRANSIENT) {
                Node n = nmgr.getNode(this, propname, rel);

                if (n != null) {
                    if ((n.parentHandle == null) &&
                            !nmgr.isRootNode(n)) {
                        n.setParent(this);
                        n.name = propname;
                        n.anonymous = false;
                    }
                    return new Property(propname, this, n);
                }
            }
        }

        // 4) nothing to be found - return null
        return null;
    }

    /**
     *
     *
     * @param propname ...
     *
     * @return ...
     */
    public String getString(String propname) {
        Property prop = getProperty(propname);

        try {
            return prop.getStringValue();
        } catch (Exception ignore) {
        }

        return null;
    }

    /**
     *
     *
     * @param propname ...
     *
     * @return ...
     */
    public long getInteger(String propname) {
        Property prop = getProperty(propname);

        try {
            return prop.getIntegerValue();
        } catch (Exception ignore) {
        }

        return 0;
    }

    /**
     *
     *
     * @param propname ...
     *
     * @return ...
     */
    public double getFloat(String propname) {
        Property prop = getProperty(propname);

        try {
            return prop.getFloatValue();
        } catch (Exception ignore) {
        }

        return 0.0;
    }

    /**
     *
     *
     * @param propname ...
     *
     * @return ...
     */
    public Date getDate(String propname) {
        Property prop = getProperty(propname);

        try {
            return prop.getDateValue();
        } catch (Exception ignore) {
        }

        return null;
    }

    /**
     *
     *
     * @param propname ...
     *
     * @return ...
     */
    public boolean getBoolean(String propname) {
        Property prop = getProperty(propname);

        try {
            return prop.getBooleanValue();
        } catch (Exception ignore) {
        }

        return false;
    }

    /**
     *
     *
     * @param propname ...
     *
     * @return ...
     */
    public INode getNode(String propname) {
        Property prop = getProperty(propname);

        try {
            return prop.getNodeValue();
        } catch (Exception ignore) {
        }

        return null;
    }

    /**
     *
     *
     * @param propname ...
     *
     * @return ...
     */
    public Object getJavaObject(String propname) {
        Property prop = getProperty(propname);

        try {
            return prop.getJavaObjectValue();
        } catch (Exception ignore) {
        }

        return null;
    }

    /**
     * Directly set a property on this node
     *
     * @param propname ...
     * @param value ...
     */
    protected void set(String propname, Object value, int type) {
        boolean isPersistable = state != TRANSIENT && isPersistableProperty(propname);
        if (isPersistable) {
            checkWriteLock();
        }

        if (propMap == null) {
            propMap = new Hashtable();
        }

        propname = propname.trim();
        Property prop = (Property) propMap.get(propname);

        if (prop != null) {
            prop.setValue(value, type);
        } else {
            prop = new Property(propname, this);
            prop.setValue(value, type);
            propMap.put(propname, prop);
        }

        lastmodified = System.currentTimeMillis();

        if (state == CLEAN && isPersistable) {
            markAs(MODIFIED);
        }
    }

    /**
     *
     *
     * @param propname ...
     * @param value ...
     */
    public void setString(String propname, String value) {
        // nmgr.logEvent ("setting String prop");
        boolean isPersistable = state != TRANSIENT && isPersistableProperty(propname);
        if (isPersistable) {
            checkWriteLock();
        }

        if (propMap == null) {
            propMap = new Hashtable();
        }

        propname = propname.trim();
        Property prop = (Property) propMap.get(propname);
        String oldvalue = null;

        if (prop != null) {
            oldvalue = prop.getStringValue();

            // check if the value has changed
            if ((value != null) && value.equals(oldvalue)) {
                return;
            }

            prop.setStringValue(value);
        } else {
            prop = new Property(propname, this);
            prop.setStringValue(value);
            propMap.put(propname, prop);
        }

        if (dbmap != null) {

            // check if this may have an effect on the node's parerent's child collection
            // in combination with the accessname or order field.
            Node parent = (parentHandle == null) ? null : (Node) getParent();

            if ((parent != null) && (parent.getDbMapping() != null)) {
                DbMapping parentmap = parent.getDbMapping();
                Relation subrel = parentmap.getSubnodeRelation();
                String dbcolumn = dbmap.propertyToColumnName(propname);

                if (subrel != null && dbcolumn != null) {
                    // inlined version of notifyPropertyChange();
                    if (subrel.order != null && subrel.order.indexOf(dbcolumn) > -1) {
                        parent.registerSubnodeChange();
                    }
                    // check if accessname has changed
                    if (subrel.accessName != null &&
                            subrel.accessName.equals(dbcolumn)) {
                        // if any other node is contained with the new value, remove it
                        INode n = (INode) parent.getChildElement(value);

                        if ((n != null) && (n != this)) {
                            throw new RuntimeException(this +
                                    " already contains an object named " + value);
                        }

                        // check if this node is already registered with the old name;
                        // if so, remove it, then add again with the new acessname
                        if (oldvalue != null) {
                            n = (INode) parent.getChildElement(oldvalue);

                            if (n == this) {
                                parent.unset(oldvalue);
                                parent.addNode(this);

                                // let the node cache know this key's not for this node anymore.
                                nmgr.evictKey(new SyntheticKey(parent.getKey(), oldvalue));
                            }
                        }

                        setName(value);
                    }
                }
            }

            // check if the property we're setting specifies the prototype of this object.
            if (state != TRANSIENT &&
                    propname.equals(dbmap.columnNameToProperty(dbmap.getPrototypeField()))) {
                DbMapping newmap = nmgr.getDbMapping(value);

                if (newmap != null) {
                    // see if old and new prototypes have same storage - otherwise type change is ignored
                    String oldStorage = dbmap.getStorageTypeName();
                    String newStorage = newmap.getStorageTypeName();

                    if (((oldStorage == null) && (newStorage == null)) ||
                            ((oldStorage != null) && oldStorage.equals(newStorage))) {
                        // long now = System.currentTimeMillis();
                        dbmap.setLastDataChange();
                        newmap.setLastDataChange();
                        this.dbmap = newmap;
                        this.prototype = value;
                    }
                }
            }
        }

        lastmodified = System.currentTimeMillis();

        if (state == CLEAN && isPersistable) {
            markAs(MODIFIED);
        }
    }

    /**
     *
     *
     * @param propname ...
     * @param value ...
     */
    public void setInteger(String propname, long value) {
        // nmgr.logEvent ("setting bool prop");
        boolean isPersistable = state != TRANSIENT && isPersistableProperty(propname);
        if (isPersistable) {
            checkWriteLock();
        }

        if (propMap == null) {
            propMap = new Hashtable();
        }

        propname = propname.trim();
        Property prop = (Property) propMap.get(propname);

        if (prop != null) {
            prop.setIntegerValue(value);
        } else {
            prop = new Property(propname, this);
            prop.setIntegerValue(value);
            propMap.put(propname, prop);
        }

        notifyPropertyChange(propname);

        lastmodified = System.currentTimeMillis();

        if (state == CLEAN && isPersistable) {
            markAs(MODIFIED);
        }
    }

    /**
     *
     *
     * @param propname ...
     * @param value ...
     */
    public void setFloat(String propname, double value) {
        // nmgr.logEvent ("setting bool prop");
        boolean isPersistable = state != TRANSIENT && isPersistableProperty(propname);
        if (isPersistable) {
            checkWriteLock();
        }

        if (propMap == null) {
            propMap = new Hashtable();
        }

        propname = propname.trim();
        Property prop = (Property) propMap.get(propname);

        if (prop != null) {
            prop.setFloatValue(value);
        } else {
            prop = new Property(propname, this);
            prop.setFloatValue(value);
            propMap.put(propname, prop);
        }

        notifyPropertyChange(propname);

        lastmodified = System.currentTimeMillis();

        if (state == CLEAN && isPersistable) {
            markAs(MODIFIED);
        }
    }

    /**
     *
     *
     * @param propname ...
     * @param value ...
     */
    public void setBoolean(String propname, boolean value) {
        // nmgr.logEvent ("setting bool prop");
        boolean isPersistable = state != TRANSIENT && isPersistableProperty(propname);
        if (isPersistable) {
            checkWriteLock();
        }

        if (propMap == null) {
            propMap = new Hashtable();
        }

        propname = propname.trim();
        Property prop = (Property) propMap.get(propname);

        if (prop != null) {
            prop.setBooleanValue(value);
        } else {
            prop = new Property(propname, this);
            prop.setBooleanValue(value);
            propMap.put(propname, prop);
        }

        notifyPropertyChange(propname);

        lastmodified = System.currentTimeMillis();

        if (state == CLEAN && isPersistable) {
            markAs(MODIFIED);
        }
    }

    /**
     *
     *
     * @param propname ...
     * @param value ...
     */
    public void setDate(String propname, Date value) {
        // nmgr.logEvent ("setting date prop");
        boolean isPersistable = state != TRANSIENT && isPersistableProperty(propname);
        if (isPersistable) {
            checkWriteLock();
        }

        if (propMap == null) {
            propMap = new Hashtable();
        }

        propname = propname.trim();
        Property prop = (Property) propMap.get(propname);

        if (prop != null) {
            prop.setDateValue(value);
        } else {
            prop = new Property(propname, this);
            prop.setDateValue(value);
            propMap.put(propname, prop);
        }

        notifyPropertyChange(propname);

        lastmodified = System.currentTimeMillis();

        if (state == CLEAN && isPersistable) {
            markAs(MODIFIED);
        }
    }

    /**
     *
     *
     * @param propname ...
     * @param value ...
     */
    public void setJavaObject(String propname, Object value) {
        // nmgr.logEvent ("setting jobject prop");
        boolean isPersistable = state != TRANSIENT && isPersistableProperty(propname);
        if (isPersistable) {
            checkWriteLock();
        }

        if (propMap == null) {
            propMap = new Hashtable();
        }

        propname = propname.trim();
        Property prop = (Property) propMap.get(propname);

        if (prop != null) {
            prop.setJavaObjectValue(value);
        } else {
            prop = new Property(propname, this);
            prop.setJavaObjectValue(value);
            propMap.put(propname, prop);
        }

        notifyPropertyChange(propname);

        lastmodified = System.currentTimeMillis();

        if (state == CLEAN && isPersistable) {
            markAs(MODIFIED);
        }
    }

    /**
     *
     *
     * @param propname ...
     * @param value ...
     */
    public void setNode(String propname, INode value) {
        // nmgr.logEvent ("setting node prop");
        // check if types match, otherwise throw exception
        Relation rel = (dbmap == null) ?
                null : dbmap.getExactPropertyRelation(propname);
        DbMapping nmap = (rel == null) ? null : rel.getPropertyMapping();
        DbMapping vmap = value.getDbMapping();

        if ((nmap != null) && (nmap != vmap)) {
            if (vmap == null) {
                value.setDbMapping(nmap);
            } else if (!nmap.isStorageCompatible(vmap) && !rel.isComplexReference()) {
                throw new RuntimeException("Can't set " + propname +
                                           " to object with prototype " +
                                           value.getPrototype() + ", was expecting " +
                                           nmap.getTypeName());
            }
        }

        if (state != TRANSIENT) {
            checkWriteLock();
        }

        Node n = null;

        if (value instanceof Node) {
            n = (Node) value;
        } else {
            throw new RuntimeException("Can't add fixed-transient node to a persistent node");
        }

        boolean isPersistable = isPersistableProperty(propname);
        // if the new node is marked as TRANSIENT and this node is not, mark new node as NEW
        if (state != TRANSIENT && n.state == TRANSIENT && isPersistable) {
            n.makePersistable();
        }

        if (state != TRANSIENT) {
            n.checkWriteLock();
        }

        // check if the main identity of this node is as a named property
        // or as an anonymous node in a collection
        if (n != this && !nmgr.isRootNode(n) && isPersistable) {
            // avoid calling getParent() because it would return bogus results
            // for the not-anymore transient node
            Node nparent = (n.parentHandle == null) ? null
                                                    : n.parentHandle.getNode(nmgr);

            // if the node doesn't have a parent yet, or it has one but it's
            // transient while we are persistent, make this the nodes new parent.
            if ((nparent == null) ||
               ((state != TRANSIENT) && (nparent.getState() == TRANSIENT))) {
                n.setParent(this);
                n.name = propname;
                n.anonymous = false;
            }
        }

        propname = propname.trim();
        if (rel == null && dbmap != null) {
            // widen relation to non-exact (collection) mapping
            rel = dbmap.getPropertyRelation(propname);
        }

        if (rel != null && state != TRANSIENT && (rel.countConstraints() > 1 || rel.isComplexReference())) {
            rel.setConstraints(this, n);
            if (rel.isComplexReference()) {
                Key key = new MultiKey(n.getDbMapping(), rel.getKeyParts(this));
                nmgr.nmgr.registerNode(n, key);
                return;
            }
        }

        Property prop = (propMap == null) ? null : (Property) propMap.get(propname);

        if (prop != null) {
            if ((prop.getType() == IProperty.NODE) &&
                    n.getHandle().equals(prop.getNodeHandle())) {
                // nothing to do, just clean up locks and return
                if (state == CLEAN) {
                    clearWriteLock();
                }

                if (n.state == CLEAN) {
                    n.clearWriteLock();
                }

                return;
            }
        } else {
            prop = new Property(propname, this);
        }

        prop.setNodeValue(n);

        if ((rel == null) ||
                rel.isReference() ||
                state == TRANSIENT ||
                rel.otherType == null ||
                !rel.otherType.isRelational()) {
            // the node must be stored as explicit property
            if (propMap == null) {
                propMap = new Hashtable();
            }

            propMap.put(propname, prop);

            if (state == CLEAN && isPersistable) {
                markAs(MODIFIED);
            }
        }

        // don't check node in transactor cache if node is transient -
        // this is done anyway when the node becomes persistent.
        if (n.state != TRANSIENT) {
            // check node in with transactor cache
            Transactor tx = Transactor.getInstanceOrFail();

            // tx.visitCleanNode (new DbKey (dbm, nID), n);
            // UPDATE: using n.getKey() instead of manually constructing key. HW 2002/09/13
            tx.visitCleanNode(n.getKey(), n);

            // if the field is not the primary key of the property, also register it
            if ((rel != null) && (rel.accessName != null) && (state != TRANSIENT)) {
                Key secKey = new SyntheticKey(getKey(), propname);
                nmgr.registerNode(n, secKey);
                tx.visitCleanNode(secKey, n);
            }
        }

        lastmodified = System.currentTimeMillis();

        if (n.state == DELETED) {
            n.markAs(MODIFIED);
        }
    }

    private boolean isPersistableProperty(String propname) {
        return propname.length() > 0 && propname.charAt(0) != '_';
    }

    /**
     * Remove a property. Note that this works only for explicitly set properties, not for those
     * specified via property relation.
     */
    public void unset(String propname) {

        try {
            // if node is relational, leave a null property so that it is
            // updated in the DB. Otherwise, remove the property.
            Property p = null;
            boolean relational = (dbmap != null) && dbmap.isRelational();

            if (propMap != null) {
                if (relational) {
                    p = (Property) propMap.get(propname);
                } else {
                    p = (Property) propMap.remove(propname);
                }
            }

            if (p != null) {
                boolean isPersistable = state != TRANSIENT && isPersistableProperty(propname);
                if (isPersistable) {
                    checkWriteLock();
                }

                if (relational) {
                    p.setStringValue(null);
                    notifyPropertyChange(propname);
                }

                lastmodified = System.currentTimeMillis();

                if (state == CLEAN && isPersistable) {
                    markAs(MODIFIED);
                }
            } else if (dbmap != null) {
                // check if this is a complex constraint and we have to
                // unset constraints.
                Relation rel = dbmap.getExactPropertyRelation(propname);

                if (rel != null && (rel.isComplexReference())) {
                    p = getProperty(propname);
                    rel.unsetConstraints(this, p.getNodeValue());
                }
            }
        } catch (Exception x) {
            getApp().logError("Error unsetting property", x);
        }
    }

    /**
     *
     *
     * @return ...
     */
    public long lastModified() {
        return lastmodified;
    }

    /**
     *
     *
     * @return ...
     */
    public long created() {
        return created;
    }

    /**
     * Return a string representation for this node. This tries to call the
     * javascript implemented toString() if it is defined.
     * @return a string representing this node.
     */
    public String toString() {
        try {
            // We need to reach deap into helma.framework.core to invoke toString(),
            // but the functionality is really worth it.
            RequestEvaluator reval = getApp().getCurrentRequestEvaluator();
            if (reval != null) {
                Object str = reval.invokeDirectFunction(this, "toString", RequestEvaluator.EMPTY_ARGS);
                if (str instanceof String)
                    return (String) str;
            }
        } catch (Exception x) {
            // fall back to default representation
        }
        return "HopObject " + name;
    }

    /**
     * Tell whether this node is stored inside a relational db. This doesn't mean
     * it actually is stored in a relational db, just that it would be, if the node was
     * persistent
     */
    public boolean isRelational() {
        return (dbmap != null) && dbmap.isRelational();
    }

    /**
     * Public method to make a node persistent.
     */
    public void persist() {
        if (state == TRANSIENT) {
            makePersistable();
        } else if (state == CLEAN) {
            markAs(MODIFIED);
        }

    }

    /**
     * Turn node status from TRANSIENT to NEW so that the Transactor will
     * know it has to insert this node. Recursively persistifies all child nodes
     * and references. This method will immediately cause the node it is called upon to
     * be stored in db when the transaction is committed, so it should be called
     * with care.
     */
    private void makePersistable() {
        // if this isn't a transient node, do nothing.
        if (state != TRANSIENT) {
            return;
        }

        // mark as new
        setState(NEW);

        // generate a real, persistent ID for this object
        id = nmgr.generateID(dbmap);
        getHandle().becomePersistent();

        // register node with the transactor
        Transactor tx = Transactor.getInstanceOrFail();
        tx.visitDirtyNode(this);
        tx.visitCleanNode(this);

        // recursively make children persistable
        makeChildrenPersistable();
    }

    /**
     * Recursively turn node status from TRANSIENT to NEW on child nodes
     * so that the Transactor knows they are to be persistified. This method
     * can be called on TRANSIENT nodes that have just been made perstable
     * using makePersistable() or converted to virtual using convertToVirtual().
     */
    private void makeChildrenPersistable() {
        Relation subrel = dbmap == null ? null : dbmap.getSubnodeRelation();
        for (Enumeration e = getLoadedSubnodes(); e.hasMoreElements();) {
            Node node = (Node) e.nextElement();

            if (node.state == TRANSIENT) {
                DbMapping submap = node.getDbMapping();
                if (submap != null && submap.isVirtual() && !submap.needsPersistence()) {
                    convertToVirtual(node);
                } else {
                    node.makePersistable();
                    if (subrel != null && subrel.countConstraints() > 1) {
                        subrel.setConstraints(this, node);
                    }
                }
            }
        }

        // no need to make properties of virtual nodes persistable
        if (state == VIRTUAL) return;

        for (Enumeration e = properties(); e.hasMoreElements();) {
            String propname = (String) e.nextElement();
            IProperty next = get(propname);

            if (next == null || next.getType() != IProperty.NODE) {
                continue;
            }

            // check if this property actually needs to be persisted.
            Node node = (Node) next.getNodeValue();
            Relation rel = null;

            if (node == null || node == this) {
                continue;
            }

            rel = dbmap == null ? null : dbmap.getExactPropertyRelation(next.getName());
            if (rel != null && rel.isVirtual() && !rel.needsPersistence()) {
                convertToVirtual(node);
            } else {
                node.makePersistable();
                if (rel != null && rel.isComplexReference()) {
                    // if this is a complex reference, make binding properties are set
                    rel.setConstraints(this, node);
                }
            }
        }
    }

    /**
     * Convert a node to a virtual (collection or group ) node. This is used when we
     * encounter a node that is defined as virtual from within the  makePeristable() and
     * makeChildrenPersistable() methods. It will first mark the node as virtual and then
     * call makeChildrenPersistable() on it.
     * @param node a previously transient node to be converted to a virtual node.
     */
    private void convertToVirtual(Node node) {
        // Make node a virtual node with this as parent node. what we do is
        // basically to replay the things done in the constructor for virtual nodes.
        node.setState(VIRTUAL);
        node.primaryKey = new SyntheticKey(getKey(), node.name);
        node.id = node.name;
        node.makeChildrenPersistable();
    }

    /**
     * Get the cache node for this node. This can be
     * used to store transient cache data per node from Javascript.
     */
    public synchronized INode getCacheNode() {
        if (cacheNode == null) {
            cacheNode = new TransientNode();
        }

        return cacheNode;
    }

    /**
     * Reset the cache node for this node.
     */
    public synchronized void clearCacheNode() {
        cacheNode = null;
    }

    /**
     * This method walks down node path to the first non-virtual node and return it.
     *  limit max depth to 5, since there shouldn't be more then 2 layers of virtual nodes.
     */
    public Node getNonVirtualParent() {
        Node node = this;

        for (int i = 0; i < 5; i++) {
            if (node == null) {
                break;
            }

            if (node.getState() == Node.TRANSIENT) {
                DbMapping map = node.getDbMapping();
                if (map == null || !map.isVirtual())
                    return node;
            } else if (node.getState() != Node.VIRTUAL) {
                return node;
            }

            node = (Node) node.getParent();
        }

        return null;
    }

    /**
     *  Instances of this class may be used to mark an entry in the object cache as null.
     *  This method tells the caller whether this is the case.
     */
    public boolean isNullNode() {
        return nmgr == null;
    }

    String generateTransientID() {
        // make transient ids differ from persistent ones
        // and are unique within on runtime session
        return "t" + idgen++;
    }

    /**
     * We overwrite hashCode to make it dependant from the prototype. That way, when the prototype
     * changes, the node will automatically get a new ESNode wrapper, since they're cached in a hashtable.
     * You gotta love these hash code tricks ;-)
     */
    public int hashCode() {
        if (prototype == null) {
            return super.hashCode();
        } else {
            return super.hashCode() + prototype.hashCode();
        }
    }

    /**
     *
     */
    public void dump() {
        System.err.println("subnodes: " + subnodes);
        System.err.println("properties: " + propMap);
    }

    /**
     * Get the application this node belongs to.
     * @return the app we belong to
     */
    private Application getApp() {
        return nmgr.nmgr.app;
    }
}
TOP

Related Classes of helma.objectmodel.db.Node

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.