All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apache.jackrabbit.jcr2spi.xml.SessionImporter Maven / Gradle / Ivy

/*
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You under the Apache License, Version 2.0
 * (the "License"); you may not use this file except in compliance with
 * the License.  You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package org.apache.jackrabbit.jcr2spi.xml;

import org.apache.jackrabbit.jcr2spi.SessionImpl;
import org.apache.jackrabbit.jcr2spi.SessionListener;
import org.apache.jackrabbit.jcr2spi.hierarchy.NodeEntry;
import org.apache.jackrabbit.jcr2spi.hierarchy.PropertyEntry;
import org.apache.jackrabbit.jcr2spi.nodetype.EffectiveNodeType;
import org.apache.jackrabbit.jcr2spi.nodetype.EffectiveNodeTypeProvider;
import org.apache.jackrabbit.jcr2spi.operation.AddNode;
import org.apache.jackrabbit.jcr2spi.operation.AddProperty;
import org.apache.jackrabbit.jcr2spi.operation.Operation;
import org.apache.jackrabbit.jcr2spi.operation.Remove;
import org.apache.jackrabbit.jcr2spi.operation.SetMixin;
import org.apache.jackrabbit.jcr2spi.operation.SetPropertyValue;
import org.apache.jackrabbit.jcr2spi.state.ItemStateValidator;
import org.apache.jackrabbit.jcr2spi.state.NodeState;
import org.apache.jackrabbit.jcr2spi.state.PropertyState;
import org.apache.jackrabbit.jcr2spi.state.SessionItemStateManager;
import org.apache.jackrabbit.jcr2spi.state.Status;
import org.apache.jackrabbit.jcr2spi.util.LogUtil;
import org.apache.jackrabbit.jcr2spi.util.ReferenceChangeTracker;
import org.apache.jackrabbit.spi.Name;
import org.apache.jackrabbit.spi.NodeId;
import org.apache.jackrabbit.spi.Path;
import org.apache.jackrabbit.spi.QNodeDefinition;
import org.apache.jackrabbit.spi.QPropertyDefinition;
import org.apache.jackrabbit.spi.QValue;
import org.apache.jackrabbit.spi.commons.conversion.NamePathResolver;
import org.apache.jackrabbit.spi.commons.name.NameConstants;
import org.apache.jackrabbit.spi.commons.value.ValueFormat;
import org.apache.jackrabbit.util.Base64;
import org.apache.jackrabbit.util.TransientFileFactory;
import org.apache.jackrabbit.value.ValueHelper;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import javax.jcr.ImportUUIDBehavior;
import javax.jcr.ItemExistsException;
import javax.jcr.ItemNotFoundException;
import javax.jcr.PathNotFoundException;
import javax.jcr.PropertyType;
import javax.jcr.RepositoryException;
import javax.jcr.Session;
import javax.jcr.Value;
import javax.jcr.lock.LockException;
import javax.jcr.nodetype.ConstraintViolationException;
import javax.jcr.version.VersionException;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.Reader;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.Stack;
import java.util.UUID;

/**
 * SessionImporter...
 */
public class SessionImporter implements Importer, SessionListener {

    private static Logger log = LoggerFactory.getLogger(SessionImporter.class);

    private final NodeState importTarget;
    private final int uuidBehavior;

    private final SessionImpl session;
    private final SessionItemStateManager stateMgr;

    private final Stack parents;

    private boolean importerClosed;
    private boolean sessionClosed;

    /**
     * helper object that keeps track of remapped uuid's and imported reference
     * properties that might need correcting depending on the uuid mappings
     */
    private final ReferenceChangeTracker refTracker;

    /**
     * Creates a new WorkspaceImporter instance.
     *
     * @param parentPath Path of target node where to add the imported
     * subtree.
     * @param session
     * @param uuidBehavior Flag that governs how incoming UUIDs are handled.
     * @throws PathNotFoundException If no node exists at parentPath
     * or if the current session is not granted read access.
     * @throws ConstraintViolationException If the node at parentPath
     * is protected.
     * @throws VersionException If the node at parentPath is not
     * checked-out.
     * @throws LockException If a lock prevents the addition of the subtree.
     * @throws RepositoryException If another error occurs.
     */
    public SessionImporter(Path parentPath, SessionImpl session,
                           SessionItemStateManager stateManager, int uuidBehavior)
        throws PathNotFoundException, ConstraintViolationException,
        VersionException, LockException, RepositoryException {

        this.session = session;
        this.stateMgr = stateManager;
        this.uuidBehavior = uuidBehavior;

        // perform preliminary checks
        try {
            importTarget = session.getHierarchyManager().getNodeState(parentPath);

            // check if import target is writable, not-locked and checked-out.
            int options = ItemStateValidator.CHECK_ACCESS | ItemStateValidator.CHECK_LOCK | ItemStateValidator.CHECK_VERSIONING;
            session.getValidator().checkIsWritable(importTarget, options);

            refTracker = new ReferenceChangeTracker();
            parents = new Stack();
            parents.push(importTarget);
        } catch (ItemNotFoundException e) {
            throw new PathNotFoundException(LogUtil.safeGetJCRPath(parentPath, session.getPathResolver()));
        }
    }

    //-----------------------------------------------------------< Importer >---
    /**
     * {@inheritDoc}
     */
    public void start() throws RepositoryException {
        // explicitly set status of importer and start listening on session
        setClosed(false);
    }

   /**
     * {@inheritDoc}
     */
    public void startNode(NodeInfo nodeInfo, List propInfos, NamePathResolver resolver)
            throws RepositoryException {
       if (isClosed()) {
           // workspace-importer only: ignore if import has been aborted before.
           return;
       }
       checkSession();
       NodeState parent = parents.peek();
       if (parent == null) {
           // parent node was skipped, skip this child node also
           parents.push(null); // push null onto stack for skipped node
           log.debug("Skipping node '" + nodeInfo.getName() + "'.");
           return;
       }

       NodeEntry parentEntry = (NodeEntry) parent.getHierarchyEntry();
       NodeState nodeState = null;

       if (parentEntry.hasNodeEntry(nodeInfo.getName())) {
           try {
               // a valid child node with that name already exists
               NodeEntry entry = parentEntry.getNodeEntry(nodeInfo.getName(), Path.INDEX_DEFAULT);
               NodeState existing = entry.getNodeState();

               QNodeDefinition def = existing.getDefinition();
               if (!def.allowsSameNameSiblings()) {
                   // existing doesn't allow same-name siblings, check for conflicts
                   EffectiveNodeTypeProvider provider = session.getEffectiveNodeTypeProvider();
                   Name[] ntNames = existing.getAllNodeTypeNames();
                   EffectiveNodeType entExisting = provider.getEffectiveNodeType(ntNames);
                   if (def.isProtected() && entExisting.includesNodeType(nodeInfo.getNodeTypeName())) {
                       // skip protected node
                       parents.push(null); // push null onto stack for skipped node
                       log.debug("skipping protected node " + LogUtil.safeGetJCRPath(existing, session.getPathResolver()));
                       return;
                   }
                   if (def.isAutoCreated() && entExisting.includesNodeType(nodeInfo.getNodeTypeName())) {
                       // this node has already been auto-created, no need to create it
                       nodeState = existing;
                   } else {
                       throw new ItemExistsException(LogUtil.safeGetJCRPath(existing, session.getPathResolver()));
                   }
               }
           } catch (ItemNotFoundException e) {
               // 'existing' doesn't exist any more -> ignore
           }
       }

       if (nodeState == null) {
           // node does not exist -> create new one
           if (nodeInfo.getUUID() == null) {
               // no potential uuid conflict, add new node from given info
               nodeState = importNode(nodeInfo, parent);
           } else {
               // make sure the import does not define a uuid without having
               // a primaryType or mixin that makes the new node referenceable
               checkIncludesMixReferenceable(nodeInfo);

               // potential uuid conflict
               try {
                   NodeId conflictingId = session.getIdFactory().createNodeId(nodeInfo.getUUID());
                   NodeEntry conflicting = session.getHierarchyManager().getNodeEntry(conflictingId);
                   // assert that the entry is available
                   conflicting.getItemState();

                   nodeState = resolveUUIDConflict(parent, conflicting, nodeInfo);
               } catch (ItemNotFoundException e) {
                   // no conflict: create new with given uuid
                   nodeState = importNode(nodeInfo, parent);
               }
           }
       }

       // node state may be 'null' if applicable def is protected
       if (nodeState != null) {
           // process properties
    	   for (PropInfo pi : propInfos) {
               importProperty(pi, nodeState, resolver);
           }
       }

       // push current nodeState onto stack of parents
       parents.push(nodeState);
   }

    /**
     * {@inheritDoc}
     */
    public void endNode(NodeInfo nodeInfo) throws RepositoryException {
        if(isClosed()) {
            // workspace-importer only: ignore if import has been aborted before.
            return;
        }
        parents.pop();
    }

    /**
     * {@inheritDoc}
     */
    public void end() throws RepositoryException {
        if(isClosed()) {
            // workspace-importer only: ignore if import has been aborted before.
            return;
        }

        try {
            checkSession();
            // adjust references referring to remapped uuids
            stateMgr.adjustReferences(refTracker);
        } finally {
            // close this importer since we are done.
            setClosed(true);
        }
    }
    //----------------------------------------------------< SessionListener >---
    /**
     *
     * @param session
     * @see SessionListener#loggingOut(Session)
     */
    public void loggingOut(Session session) {
        // the session will be be valid any more, thus any further calls on
        // the importer must fail
        sessionClosed = true;
    }

    /**
     *
     * @param session
     * @see SessionListener#loggedOut(Session)
     */
    public void loggedOut(Session session) {
        // ignore
    }

    //--------------------------------------------< Importer/Session Status >---
    private void setClosed(boolean isClosed) {
        importerClosed = isClosed;
        if (isClosed) {
            session.removeListener(this);
        } else {
            session.addListener(this);
        }
    }

    private boolean isClosed() {
        return importerClosed;
    }

    private void checkSession() throws RepositoryException {
        if (sessionClosed) {
            throw new RepositoryException("This session has been closed.");
        }
    }

    //----------------------------------------------------< Private methods >---
    /**
     * @param parent
     * @param conflicting
     * @param nodeInfo
     * @return
     * @throws RepositoryException
     */
    NodeState resolveUUIDConflict(NodeState parent, NodeEntry conflicting,
                                  NodeInfo nodeInfo) throws ItemExistsException, RepositoryException {
        NodeState nodeState;
        switch (uuidBehavior) {
            case ImportUUIDBehavior.IMPORT_UUID_CREATE_NEW:
                String originalUUID = nodeInfo.getUUID();
                String newUUID = UUID.randomUUID().toString();
                // reset id on nodeInfo to force creation with new uuid:
                nodeInfo.setUUID(newUUID);
                nodeState = importNode(nodeInfo, parent);
                if (nodeState != null) {
                    // remember uuid mapping
                    refTracker.mappedUUIDs(originalUUID, newUUID);
                }
                break;

            case ImportUUIDBehavior.IMPORT_UUID_COLLISION_THROW:
                String msg = "a node with uuid " + nodeInfo.getUUID() + " already exists!";
                log.debug(msg);
                throw new ItemExistsException(msg);

            case ImportUUIDBehavior.IMPORT_UUID_COLLISION_REMOVE_EXISTING:
                // make sure conflicting node is not importTarget or an ancestor thereof
                Path p0 = importTarget.getPath();
                Path p1 = conflicting.getPath();
                if (p1.equals(p0) || p1.isAncestorOf(p0)) {
                    msg = "cannot remove ancestor node";
                    log.debug(msg);
                    throw new ConstraintViolationException(msg);
                }
                // do remove conflicting (recursive) including validation check
                try {
                    Operation op = Remove.create(conflicting.getNodeState());
                    stateMgr.execute(op);
                } catch (ItemNotFoundException e) {
                    // conflicting does not exist any more. no need for a removal
                }
                // create new with given uuid:
                nodeState = importNode(nodeInfo, parent);
                break;

            case ImportUUIDBehavior.IMPORT_UUID_COLLISION_REPLACE_EXISTING:
                if (conflicting.getNodeState().isRoot()) {
                    msg = "Root node cannot be replaced";
                    log.debug(msg);
                    throw new RepositoryException(msg);
                }

                // 'replace' current parent with parent of conflicting
                parent = conflicting.getParent().getNodeState();

                // do remove conflicting (recursive), including validation checks
                Operation op = Remove.create(conflicting.getNodeState());
                stateMgr.execute(op);
                // create new with given uuid at same location as conflicting
                nodeState = importNode(nodeInfo, parent);
                break;

            default:
                msg = "Unknown uuidBehavior: " + uuidBehavior;
                log.debug(msg);
                throw new RepositoryException(msg);
        }
        return nodeState;
    }

    /**
     *
     * @param nodeInfo
     * @param parent
     * @return
     * @throws ConstraintViolationException
     * @throws ItemNotFoundException
     * @throws RepositoryException
     */
    private NodeState importNode(NodeInfo nodeInfo, NodeState parent) throws ConstraintViolationException, ItemNotFoundException, RepositoryException {
        Name[] parentNtNames = parent.getAllNodeTypeNames();                
        if (parent.hasPropertyName(nodeInfo.getName())) {
            /**
             * a property with the same name already exists; if this property
             * has been imported as well (e.g. through document view import
             * where an element can have the same name as one of the attributes
             * of its parent element) we have to rename the conflicting property;
             *
             * see http://issues.apache.org/jira/browse/JCR-61
             */
            PropertyState conflicting = parent.getPropertyState(nodeInfo.getName());
            if (conflicting.getStatus() == Status.NEW) {
                // assume this property has been imported as well;
                // rename conflicting property
                // TODO: use better reversible escaping scheme to create unique name
                Name newName = session.getNameFactory().create(nodeInfo.getName().getNamespaceURI(), nodeInfo.getName().getLocalName() + "_");
                if (parent.hasPropertyName(newName)) {
                    newName = session.getNameFactory().create(newName.getNamespaceURI(), newName.getLocalName() + "_");
                }
                // since name changes, need to find new applicable definition
                QPropertyDefinition propDef;
                if (conflicting.getValues().length == 1) {
                    // could be single- or multi-valued (n == 1)
                    try {
                        // try single-valued
                        propDef = session.getItemDefinitionProvider().getQPropertyDefinition(parentNtNames, newName, conflicting.getType(), false);
                    } catch (ConstraintViolationException cve) {
                        // try multi-valued
                        propDef = session.getItemDefinitionProvider().getQPropertyDefinition(parentNtNames, newName, conflicting.getType(), true);
                    }
                } else {
                    // can only be multi-valued (n == 0 || n > 1)
                    propDef = session.getItemDefinitionProvider().getQPropertyDefinition(parentNtNames, newName, conflicting.getType(), true);
                }

                Operation ap = AddProperty.create(parent, newName, conflicting.getType(), propDef, conflicting.getValues());
                stateMgr.execute(ap);
                Operation rm = Remove.create(conflicting);
                stateMgr.execute(rm);
            }
        }

        // do create new nodeState
        QNodeDefinition def = session.getItemDefinitionProvider().getQNodeDefinition(parentNtNames, nodeInfo.getName(), nodeInfo.getNodeTypeName());
        if (def.isProtected()) {
            log.debug("Skipping protected nodeState (" + nodeInfo.getName() + ")");
            return null;
        } else {
            Name ntName = nodeInfo.getNodeTypeName();
            if (ntName == null) {
                // use default node type
                ntName = def.getDefaultPrimaryType();
            }
            Operation an = AddNode.create(parent, nodeInfo.getName(), ntName, nodeInfo.getUUID());
            stateMgr.execute(an);
            // retrieve id of state that has been created during execution of AddNode
            NodeState childState = (NodeState) ((AddNode) an).getAddedStates().get(0);

            // and set mixin types
            Name[] mixinNames = nodeInfo.getMixinNames();
            if (mixinNames != null && mixinNames.length > 0) {
                Operation sm = SetMixin.create(childState, nodeInfo.getMixinNames());
                stateMgr.execute(sm);
            }
            return childState;
        }
    }

    /**
     *
     * @param pi
     * @param parentState
     * @param resolver
     * @throws RepositoryException
     * @throws ConstraintViolationException
     */
    private void importProperty(PropInfo pi, NodeState parentState, NamePathResolver resolver) throws RepositoryException, ConstraintViolationException {
        Name propName = pi.getName();
        TextValue[] tva = pi.getValues();
        int infoType = pi.getType();

        PropertyState propState = null;
        QPropertyDefinition def = null;

        NodeEntry parentEntry = (NodeEntry) parentState.getHierarchyEntry();
        PropertyEntry pEntry = parentEntry.getPropertyEntry(propName);
        if (pEntry != null) {
            // a property with that name already exists...
            try {
                PropertyState existing = pEntry.getPropertyState();
                def = existing.getDefinition();
                if (def.isProtected()) {
                    // skip protected property
                    log.debug("skipping protected property " + LogUtil.safeGetJCRPath(existing, session.getPathResolver()));
                    return;
                }
                if (def.isAutoCreated()
                    && (existing.getType() == infoType || infoType == PropertyType.UNDEFINED)
                    && def.isMultiple() == existing.isMultiValued()) {
                    // this property has already been auto-created, no need to create it
                    propState = existing;
                } else {
                    throw new ItemExistsException(LogUtil.safeGetJCRPath(existing, session.getPathResolver()));
                }
            } catch (ItemNotFoundException e) {
                // property doesn't exist any more
                // -> ignore
            }
        }

        Name[] parentNtNames = parentState.getAllNodeTypeNames();
        if (def == null) {
            // there's no property with that name, find applicable definition
            if (tva.length == 1) {
                // could be single- or multi-valued (n == 1)
                def = session.getItemDefinitionProvider().getQPropertyDefinition(parentNtNames, propName, infoType);
            } else {
                // can only be multi-valued (n == 0 || n > 1)
                def = session.getItemDefinitionProvider().getQPropertyDefinition(parentNtNames, propName, infoType, true);
            }
            if (def.isProtected()) {
                // skip protected property
                log.debug("skipping protected property " + propName);
                return;
            }
        }

        // retrieve the target property type needed for creation of QValue(s)
        // including an eventual conversion. the targetType is then needed for
        // setting/updating the type of the property-state.
        int targetType = def.getRequiredType();
        if (targetType == PropertyType.UNDEFINED) {
            if (infoType == PropertyType.UNDEFINED) {
                targetType = PropertyType.STRING;
            } else {
                targetType = infoType;
            }
        }

        QValue[] values = getPropertyValues(pi, targetType, def.isMultiple(), resolver);
        if (propState == null) {
            // create new property
            Operation ap = AddProperty.create(parentState, propName, targetType, def, values);
            stateMgr.execute(ap);
            propState = parentEntry.getPropertyEntry(propName).getPropertyState();
        } else {
            // modify value of existing property
            Operation sp = SetPropertyValue.create(propState, values, targetType);
            stateMgr.execute(sp);
        }

        // store reference for later resolution
        if (propState.getType() == PropertyType.REFERENCE) {
            refTracker.processedReference(propState);
        }
    }

    /**
     *
     * @param propertyInfo
     * @param targetType
     * @param isMultiple
     * @param resolver The name/path resolver used to build QValues.
     * @return
     * @throws RepositoryException
     */
    private QValue[] getPropertyValues(PropInfo propertyInfo, int targetType,
                                       boolean isMultiple, NamePathResolver resolver)
            throws RepositoryException {
        TextValue[] tva = propertyInfo.getValues();
        // check multi-valued characteristic
        if ((tva.length == 0 || tva.length > 1) && !isMultiple) {
            throw new ConstraintViolationException(propertyInfo.getName() + " is not multi-valued.");
        }
        // convert serialized values to QValue objects
        QValue[] iva = new QValue[tva.length];
        for (int i = 0; i < tva.length; i++) {
            iva[i] = buildQValue(tva[i], targetType, resolver);
        }
        return iva;
    }

    /**
     *
     * @param tv
     * @param targetType
     * @param resolver The name/path resolver used to build a QValue.
     * @return
     * @throws RepositoryException
     */
    private QValue buildQValue(TextValue tv, int targetType, NamePathResolver resolver) throws RepositoryException {
        QValue iv;
        try {
            switch (targetType) {
                case PropertyType.BINARY:
                    // base64 encoded BINARY type
                    if (tv.length() < 0x10000) {
                        // < 65kb: deserialize BINARY type in memory
                        ByteArrayOutputStream baos = new ByteArrayOutputStream();
                        Base64.decode(tv.retrieve(), baos);
                        // no need to close ByteArrayOutputStream
                        //baos.close();
                        iv = session.getQValueFactory().create(baos.toByteArray());
                    } else {
                        // >= 65kb: deserialize BINARY type
                        // using Reader and temporary file
                        TransientFileFactory fileFactory = TransientFileFactory.getInstance();
                        File tmpFile = fileFactory.createTransientFile("bin", null, null);
                        FileOutputStream out = new FileOutputStream(tmpFile);
                        Reader reader = tv.reader();
                        try {
                            Base64.decode(reader, out);
                        } finally {
                            reader.close();
                            out.close();
                        }
                        iv = session.getQValueFactory().create(tmpFile);
                    }
                    break;
                default:
                    // build iv using namespace context of xml document
                    Value v = ValueHelper.convert(tv.retrieve(), targetType, session.getValueFactory());
                    iv = ValueFormat.getQValue(v, resolver, session.getQValueFactory());
                    break;
            }
            return iv;
        } catch (IOException e) {
            String msg = "failed to retrieve serialized value";
            log.debug(msg, e);
            throw new RepositoryException(msg, e);
        }
    }

    /**
     * Validate the given NodeInfo: make sure, that if a uuid is
     * defined, the primary or the mixin types include mix:referenceable.
     *
     * @param nodeInfo
     * @throws RepositoryException
     */
    private void checkIncludesMixReferenceable(Importer.NodeInfo nodeInfo) throws RepositoryException {
        List l = new ArrayList();
        l.add(nodeInfo.getNodeTypeName());
        Name[] mixinNames = nodeInfo.getMixinNames();
        if (mixinNames != null && mixinNames.length > 0) {
            l.addAll(Arrays.asList(nodeInfo.getMixinNames()));
        }
        if (l.contains(NameConstants.MIX_REFERENCEABLE)) {
            // shortcut
            return;
        }
        Name[] ntNames = l.toArray(new Name[l.size()]);
        EffectiveNodeType ent = session.getEffectiveNodeTypeProvider().getEffectiveNodeType(ntNames);
        if (!ent.includesNodeType(NameConstants.MIX_REFERENCEABLE)) {
            throw new ConstraintViolationException("XML defines jcr:uuid without defining import node to be referenceable.");
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy