org.modeshape.jcr.JcrObservationManager Maven / Gradle / Ivy
/*
* ModeShape (http://www.modeshape.org)
*
* Licensed 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.modeshape.jcr;
import static org.modeshape.jcr.api.observation.Event.Sequencing.NODE_SEQUENCED;
import static org.modeshape.jcr.api.observation.Event.Sequencing.NODE_SEQUENCING_FAILURE;
import static org.modeshape.jcr.api.observation.Event.Sequencing.OUTPUT_PATH;
import static org.modeshape.jcr.api.observation.Event.Sequencing.SELECTED_PATH;
import static org.modeshape.jcr.api.observation.Event.Sequencing.SEQUENCED_NODE_ID;
import static org.modeshape.jcr.api.observation.Event.Sequencing.SEQUENCED_NODE_PATH;
import static org.modeshape.jcr.api.observation.Event.Sequencing.SEQUENCER_NAME;
import static org.modeshape.jcr.api.observation.Event.Sequencing.SEQUENCING_FAILURE_CAUSE;
import static org.modeshape.jcr.api.observation.Event.Sequencing.USER_ID;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.NoSuchElementException;
import java.util.Set;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReadWriteLock;
import java.util.concurrent.locks.ReentrantLock;
import java.util.concurrent.locks.ReentrantReadWriteLock;
import javax.jcr.AccessDeniedException;
import javax.jcr.RangeIterator;
import javax.jcr.RepositoryException;
import javax.jcr.nodetype.NodeType;
import javax.jcr.observation.Event;
import javax.jcr.observation.EventIterator;
import javax.jcr.observation.EventJournal;
import javax.jcr.observation.EventListener;
import javax.jcr.observation.EventListenerIterator;
import javax.jcr.observation.ObservationManager;
import org.modeshape.common.annotation.Immutable;
import org.modeshape.common.annotation.NotThreadSafe;
import org.modeshape.common.annotation.ThreadSafe;
import org.modeshape.common.logging.Logger;
import org.modeshape.common.util.CheckArg;
import org.modeshape.common.util.StringUtil;
import org.modeshape.jcr.api.monitor.ValueMetric;
import org.modeshape.jcr.api.observation.PropertyEvent;
import org.modeshape.jcr.api.value.DateTime;
import org.modeshape.jcr.cache.NodeKey;
import org.modeshape.jcr.cache.change.AbstractNodeChange;
import org.modeshape.jcr.cache.change.AbstractSequencingChange;
import org.modeshape.jcr.cache.change.Change;
import org.modeshape.jcr.cache.change.ChangeSet;
import org.modeshape.jcr.cache.change.ChangeSetListener;
import org.modeshape.jcr.cache.change.NodeAdded;
import org.modeshape.jcr.cache.change.NodeMoved;
import org.modeshape.jcr.cache.change.NodeRemoved;
import org.modeshape.jcr.cache.change.NodeRenamed;
import org.modeshape.jcr.cache.change.NodeReordered;
import org.modeshape.jcr.cache.change.NodeSequenced;
import org.modeshape.jcr.cache.change.NodeSequencingFailure;
import org.modeshape.jcr.cache.change.Observable;
import org.modeshape.jcr.cache.change.PropertyAdded;
import org.modeshape.jcr.cache.change.PropertyChanged;
import org.modeshape.jcr.cache.change.PropertyRemoved;
import org.modeshape.jcr.journal.ChangeJournal;
import org.modeshape.jcr.journal.JournalRecord;
import org.modeshape.jcr.value.Name;
import org.modeshape.jcr.value.Path;
import org.modeshape.jcr.value.PathFactory;
/**
* The implementation of JCR {@link ObservationManager}.
*
* @author Horia Chiorean
*/
@ThreadSafe
final class JcrObservationManager implements ObservationManager, ChangeSetListener {
protected static final Logger LOGGER = Logger.getLogger(JcrObservationManager.class);
/**
* The key for storing the {@link JcrObservationManager#setUserData(String) observation user data} in the
* {@link ExecutionContext}'s {@link ExecutionContext#getData() data}.
*/
static final String OBSERVATION_USER_DATA_KEY = "org.modeshape.jcr.observation.userdata";
/**
* The keys which provide extra information in case of a move
*/
static final String MOVE_FROM_KEY = "srcAbsPath";
static final String MOVE_TO_KEY = "destAbsPath";
/**
* The keys which provide extra information in case of a reorder
*/
static final String ORDER_DEST_KEY = "destChildRelPath";
static final String ORDER_SRC_KEY = "srcChildRelPath";
/**
* The repository observable the JCR listeners will be registered with.
*/
private final Observable repositoryObservable;
/**
* The map of the JCR repository listeners and their associated wrapped class.
*/
private final Map listeners;
/**
* The associated session.
*/
protected final JcrSession session;
/**
* An object recording various metrics
*/
private final RepositoryStatistics repositoryStatistics;
/**
* A map of [changeSetHashCode, integer] which keep track of the events which have been received by the observation manager
* and dispatched to the individual listeners. This is used for statistic purposes only.
*/
private final Map changesReceivedAndDispatched;
/**
* A lock which guards the JcrObservationManager#changesReceivedAndDispatched map
*/
private final Lock changesLock;
/**
* A lock used to provide thread-safe guarantees when working it the repository observable
*/
private final ReadWriteLock listenersLock;
/**
* @param session the owning session (never null
)
* @param repositoryObservable the repository observable used to register JCR listeners (never null
)
* @param statistics a {@link RepositoryStatistics} instance with which the {@link ValueMetric#EVENT_QUEUE_SIZE} metric is
* updated
* @throws IllegalArgumentException if either parameter is null
*/
JcrObservationManager( JcrSession session,
Observable repositoryObservable,
RepositoryStatistics statistics ) {
CheckArg.isNotNull(session, "session");
CheckArg.isNotNull(repositoryObservable, "repositoryObservable");
CheckArg.isNotNull(statistics, "statistics");
this.session = session;
this.repositoryObservable = repositoryObservable;
this.listenersLock = new ReentrantReadWriteLock(true);
this.listeners = new HashMap<>();
this.repositoryStatistics = statistics;
this.changesLock = new ReentrantLock();
this.changesReceivedAndDispatched = new HashMap<>();
// this is registered as an observer just so it can count the number of events dispatched
this.repositoryObservable.register(this);
}
@Override
public void addEventListener( EventListener listener,
int eventTypes,
String absPath,
boolean isDeep,
String[] uuid,
String[] nodeTypeName,
boolean noLocal ) throws RepositoryException {
CheckArg.isNotNull(listener, "listener");
checkSession(); // make sure session is still active
// create wrapper and register
JcrListenerAdapter adapter = new JcrListenerAdapter(listener, eventTypes, absPath, isDeep, uuid, nodeTypeName, noLocal);
// unregister if already registered
try {
listenersLock.writeLock().lock();
this.repositoryObservable.unregister(adapter);
this.repositoryObservable.register(adapter);
this.listeners.put(listener, adapter);
} finally {
listenersLock.writeLock().unlock();
}
}
/**
* @throws RepositoryException if session is not active
*/
void checkSession() throws RepositoryException {
session.checkLive();
}
@Override
public void notify( ChangeSet changeSet ) {
incrementEventQueueStatistic(changeSet);
}
private void incrementEventQueueStatistic( ChangeSet changeSet ) {
// whenever a change set is received from the bus, increment the que size
if (repositoryStatistics != null) {
repositoryStatistics.increment(ValueMetric.EVENT_QUEUE_SIZE);
}
changesLock.lock();
try {
if (!this.changesReceivedAndDispatched.containsKey(changeSet.hashCode())) {
// none of the adapters have processed this change set yet, register it
changesReceivedAndDispatched.put(changeSet.hashCode(), new AtomicInteger(listeners.size()));
}
} finally {
changesLock.unlock();
}
}
protected void decrementEventQueueStatistic( ChangeSet changeSet ) {
changesLock.lock();
try {
if (changesReceivedAndDispatched.containsKey(changeSet.hashCode())) {
// the change set has already been registered (and the que size incremented)
int timesProcessed = changesReceivedAndDispatched.get(changeSet.hashCode()).decrementAndGet();
if (timesProcessed == 0) {
if (repositoryStatistics != null) {
repositoryStatistics.decrement(ValueMetric.EVENT_QUEUE_SIZE);
}
changesReceivedAndDispatched.remove(changeSet.hashCode());
}
} else {
// the change got to this adapter (from the bus) before it got to the observation manager, so it'll be registered
// this listener already received the event, so total count of expected listeners is decremented by 1
changesReceivedAndDispatched.put(changeSet.hashCode(), new AtomicInteger(listeners.size() - 1));
}
} finally {
changesLock.unlock();
}
}
@Override
public EventListenerIterator getRegisteredEventListeners() throws RepositoryException {
checkSession(); // make sure session is still active
try {
listenersLock.readLock().lock();
return new JcrEventListenerIterator(Collections.unmodifiableSet(this.listeners.keySet()));
} finally {
listenersLock.readLock().unlock();
}
}
/**
* Remove all of the listeners. This is typically called when the {@link JcrSession#logout() session logs out}.
*/
void removeAllEventListeners() {
try {
listenersLock.writeLock().lock();
for (JcrListenerAdapter listener : this.listeners.values()) {
assert (listener != null);
this.repositoryObservable.unregister(listener);
}
this.listeners.clear();
} finally {
listenersLock.writeLock().unlock();
}
}
@Override
public void removeEventListener( EventListener listener ) throws RepositoryException {
checkSession(); // make sure session is still active
CheckArg.isNotNull(listener, "listener");
try {
listenersLock.writeLock().lock();
JcrListenerAdapter jcrListener = this.listeners.remove(listener);
if (jcrListener != null) {
this.repositoryObservable.unregister(jcrListener);
}
} finally {
listenersLock.writeLock().unlock();
}
}
@Override
public void setUserData( String userData ) {
// User data value may be null
this.session.addContextData(OBSERVATION_USER_DATA_KEY, userData);
}
/**
* {@inheritDoc}
*
* ModeShape supports journaled observation only if journaling is configured as such in the repository configuration.
*
*
* @see javax.jcr.observation.ObservationManager#getEventJournal()
* @see RepositoryConfiguration#getJournaling()
*/
@Override
public EventJournal getEventJournal() {
return session.repository().journalId() != null ? new JcrEventJournal() : null;
}
/**
* {@inheritDoc}
*
* ModeShape supports journaled observation only if journaling is configured as such in the repository configuration.
*
*
* @see javax.jcr.observation.ObservationManager#getEventJournal(int, java.lang.String, boolean, java.lang.String[],
* java.lang.String[])
* @see RepositoryConfiguration#getJournaling()
*/
@Override
public EventJournal getEventJournal( int eventTypes,
String absPath,
boolean isDeep,
String[] uuid,
String[] nodeTypeName ) {
return session.repository().journalId() != null ? new JcrEventJournal(absPath, eventTypes, isDeep, nodeTypeName, uuid) : null;
}
/**
* The JcrListener
class wraps JCR {@link EventListener} and is responsible for converting
* {@link org.modeshape.jcr.cache.change.Change events} into JCR {@link Event events}.
*/
@NotThreadSafe
protected final class JcrListenerAdapter implements ChangeSetListener {
/**
* The JCR event listener.
*/
private final EventListener delegate;
private final ChangeSetConverter changeSetConverter;
/**
* @param delegate the JCR listener
* @param eventTypes a combination of one or more JCR event types
* @param absPath the absolute path of a node or null
if all node paths
* @param isDeep indicates if paths below absPath
should be considered
* @param uuids UUIDs or null
* @param nodeTypeNames node type names or null
* @param noLocal indicates if events from this listener's session should be ignored
*/
JcrListenerAdapter( EventListener delegate,
int eventTypes,
String absPath,
boolean isDeep,
String[] uuids,
String[] nodeTypeNames,
boolean noLocal ) {
assert (delegate != null);
this.delegate = delegate;
this.changeSetConverter = new ChangeSetConverter(absPath, eventTypes, isDeep, nodeTypeNames, noLocal, uuids);
}
@Override
public void notify( ChangeSet changeSet ) {
decrementEventQueueStatistic(changeSet);
Collection events = changeSetConverter.convert(changeSet);
// notify delegate
if (!events.isEmpty()) {
this.delegate.onEvent(new JcrEventIterator(events));
}
}
@Override
public boolean equals( Object obj ) {
return (obj != null) && (obj instanceof JcrListenerAdapter) && (this.delegate == ((JcrListenerAdapter)obj).delegate);
}
@Override
public int hashCode() {
return this.delegate.hashCode();
}
@Override
public String toString() {
return delegate.toString();
}
}
/**
* An implementation of JCR {@link javax.jcr.RangeIterator} extended by the event and event listener iterators.
*
* @param the type being iterated over
*/
protected static class JcrRangeIterator implements RangeIterator {
/**
* The elements being iterated over.
*/
private final List extends E> elements;
/**
* The current position in the iterator.
*/
private int position = 0;
/**
* @param elements the elements to iterator over
* @throws IllegalArgumentException if elements
is null
*/
public JcrRangeIterator( Collection extends E> elements ) {
CheckArg.isNotNull(elements, "elements");
this.elements = new ArrayList<>(elements);
}
@Override
public long getPosition() {
return this.position;
}
@Override
public long getSize() {
return this.elements.size();
}
@Override
public boolean hasNext() {
return (getPosition() < getSize());
}
@Override
public Object next() {
if (!hasNext()) {
throw new NoSuchElementException();
}
Object element = this.elements.get(this.position);
++this.position;
return element;
}
@Override
public void remove() {
throw new UnsupportedOperationException();
}
@Override
public void skip( long skipNum ) {
this.position += skipNum;
if (!hasNext()) {
throw new NoSuchElementException();
}
}
}
/**
* An implementation of the JCR {@link EventListenerIterator}.
*/
protected static class JcrEventListenerIterator extends JcrRangeIterator implements EventListenerIterator {
/**
* @param listeners the listeners being iterated over
* @throws IllegalArgumentException if listeners
is null
*/
public JcrEventListenerIterator( Collection listeners ) {
super(listeners);
}
@Override
public EventListener nextEventListener() {
return (EventListener)next();
}
}
/**
* An implementation of JCR {@link javax.jcr.observation.EventIterator}.
*/
protected static class JcrEventIterator extends JcrRangeIterator implements EventIterator {
/**
* @param events the events being iterated over
* @throws IllegalArgumentException if events
is null
*/
public JcrEventIterator( Collection events ) {
super(events);
}
@Override
public Event nextEvent() {
return (Event)next();
}
}
/**
* The information related to and shared by a set of events that represent a single logical operation.
*/
@Immutable
protected static class JcrEventBundle {
/**
* The date and time of the event bundle.
*/
private final DateTime date;
/**
* The user ID.
*/
private final String userId;
/**
* The optional user data string (may be null)
*/
private final String userData;
public JcrEventBundle( DateTime dateTime,
String userId,
String userData ) {
this.userId = userId;
this.userData = userData;
this.date = dateTime;
}
public String getUserID() {
return this.userId;
}
/**
* @return date
*/
public DateTime getDate() {
return date;
}
/**
* @return userData
*/
public String getUserData() {
return userData;
}
}
/**
* An implementation of JCR {@link org.modeshape.jcr.api.observation.Event}.
*/
@Immutable
protected static class JcrEvent implements org.modeshape.jcr.api.observation.Event {
private final String id;
/**
* The node path.
*/
private final String path;
/**
* The event type.
*/
private final int type;
/**
* The immutable bundle information, which may be shared amongst multiple events.
*/
private final JcrEventBundle bundle;
/**
* A map of extra information for regarding the event
*/
private final Map info;
/**
* The primary type of the node.
*/
private final NodeType nodePrimaryType;
/**
* The mixin types of the node.
*/
private final NodeType[] nodeMixinTypes;
JcrEvent( JcrEventBundle bundle,
int type,
String path,
String id,
NodeType nodePrimaryType,
Set nodeMixinTypes ) {
this(bundle, type, path, id, null, nodePrimaryType, nodeMixinTypes);
}
JcrEvent( JcrEventBundle bundle,
int type,
String path,
String id,
Map info,
NodeType nodePrimaryType,
Set nodeMixinTypes ) {
this.type = type;
this.path = path;
this.bundle = bundle;
this.id = id;
this.info = info;
this.nodePrimaryType = nodePrimaryType;
this.nodeMixinTypes = nodeMixinTypes != null ? nodeMixinTypes.toArray(new NodeType[0]) : new NodeType[0];
}
@Override
public String getPath() {
return this.path;
}
@Override
public int getType() {
return this.type;
}
@Override
public String getUserID() {
return bundle.getUserID();
}
@Override
public long getDate() {
return bundle.getDate().getMilliseconds();
}
@Override
public String getIdentifier() {
return id;
}
@Override
public String getUserData() {
return bundle.getUserData();
}
@Override
public Map getInfo() {
return info != null ? Collections.unmodifiableMap(info) : Collections.emptyMap();
}
@Override
public NodeType getPrimaryNodeType() {
return nodePrimaryType;
}
@Override
public NodeType[] getMixinNodeTypes() {
return nodeMixinTypes;
}
@Override
public String toString() {
StringBuilder sb = new StringBuilder();
switch (this.type) {
case Event.NODE_ADDED:
sb.append("Node added");
break;
case Event.NODE_REMOVED:
sb.append("Node removed");
break;
case Event.PROPERTY_ADDED:
sb.append("Property added");
break;
case Event.PROPERTY_CHANGED:
sb.append("Property changed");
break;
case Event.PROPERTY_REMOVED:
sb.append("Property removed");
break;
case Event.NODE_MOVED:
if (info.containsKey(MOVE_FROM_KEY) || info.containsKey(MOVE_TO_KEY)) {
sb.append("Node moved");
sb.append(" from ").append(info.get(MOVE_FROM_KEY)).append(" to ").append(info.get(MOVE_TO_KEY));
} else {
sb.append("Node reordered");
String destination = info.get(ORDER_DEST_KEY).toString();
if (destination == null) {
destination = " at the end of the children list";
}
Object source = info.get(ORDER_SRC_KEY);
if (source != null) {
sb.append(" from ").append(source);
}
sb.append(" to ").append(destination);
}
sb.append(" by ").append(getUserID());
return sb.toString();
case NODE_SEQUENCED:
sb.append("Node sequenced");
sb.append(" sequenced node:").append(info.get(SEQUENCED_NODE_ID)).append(" at path:")
.append(info.get(SEQUENCED_NODE_PATH));
sb.append(" ,output node:").append(getIdentifier()).append(" at path:").append(getPath());
return sb.toString();
case NODE_SEQUENCING_FAILURE: {
sb.append("Node sequencing failure");
sb.append(" sequenced node:").append(info.get(SEQUENCED_NODE_ID)).append(" at path:")
.append(info.get(SEQUENCED_NODE_PATH));
sb.append(" ,cause: ").append(getInfo().get(SEQUENCING_FAILURE_CAUSE));
return sb.toString();
}
}
sb.append(" at ").append(path).append(" by ").append(getUserID());
return sb.toString();
}
}
protected static class JcrPropertyEvent extends JcrEvent implements PropertyEvent {
private final Object currentValue;
private final Object oldValue;
JcrPropertyEvent( JcrEventBundle bundle,
int type,
String path,
String id,
Object currentValue,
Object oldValue,
NodeType nodePrimaryType,
Set nodeMixinTypes ) {
super(bundle, type, path, id, nodePrimaryType, nodeMixinTypes);
this.currentValue = currentValue;
this.oldValue = oldValue;
}
JcrPropertyEvent( JcrEventBundle bundle,
int type,
String path,
String id,
Object currentValue,
NodeType nodePrimaryType,
Set nodeMixinTypes ) {
this(bundle, type, path, id, currentValue, null, nodePrimaryType, nodeMixinTypes);
}
@Override
public Object getCurrentValue() {
return firstValueFrom(currentValue);
}
@Override
public boolean isMultiValue() {
return currentValue instanceof Object[];
}
@Override
public List> getCurrentValues() {
return listValueFrom(currentValue);
}
@Override
public boolean wasMultiValue() {
return oldValue instanceof Object[];
}
@Override
public Object getPreviousValue() {
return firstValueFrom(oldValue);
}
@Override
public List> getPreviousValues() {
return listValueFrom(oldValue);
}
private List> listValueFrom( Object value ) {
if (value == null) {
return null;
}
if (value instanceof Object[]) {
return ((Object[])value).length > 0 ? Arrays.asList((Object[])value) : Collections.emptyList();
}
return Arrays.asList(value);
}
private Object firstValueFrom( Object value ) {
if (value == null) {
return null;
}
if (value instanceof Object[]) {
return ((Object[])value).length > 0 ? ((Object[])value)[0] : null;
}
return value;
}
}
protected final class ChangeSetConverter {
/**
* The node path whose events should be handled (or null
) if all node paths should be handled.
*/
private final String absPath;
/**
* The event types this listener is interested in handling.
*/
private final int eventTypes;
/**
* A flag indicating if events of child nodes of the absPath
should be processed.
*/
private final boolean isDeep;
/**
* The node type names or null
. If a node with one of these types is the source node of an event than this
* listener wants to process that event. If null
or empty than this listener wants to handle nodes of any
* type.
*/
private final String[] nodeTypeNames;
/**
* A flag indicating if events generated by the session that registered this listener should be ignored.
*/
private final boolean noLocal;
/**
* The node UUIDs or null
. If a node with one of these UUIDs is the source node of an event than this
* listener wants to handle this event. If null
or empty than this listener wants to handle nodes with any
* UUID.
*/
private final Set uuids;
protected ChangeSetConverter( String absPath,
int eventTypes,
boolean isDeep,
String[] nodeTypeNames,
boolean noLocal,
String[] uuids ) {
this.absPath = absPath;
this.eventTypes = eventTypes;
this.isDeep = isDeep;
this.nodeTypeNames = nodeTypeNames;
this.noLocal = noLocal;
if (uuids == null) {
this.uuids = null;
} else if (uuids.length == 0) {
this.uuids = Collections.emptySet();
} else {
this.uuids = new HashSet<>(Arrays.asList(uuids));
}
}
protected ChangeSetConverter() {
this(null, org.modeshape.jcr.api.observation.Event.ALL_EVENTS, true, null, false, null);
}
protected List convert( ChangeSet changeSet ) {
List events = new ArrayList<>();
if (shouldRejectChangeSet(changeSet)) {
return events;
}
String userData = changeSet.getUserData().get(OBSERVATION_USER_DATA_KEY);
JcrEventBundle bundle = new JcrEventBundle(changeSet.getTimestamp(), changeSet.getUserId(), userData);
for (Change change : changeSet) {
processChange(events, bundle, change);
}
return events;
}
private boolean shouldRejectChangeSet( ChangeSet changeSet ) {
return !acceptBasedOnOriginatingSession(changeSet) || !acceptBasedOnOriginatingWorkspace(changeSet);
}
private void processChange( List events,
JcrEventBundle bundle,
Change change ) {
if (!(change instanceof AbstractNodeChange)) {
return;
}
AbstractNodeChange nodeChange = (AbstractNodeChange)change;
if (LOGGER.isDebugEnabled()) {
LOGGER.debug("Processing change: " + nodeChange);
}
if (shouldRejectChange(nodeChange)) {
return;
}
// process event making sure we have the right event type
Path newPath = nodeChange.getPath();
String nodeId = nodeIdentifier(nodeChange.getKey());
NodeType primaryType = nodeType(nodeChange.getPrimaryType());
Set mixinTypes = nodeTypes(nodeChange.getMixinTypes());
// node moved
if (nodeChange instanceof NodeMoved) {
NodeMoved nodeMovedChange = (NodeMoved)nodeChange;
Path oldPath = nodeMovedChange.getOldPath();
fireNodeMoved(events, bundle, newPath, nodeId, oldPath, primaryType, mixinTypes);
} else if (nodeChange instanceof NodeRenamed) {
NodeRenamed nodeRenamedChange = (NodeRenamed)nodeChange;
Path oldPath = pathFactory().create(newPath.subpath(0, newPath.size() - 1), nodeRenamedChange.getOldSegment());
fireNodeMoved(events, bundle, newPath, nodeId, oldPath, primaryType, mixinTypes);
} else if (nodeChange instanceof NodeReordered) {
NodeReordered nodeReordered = (NodeReordered)nodeChange;
Path oldPath = nodeReordered.getOldPath();
if (eventListenedFor(Event.NODE_MOVED)) {
Map info = new HashMap<>();
// check if the reordering wasn't at the end by any chance
if (nodeReordered.getReorderedBeforePath() != null) {
info.put(ORDER_DEST_KEY, stringFor(nodeReordered.getReorderedBeforePath().getLastSegment()));
} else {
info.put(ORDER_DEST_KEY, null);
}
if (oldPath != null) {
info.put(ORDER_SRC_KEY, stringFor(oldPath.getLastSegment()));
}
events.add(new JcrEvent(bundle, Event.NODE_MOVED, stringFor(newPath), nodeId,
Collections.unmodifiableMap(info), primaryType, mixinTypes));
}
fireExtraEventsForMove(events, bundle, newPath, nodeId, oldPath, primaryType, mixinTypes);
} else if (nodeChange instanceof NodeAdded && eventListenedFor(Event.NODE_ADDED)) {
// create event for added node
events.add(new JcrEvent(bundle, Event.NODE_ADDED, stringFor(newPath), nodeId, primaryType, mixinTypes));
} else if (nodeChange instanceof NodeRemoved && eventListenedFor(Event.NODE_REMOVED)) {
// create event for removed node
events.add(new JcrEvent(bundle, Event.NODE_REMOVED, stringFor(newPath), nodeId, primaryType, mixinTypes));
} else if (nodeChange instanceof PropertyChanged && eventListenedFor(Event.PROPERTY_CHANGED)) {
// create event for changed property
PropertyChanged propertyChanged = (PropertyChanged)nodeChange;
Name propertyName = propertyChanged.getNewProperty().getName();
Path propertyPath = pathFactory().create(newPath, stringFor(propertyName));
boolean isMultiValue = propertyChanged.getNewProperty().isMultiple();
Object currentValue = isMultiValue ? propertyChanged.getNewProperty().getValuesAsArray() : propertyChanged.getNewProperty()
.getFirstValue();
Object oldValue = null;
if (propertyChanged.getOldProperty() != null) {
boolean wasMultiValue = propertyChanged.getOldProperty().isMultiple();
oldValue = wasMultiValue ? propertyChanged.getOldProperty().getValuesAsArray() : propertyChanged.getOldProperty()
.getFirstValue();
}
events.add(new JcrPropertyEvent(bundle, Event.PROPERTY_CHANGED, stringFor(propertyPath), nodeId, currentValue,
oldValue, primaryType, mixinTypes));
} else if (nodeChange instanceof PropertyAdded && eventListenedFor(Event.PROPERTY_ADDED)) {
PropertyAdded propertyAdded = (PropertyAdded)nodeChange;
Name propertyName = propertyAdded.getProperty().getName();
Path propertyPath = pathFactory().create(newPath, stringFor(propertyName));
boolean isMultiValue = propertyAdded.getProperty().isMultiple();
Object currentValue = isMultiValue ? propertyAdded.getProperty().getValuesAsArray() : propertyAdded.getProperty()
.getFirstValue();
events.add(new JcrPropertyEvent(bundle, Event.PROPERTY_ADDED, stringFor(propertyPath), nodeId, currentValue,
primaryType, mixinTypes));
} else if (nodeChange instanceof PropertyRemoved && eventListenedFor(Event.PROPERTY_REMOVED)) {
// create event for removed property
PropertyRemoved propertyRemoved = (PropertyRemoved)nodeChange;
Name propertyName = propertyRemoved.getProperty().getName();
Path propertyPath = pathFactory().create(newPath, propertyName);
boolean isMultiValue = propertyRemoved.getProperty().isMultiple();
Object currentValue = isMultiValue ? propertyRemoved.getProperty().getValuesAsArray() : propertyRemoved.getProperty()
.getFirstValue();
events.add(new JcrPropertyEvent(bundle, Event.PROPERTY_REMOVED, stringFor(propertyPath), nodeId, currentValue,
primaryType, mixinTypes));
} else if (nodeChange instanceof NodeSequenced && eventListenedFor(NODE_SEQUENCED)) {
// create event for the sequenced node
NodeSequenced sequencedChange = (NodeSequenced)nodeChange;
Map infoMap = createEventInfoMapForSequencerChange(sequencedChange);
events.add(new JcrEvent(bundle, NODE_SEQUENCED, stringFor(sequencedChange.getOutputNodePath()),
nodeIdentifier(sequencedChange.getOutputNodeKey()), infoMap, primaryType, mixinTypes));
} else if (nodeChange instanceof NodeSequencingFailure && eventListenedFor(NODE_SEQUENCING_FAILURE)) {
// create event for the sequencing failure
NodeSequencingFailure sequencingFailure = (NodeSequencingFailure)nodeChange;
Map infoMap = createEventInfoMapForSequencerChange(sequencingFailure);
infoMap.put(SEQUENCING_FAILURE_CAUSE, sequencingFailure.getCause());
events.add(new JcrEvent(bundle, NODE_SEQUENCING_FAILURE, stringFor(sequencingFailure.getPath()), nodeId, infoMap,
primaryType, mixinTypes));
}
}
private Map createEventInfoMapForSequencerChange( AbstractSequencingChange sequencingChange ) {
Map infoMap = new HashMap<>();
infoMap.put(SEQUENCED_NODE_PATH, stringFor(sequencingChange.getPath()));
infoMap.put(SEQUENCED_NODE_ID, nodeIdentifier(sequencingChange.getKey()));
infoMap.put(OUTPUT_PATH, sequencingChange.getOutputPath());
infoMap.put(SELECTED_PATH, sequencingChange.getSelectedPath());
infoMap.put(SEQUENCER_NAME, sequencingChange.getSequencerName());
infoMap.put(USER_ID, sequencingChange.getUserId());
return infoMap;
}
private void fireNodeMoved( List events,
JcrEventBundle bundle,
Path newPath,
String nodeId,
Path oldPath,
NodeType nodePrimaryType,
Set nodeMixinTypes ) {
if (eventListenedFor(Event.NODE_MOVED)) {
Map info = new HashMap<>();
info.put(MOVE_FROM_KEY, stringFor(oldPath));
info.put(MOVE_TO_KEY, stringFor(newPath));
events.add(new JcrEvent(bundle, Event.NODE_MOVED, stringFor(newPath), nodeId, Collections.unmodifiableMap(info),
nodePrimaryType, nodeMixinTypes));
}
fireExtraEventsForMove(events, bundle, newPath, nodeId, oldPath, nodePrimaryType, nodeMixinTypes);
}
private void fireExtraEventsForMove( List events,
JcrEventBundle bundle,
Path newPath,
String nodeId,
Path oldPath,
NodeType nodePrimaryType,
Set nodeMixinTypes ) {
// JCR 1.0 expects these methods in addition to the NODE_MOVED event
if (eventListenedFor(Event.NODE_ADDED)) {
events.add(new JcrEvent(bundle, Event.NODE_ADDED, stringFor(newPath), nodeId, nodePrimaryType, nodeMixinTypes));
}
if (eventListenedFor(Event.NODE_REMOVED)) {
events.add(new JcrEvent(bundle, Event.NODE_REMOVED, stringFor(oldPath), nodeId, nodePrimaryType, nodeMixinTypes));
}
}
private boolean shouldRejectChange( AbstractNodeChange nodeChange ) {
return !acceptBasedOnUuid(nodeChange) || !acceptBasedOnPath(nodeChange) || !acceptBasedOnPermission(nodeChange)
|| !acceptIfLockChange(nodeChange) || !acceptBasedOnNodeTypeName(nodeChange);
}
/**
* In case of changes involving locks from the system workspace, the TCK expects that the only property changes be for
* lock owner and lock isDeep, which will be fired from the locked node. Therefore, we should exclude property
* notifications from the lock node from the system workspace.
*
* @param nodeChange the internal event
* @return true if the change should be accepted/propagated
*/
private boolean acceptIfLockChange( AbstractNodeChange nodeChange ) {
if (!(nodeChange instanceof PropertyAdded || nodeChange instanceof PropertyRemoved || nodeChange instanceof PropertyChanged)) {
return true;
}
Path path = nodeChange.getPath();
if (path.size() < 2) {
return true;
}
Name firstSegmentName = path.subpath(0, 1).getLastSegment().getName();
boolean isSystemLockChange = JcrLexicon.SYSTEM.equals(firstSegmentName)
&& ModeShapeLexicon.LOCKS.equals(path.getParent().getLastSegment().getName());
return !isSystemLockChange;
}
private boolean eventListenedFor( int eventType ) {
return (this.eventTypes & eventType) == eventType;
}
/**
* @param nodeChange the change being processed
* @return true
if the {@link JcrSession#checkPermission(org.modeshape.jcr.value.Path, String...)} returns
* true for a {@link ModeShapePermissions#READ} permission on the node from the change
*/
private boolean acceptBasedOnPermission( AbstractNodeChange nodeChange ) {
try {
session.checkPermission(parentNodePathOfChange(nodeChange), ModeShapePermissions.READ);
return true;
} catch (AccessDeniedException e) {
return false;
}
}
private boolean acceptBasedOnOriginatingWorkspace( ChangeSet changeSet ) {
boolean sameWorkspace = getWorkspaceName().equalsIgnoreCase(changeSet.getWorkspaceName());
boolean isSystemWorkspace = getSystemWorkspaceName().equalsIgnoreCase(changeSet.getWorkspaceName());
return sameWorkspace || isSystemWorkspace;
}
/**
* @param changeSet the changes being processed
* @return true
if event occurred in a different session or if events from same session should be processed
*/
private boolean acceptBasedOnOriginatingSession( ChangeSet changeSet ) {
return !this.noLocal || !getSessionId().equals(changeSet.getSessionId());
}
/**
* @param change the change being processed
* @return true
if all node types should be processed or if changed node type name matches a specified type
*/
private boolean acceptBasedOnNodeTypeName( AbstractNodeChange change ) {
// JSR 283#12.5.3.4.3
if (nodeTypeNames != null && nodeTypeNames.length == 0) {
return false;
}
String[] mixinStrings = null;
if (shouldCheckNodeType()) {
String primaryTypeName = null;
try {
Path parentPath = parentNodePathOfChange(change);
AbstractJcrNode parentNode = session.node(parentPath);
Set parentMixinNames = parentNode.getMixinTypeNames();
mixinStrings = new String[parentMixinNames.size()];
int i = 0;
for (Name mixinName : parentMixinNames) {
mixinStrings[i++] = stringFor(mixinName);
}
primaryTypeName = stringFor(parentNode.getPrimaryTypeName());
return getNodeTypeManager().isDerivedFrom(this.nodeTypeNames, primaryTypeName, mixinStrings);
} catch (RepositoryException e) {
LOGGER.error(e, JcrI18n.cannotPerformNodeTypeCheck, primaryTypeName, Arrays.toString(mixinStrings),
this.nodeTypeNames);
return false;
}
}
return true;
}
/**
* @param change the change being processed
* @return true
if there is no absolute path or if change path matches or optionally is a deep match
*/
private boolean acceptBasedOnPath( AbstractNodeChange change ) {
if (!StringUtil.isBlank(absPath)) {
Path matchPath = session.pathFactory().create(this.absPath);
Path parentPath = parentNodePathOfChange(change);
return this.isDeep ? matchPath.isAtOrAbove(parentPath) : matchPath.equals(parentPath);
}
return true;
}
/**
* @param change the change being processed
* @return true
if there are no UUIDs to match or change UUID matches
*/
private boolean acceptBasedOnUuid( AbstractNodeChange change ) {
// JSR_283#12.5.3.4.2
return this.uuids == null || (!this.uuids.isEmpty() && this.uuids.contains(nodeIdentifier(change.getKey())));
}
private Path parentNodePathOfChange( AbstractNodeChange change ) {
Path changePath = change.getPath();
if (change instanceof PropertyAdded || change instanceof PropertyRemoved || change instanceof PropertyChanged) {
return changePath;
}
return changePath.isRoot() ? changePath : changePath.getParent();
}
/**
* @return true
if the node type of the event locations need to be checked
*/
private boolean shouldCheckNodeType() {
return ((this.nodeTypeNames != null) && (this.nodeTypeNames.length > 0));
}
private String stringFor( Path path ) {
return session.stringFactory().create(path);
}
private String stringFor( Path.Segment segment ) {
return session.stringFactory().create(segment);
}
private String stringFor( Name name ) {
return session.stringFactory().create(name);
}
private PathFactory pathFactory() {
return session.pathFactory();
}
private String getSessionId() {
return session.sessionId();
}
private String getWorkspaceName() {
return session.getWorkspace().getName();
}
private String getSystemWorkspaceName() {
return session.repository().systemWorkspaceName();
}
private String nodeIdentifier( NodeKey key ) {
return session.nodeIdentifier(key);
}
private NodeType nodeType( Name name ) {
return session.repository().nodeTypeManager().getNodeTypes().getNodeType(name);
}
private Set nodeTypes( Set names ) {
NodeTypes nodeTypes = session.repository().nodeTypeManager().getNodeTypes();
Set result = new HashSet<>(names.size());
for (Name name : names) {
result.add(nodeTypes.getNodeType(name));
}
return result;
}
private JcrNodeTypeManager getNodeTypeManager() throws RepositoryException {
return session.getWorkspace().getNodeTypeManager();
}
}
protected class JcrEventJournal implements EventJournal {
private final JcrObservationManager.ChangeSetConverter changeSetConverter;
private long position = -1;
private Iterator eventsIterator = null;
private Iterator recordsIterator = null;
private org.joda.time.DateTime laterThanDate = null;
protected JcrEventJournal() {
this.changeSetConverter = new ChangeSetConverter();
}
protected JcrEventJournal( String absPath,
int eventTypes,
boolean isDeep,
String[] nodeTypeNames,
String[] uuids ) {
this.changeSetConverter = new ChangeSetConverter(absPath, eventTypes, isDeep, nodeTypeNames, false, uuids);
}
@Override
public void skipTo( long date ) {
laterThanDate = new org.joda.time.DateTime(date);
// reset the position and the internal iterator
position = -1;
eventsIterator = null;
}
@Override
public Event nextEvent() {
if (!advance()) {
throw new NoSuchElementException();
}
position++;
return eventsIterator.next();
}
@Override
public void skip( long skipNum ) {
if (skipNum < 0) {
throw new IllegalArgumentException("Illegal argument to skip: " + skipNum);
}
for (int i = 0; i < skipNum; i++) {
nextEvent();
}
}
@Override
public long getSize() {
return -1;
}
@Override
public long getPosition() {
return position;
}
@Override
public boolean hasNext() {
return advance();
}
@Override
public Object next() {
return nextEvent();
}
@Override
public void remove() {
throw new UnsupportedOperationException("Cannot remove events via the event journal iterator");
}
private boolean advance() {
if (eventsIterator != null && eventsIterator.hasNext()) {
return true;
}
if (position == -1) {
// we haven't advanced in this iterator yet, so always get the latest journal entries
ChangeJournal journal = session.repository().journal();
recordsIterator = laterThanDate != null ? journal.recordsNewerThan(new org.joda.time.DateTime(laterThanDate),
true, false).iterator() : journal.allRecords(false)
.iterator();
}
while (recordsIterator.hasNext()) {
// navigate to the next "valid" record
JournalRecord record = recordsIterator.next();
eventsIterator = this.changeSetConverter.convert(record.getChangeSet()).iterator();
if (eventsIterator.hasNext()) {
return true;
}
}
return false;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy