org.apache.jackrabbit.jcr2spi.ItemImpl 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;
import java.util.Collections;
import java.util.Map;
import javax.jcr.AccessDeniedException;
import javax.jcr.InvalidItemStateException;
import javax.jcr.Item;
import javax.jcr.ItemNotFoundException;
import javax.jcr.ItemVisitor;
import javax.jcr.Node;
import javax.jcr.PathNotFoundException;
import javax.jcr.ReferentialIntegrityException;
import javax.jcr.Repository;
import javax.jcr.RepositoryException;
import javax.jcr.Session;
import javax.jcr.UnsupportedRepositoryOperationException;
import javax.jcr.lock.LockException;
import javax.jcr.nodetype.ConstraintViolationException;
import javax.jcr.version.VersionException;
import org.apache.commons.collections.map.ReferenceMap;
import org.apache.jackrabbit.jcr2spi.config.CacheBehaviour;
import org.apache.jackrabbit.jcr2spi.hierarchy.HierarchyEntry;
import org.apache.jackrabbit.jcr2spi.hierarchy.NodeEntry;
import org.apache.jackrabbit.jcr2spi.operation.Operation;
import org.apache.jackrabbit.jcr2spi.operation.Remove;
import org.apache.jackrabbit.jcr2spi.state.ItemState;
import org.apache.jackrabbit.jcr2spi.state.ItemStateLifeCycleListener;
import org.apache.jackrabbit.jcr2spi.state.ItemStateValidator;
import org.apache.jackrabbit.jcr2spi.state.NodeState;
import org.apache.jackrabbit.jcr2spi.state.Status;
import org.apache.jackrabbit.jcr2spi.util.LogUtil;
import org.apache.jackrabbit.spi.Name;
import org.apache.jackrabbit.spi.Path;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* ItemImpl
...
*/
public abstract class ItemImpl implements Item, ItemStateLifeCycleListener {
private static Logger log = LoggerFactory.getLogger(ItemImpl.class);
private final ItemState state;
/**
* The session that created this item.
*/
protected SessionImpl session;
/**
* Listeners (weak references)
*/
@SuppressWarnings("unchecked")
protected final Map listeners =
Collections.synchronizedMap(new ReferenceMap(ReferenceMap.WEAK, ReferenceMap.WEAK));
public ItemImpl(SessionImpl session, ItemState state,
ItemLifeCycleListener[] listeners) {
this.session = session;
this.state = state;
if (listeners != null) {
for (int i = 0; i < listeners.length; i++) {
addLifeCycleListener(listeners[i]);
}
}
notifyCreated();
// add this item as listener to events of the underlying state object
state.addListener(this);
}
//-----------------------------------------------------< Item interface >---
/**
* @see javax.jcr.Item#getPath()
*/
public String getPath() throws RepositoryException {
checkStatus();
return session.getPathResolver().getJCRPath(getQPath());
}
/**
* @see javax.jcr.Item#getName()
*/
public abstract String getName() throws RepositoryException;
/**
* @see javax.jcr.Item#getAncestor(int)
*/
public Item getAncestor(int depth) throws ItemNotFoundException, AccessDeniedException, RepositoryException {
checkStatus();
if (depth == 0) {
return session.getRootNode();
}
String msg = "No ancestor at depth = " + depth;
try {
// Path.getAncestor requires relative degree, i.e. we need
// to convert absolute to relative ancestor degree
Path path = getQPath();
int relDegree = path.getAncestorCount() - depth;
if (relDegree < 0) {
throw new ItemNotFoundException(msg);
}
Path ancestorPath = path.getAncestor(relDegree);
if (relDegree == 0) {
return this;
} else {
return getItemManager().getNode(ancestorPath);
}
} catch (PathNotFoundException e) {
throw new ItemNotFoundException(msg);
}
}
/**
* @see Item#getParent()
*/
public Node getParent() throws ItemNotFoundException, AccessDeniedException, RepositoryException {
checkStatus();
// special treatment for root node
if (state.isNode() && ((NodeState)state).isRoot()) {
String msg = "Root node doesn't have a parent.";
log.debug(msg);
throw new ItemNotFoundException(msg);
}
NodeEntry parentEntry = getItemState().getHierarchyEntry().getParent();
return (Node) getItemManager().getItem(parentEntry);
}
/**
* @see javax.jcr.Item#getDepth()
*/
public int getDepth() throws RepositoryException {
checkStatus();
if (state.isNode() && ((NodeState)state).isRoot()) {
// shortcut
return Path.ROOT_DEPTH;
}
return session.getHierarchyManager().getDepth(state.getHierarchyEntry());
}
/**
* Note: as of 2.x this method returns the session irrespective of the item's
* status.
*
* @see javax.jcr.Item#getSession()
* @see Issue JCR-2529
*/
public Session getSession() throws RepositoryException {
return session;
}
/**
* @see javax.jcr.Item#isNew()
*/
public boolean isNew() {
return state.getStatus() == Status.NEW;
}
/**
* @see javax.jcr.Item#isModified()
*/
public boolean isModified() {
return state.getStatus() == Status.EXISTING_MODIFIED;
}
/**
* @see javax.jcr.Item#isSame(Item)
*/
public boolean isSame(Item otherItem) throws RepositoryException {
checkStatus();
if (this == otherItem) {
return true;
}
if (isNode() != otherItem.isNode()) {
return false;
}
if (otherItem instanceof ItemImpl) {
ItemImpl other = (ItemImpl) otherItem;
if (this.state == other.state) {
return true;
}
// check status of the other item.
other.checkStatus();
// 2 items may only be the same if the were accessed from Sessions
// bound to the same workspace
String otherWspName = other.session.getWorkspace().getName();
if (session.getWorkspace().getName().equals(otherWspName)) {
// in addition they must provide the same id irrespective of
// any transient modifications.
if (state.getStatus() != Status.NEW && other.state.getStatus() != Status.NEW ) {
// if any ancestor is _invalidated_ force it's reload in
// order to detect id changes.
updateId(state);
updateId(other.state);
return state.getWorkspaceId().equals(other.state.getWorkspaceId());
}
/* else:
- if both wsp-states are null, the items are both transiently
added and are only the same if they are obtained from the same
session. in this case, their states must be the same object,
which is covered above.
- either of the two items does not have a workspace state.
therefore the items cannot be the same, since one has been
transiently added in one but not the other session.
*/
}
}
return false;
}
/**
* @see javax.jcr.Item#accept(ItemVisitor)
*/
public abstract void accept(ItemVisitor visitor) throws RepositoryException;
/**
* @see javax.jcr.Item#isNode()
*/
public abstract boolean isNode();
/**
* @see javax.jcr.Item#save()
*/
public void save() throws AccessDeniedException, ConstraintViolationException, InvalidItemStateException, ReferentialIntegrityException, VersionException, LockException, RepositoryException {
// check state of this instance
checkStatus();
session.getSessionItemStateManager().save(getItemState());
}
/**
* @see javax.jcr.Item#refresh(boolean)
*/
public void refresh(boolean keepChanges) throws InvalidItemStateException, RepositoryException {
// check session status
session.checkIsAlive();
int status = state.getStatus();
// check if item has been removed by this or another session
if (Status.isTerminal(status) || Status.EXISTING_REMOVED == status) {
throw new InvalidItemStateException("Item '" + this + "' doesn't exist anymore");
}
/* If 'keepChanges' is true, items that do not have changes pending have
their state refreshed to reflect the current saved state */
if (keepChanges) {
if (status != Status.NEW &&
session.getCacheBehaviour() != CacheBehaviour.OBSERVATION) {
// merge current transient modifications with latest changes
// from the 'server'.
// Note, that with Observation-CacheBehaviour no manual refresh
// is required. changes get pushed automatically.
state.getHierarchyEntry().invalidate(true);
}
} else {
// check status of item state
if (status == Status.NEW) {
String msg = "Cannot refresh a new item (" + safeGetJCRPath() + ").";
log.debug(msg);
throw new RepositoryException(msg);
}
/*
Reset all transient modifications from this item and its descendants.
*/
session.getSessionItemStateManager().undo(state);
/* Unless the session is in 'observation' mode, mark all states
within this tree 'invalidated' in order to have them refreshed
from the server upon the next access.*/
if (session.getCacheBehaviour() != CacheBehaviour.OBSERVATION) {
state.getHierarchyEntry().invalidate(true);
}
}
}
/**
* @see javax.jcr.Item#remove()
*/
public void remove() throws VersionException, LockException, ConstraintViolationException, RepositoryException {
checkSupportedOption(Repository.LEVEL_2_SUPPORTED);
checkStatus();
// validation checks are performed within remove operation
Operation rm = Remove.create(getItemState());
session.getSessionItemStateManager().execute(rm);
}
//-----------------------------------------< ItemStateLifeCycleListener >---
/**
*
* @param state
* @param previousStatus
*/
public void statusChanged(ItemState state, int previousStatus) {
if (state != this.state) {
throw new IllegalArgumentException("Invalid argument: ItemState with changed status must be this.state.");
}
switch (state.getStatus()) {
/**
* Nothing to do for
* - Status#EXISTING : modifications reverted or saved
* inform listeners about an update (status was MODIFIED before)
* or a simple refresh without modification (status was INVALIDATED).
*/
case Status.EXISTING:
if (previousStatus == Status.INVALIDATED || previousStatus == Status.MODIFIED) {
notifyUpdated(previousStatus == Status.MODIFIED);
}
break;
/**
* Nothing to do for
* - Status#EXISTING_MODIFIED : transient modification
* - Status#STALE_MODIFIED : external modifications while transient changes pending
* - Status#STALE_DESTROYED : external modifications while transient changes pending
* - Status#MODIFIED : externally modified -> marker for sessionISM states only
* - Status#EXISTING_REMOVED : transient removal
*/
case Status.EXISTING_MODIFIED:
case Status.STALE_MODIFIED:
case Status.STALE_DESTROYED:
case Status.MODIFIED:
case Status.EXISTING_REMOVED:
break;
/**
* Notify listeners that this item is transiently or permanently
* destroyed.
* - Status#REMOVED : permanent removal. item will never get back to life
*/
case Status.REMOVED:
state.removeListener(this);
notifyDestroyed();
break;
/**
* Invalid status. A state can never change its state to 'New'.
*/
case Status.NEW:
// should never happen.
log.error("invalid state change to STATUS_NEW");
break;
}
}
//----------------------------------------------------------< LiveCycle >---
/**
* Notify the listeners that this instance has been created.
*/
private void notifyCreated() {
// copy listeners to array to avoid ConcurrentModificationException
ItemLifeCycleListener[] la = listeners.values().toArray(new ItemLifeCycleListener[listeners.size()]);
for (int i = 0; i < la.length; i++) {
la[i].itemCreated(this);
}
}
/**
* Notify the listeners that this instance has been updated.
*/
private void notifyUpdated(boolean modified) {
// copy listeners to array to avoid ConcurrentModificationException
ItemLifeCycleListener[] la = listeners.values().toArray(new ItemLifeCycleListener[listeners.size()]);
for (int i = 0; i < la.length; i++) {
if (la[i] != null) {
la[i].itemUpdated(this, modified);
}
}
}
/**
* Notify the listeners that this instance has been destroyed.
*/
private void notifyDestroyed() {
// copy listeners to array to avoid ConcurrentModificationException
ItemLifeCycleListener[] la = listeners.values().toArray(new ItemLifeCycleListener[listeners.size()]);
for (int i = 0; i < la.length; i++) {
if (la[i] != null) {
la[i].itemDestroyed(this);
}
}
}
/**
* Add an ItemLifeCycleListener
*
* @param listener the new listener to be informed on life cycle changes
*/
void addLifeCycleListener(ItemLifeCycleListener listener) {
if (!listeners.containsKey(listener)) {
listeners.put(listener, listener);
}
}
/**
* Remove an ItemLifeCycleListener
*
* @param listener an existing listener
*/
void removeLifeCycleListener(ItemLifeCycleListener listener) {
listeners.remove(listener);
}
//------------------------------------------------------< check methods >---
/**
* Performs a sanity check on this item and the associated session. If
* the underlying item state is in an invalidated state then it will be
* refreshed to get the current status of the item state. The status
* check is then performed on the newly retrieved status.
*
* @throws RepositoryException if this item has been rendered invalid for some reason
*/
protected void checkStatus() throws RepositoryException {
// check session status
session.checkIsAlive();
// check status of this item for read operation
if (state.getStatus() == Status.INVALIDATED) {
// refresh to get current status from persistent storage
state.getHierarchyEntry().reload(false);
}
// now check if valid
if (!state.isValid()) {
throw new InvalidItemStateException("Item '" + this + "' doesn't exist anymore. (Status = " +Status.getName(state.getStatus())+ ")");
}
}
/**
* Returns true if the repository supports the given option. False otherwise.
*
* @param option Any of the option constants defined by {@link Repository}
* that either returns 'true' or 'false'. I.e.
*
* - {@link Repository#LEVEL_1_SUPPORTED}
* - {@link Repository#LEVEL_2_SUPPORTED}
* - {@link Repository#OPTION_TRANSACTIONS_SUPPORTED}
* - {@link Repository#OPTION_VERSIONING_SUPPORTED}
* - {@link Repository#OPTION_OBSERVATION_SUPPORTED}
* - {@link Repository#OPTION_LOCKING_SUPPORTED}
* - {@link Repository#OPTION_QUERY_SQL_SUPPORTED}
*
* @return true if the repository supports the given option. False otherwise.
*/
boolean isSupportedOption(String option) {
return session.isSupportedOption(option);
}
/**
* Check if the given option is supported by the repository.
*
* @param option Any of the option constants defined by {@link Repository}
* that either returns 'true' or 'false'. I.e.
*
* - {@link Repository#LEVEL_1_SUPPORTED}
* - {@link Repository#LEVEL_2_SUPPORTED}
* - {@link Repository#OPTION_ACCESS_CONTROL_SUPPORTED}
* - {@link Repository#OPTION_ACTIVITIES_SUPPORTED}
* - {@link Repository#OPTION_BASELINES_SUPPORTED}
* - {@link Repository#OPTION_JOURNALED_OBSERVATION_SUPPORTED}
* - {@link Repository#OPTION_LIFECYCLE_SUPPORTED}
* - {@link Repository#OPTION_LOCKING_SUPPORTED}
* - {@link Repository#OPTION_NODE_AND_PROPERTY_WITH_SAME_NAME_SUPPORTED}
* - {@link Repository#OPTION_NODE_TYPE_MANAGEMENT_SUPPORTED}
* - {@link Repository#OPTION_OBSERVATION_SUPPORTED}
* - {@link Repository#OPTION_QUERY_SQL_SUPPORTED}
* - {@link Repository#OPTION_RETENTION_SUPPORTED}
* - {@link Repository#OPTION_SHAREABLE_NODES_SUPPORTED}
* - {@link Repository#OPTION_SIMPLE_VERSIONING_SUPPORTED}
* - {@link Repository#OPTION_TRANSACTIONS_SUPPORTED}
* - {@link Repository#OPTION_UNFILED_CONTENT_SUPPORTED}
* - {@link Repository#OPTION_UPDATE_MIXIN_NODE_TYPES_SUPPORTED}
* - {@link Repository#OPTION_UPDATE_PRIMARY_NODE_TYPE_SUPPORTED}
* - {@link Repository#OPTION_VERSIONING_SUPPORTED}
* - {@link Repository#OPTION_WORKSPACE_MANAGEMENT_SUPPORTED}
* - {@link Repository#OPTION_XML_EXPORT_SUPPORTED}
* - {@link Repository#OPTION_XML_IMPORT_SUPPORTED}
* - {@link Repository#WRITE_SUPPORTED}
*
* @throws UnsupportedRepositoryOperationException
* @throws RepositoryException
*/
void checkSupportedOption(String option) throws UnsupportedRepositoryOperationException, RepositoryException {
session.checkSupportedOption(option);
}
/**
* Checks if the repository supports level 2 (writing) and the status of
* this item. Note, that this method does not perform any additional
* validation checks such as access restrictions, locking, checkin status
* or protection that affect the writing to nodes and properties.
*
* @throws UnsupportedRepositoryOperationException
* @throws RepositoryException
* @see ItemStateValidator
*/
protected void checkIsWritable() throws UnsupportedRepositoryOperationException, ConstraintViolationException, RepositoryException {
checkSupportedOption(Repository.LEVEL_2_SUPPORTED);
checkStatus();
}
/**
* Returns true if the repository supports level 2 (writing). Note, that
* this method does not perform any additional validation tests such as
* access restrictions, locking, checkin status or protection that affect
* the writing to nodes and properties.
*
* @throws UnsupportedRepositoryOperationException
* @throws RepositoryException if the sanity check on this item fails.
* See {@link ItemImpl#checkStatus()}.
* @see ItemStateValidator
*/
protected boolean isWritable() throws RepositoryException {
checkStatus();
return session.isSupportedOption(Repository.LEVEL_2_SUPPORTED);
}
//------------------------------------< Implementation specific methods >---
/**
* Same as {@link Item#getName()}
except that
* this method returns a Name
instead of a
* String
.
*
* @return the name of this item as Name
* @throws RepositoryException if an error occurs.
*/
abstract Name getQName() throws RepositoryException;
/**
* Returns the primary path to this Item
.
*
* @return the primary path to this Item
*/
Path getQPath() throws RepositoryException {
return state.getPath();
}
/**
* Returns the item-state associated with this Item
.
*
* @return state associated with this Item
*/
protected ItemState getItemState() {
return state;
}
/**
* Returns the ItemManager associated with this item's Session.
*
* @return ItemManager
*/
protected ItemManager getItemManager() {
return session.getItemManager();
}
/**
* Failsafe conversion of internal Path
to JCR path for use in
* error messages etc.
*
* @return JCR path
*/
String safeGetJCRPath() {
return LogUtil.safeGetJCRPath(getItemState(), session.getPathResolver());
}
/**
*
* @param state
* @throws RepositoryException
*/
private static void updateId(ItemState state) throws RepositoryException {
HierarchyEntry he = state.getHierarchyEntry();
while (he.getStatus() != Status.INVALIDATED) {
he = he.getParent();
if (he == null) {
// root reached without intermediate invalidated entry
return;
}
}
// he is INVALIDATED -> force reloading in order to be aware of id changes
he.getItemState();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy