org.apache.jackrabbit.test.api.observation.PropertyChangedTest 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.test.api.observation;
import org.apache.jackrabbit.test.NotExecutableException;
import javax.jcr.Node;
import javax.jcr.RepositoryException;
import javax.jcr.PropertyType;
import javax.jcr.Value;
import javax.jcr.nodetype.NodeType;
import javax.jcr.observation.Event;
/**
* Test cases for {@link javax.jcr.observation.Event#PROPERTY_CHANGED} events.
*
* Configuration requirements:
*
* The {@link #testRoot} must allow child nodes of type {@link #testNodeType}.
* The child nodes that are created will be named {@link #nodeName1} and
* {@link #nodeName2}.
* {@link #testNodeType} must also support String properties with names
* {@link #propertyName1} and {@link #propertyName2}.
*
*/
public class PropertyChangedTest extends AbstractObservationTest {
/**
* Tests if a {@link javax.jcr.observation.Event#PROPERTY_CHANGED} is
* triggered when a single property is changed.
*/
public void testSinglePropertyChanged() throws RepositoryException {
Node node = testRootNode.addNode(nodeName1, testNodeType);
node.setProperty(propertyName1, "foo");
testRootNode.getSession().save();
EventResult result = new EventResult(log);
addEventListener(result, Event.PROPERTY_CHANGED);
node.getProperty(propertyName1).setValue("foobar");
testRootNode.getSession().save();
Event[] events = result.getEvents(DEFAULT_WAIT_TIMEOUT);
removeEventListener(result);
checkPropertyChanged(events, new String[]{nodeName1 + "/" + propertyName1});
}
/**
* Tests if {@link javax.jcr.observation.Event#PROPERTY_CHANGED} are
* triggered when multiple properties are changed.
* @throws RepositoryException
*/
public void testMultiPropertyChanged() throws RepositoryException {
Node node = testRootNode.addNode(nodeName1, testNodeType);
node.setProperty(propertyName1, "foo");
node.setProperty(propertyName2, "bar");
testRootNode.getSession().save();
EventResult result = new EventResult(log);
addEventListener(result, Event.PROPERTY_CHANGED);
node.getProperty(propertyName1).setValue("foobar");
node.getProperty(propertyName2).setValue("foobar");
testRootNode.getSession().save();
Event[] events = result.getEvents(DEFAULT_WAIT_TIMEOUT);
removeEventListener(result);
checkPropertyChanged(events, new String[]{nodeName1 + "/" + propertyName1,
nodeName1 + "/" + propertyName2});
}
/**
* Tests if a {@link javax.jcr.observation.Event#PROPERTY_CHANGED} is
* triggered only for changed properties and not for new properties.
*/
public void testSinglePropertyChangedWithAdded() throws RepositoryException {
Node node = testRootNode.addNode(nodeName1, testNodeType);
node.setProperty(propertyName1, "foo");
testRootNode.getSession().save();
EventResult result = new EventResult(log);
addEventListener(result, Event.PROPERTY_CHANGED);
node.getProperty(propertyName1).setValue("foobar");
node.setProperty(propertyName2, "bar"); // will not fire prop changed event
testRootNode.getSession().save();
Event[] events = result.getEvents(DEFAULT_WAIT_TIMEOUT);
removeEventListener(result);
checkPropertyChanged(events, new String[]{nodeName1 + "/" + propertyName1});
}
/**
* Tests if either a
*
* - {@link Event#PROPERTY_CHANGED}
* - {@link Event#PROPERTY_REMOVED} and {@link Event#PROPERTY_ADDED}
*
* is triggered if a property is transiently removed and set again with
* the same name but different type and then saved.
*
* If the node type {@link #testNodeType} does not suppport a property with
* name {@link #propertyName1} of type {@link PropertyType#UNDEFINED} a
* {@link NotExecutableException} is thrown.
*/
public void testPropertyRemoveCreate()
throws RepositoryException, NotExecutableException {
Node n = testRootNode.addNode(nodeName1, testNodeType);
NodeType nt = superuser.getWorkspace().getNodeTypeManager().getNodeType(testNodeType);
Value v1 = superuser.getValueFactory().createValue("foo");
Value v2 = superuser.getValueFactory().createValue(System.currentTimeMillis());
if (!nt.canSetProperty(propertyName1, v1) || !nt.canSetProperty(propertyName1, v2)) {
throw new NotExecutableException("Property " + propertyName1 + " is not of type UNDEFINED");
}
n.setProperty(propertyName1, v1);
testRootNode.getSession().save();
EventResult result = new EventResult(log);
addEventListener(result, Event.PROPERTY_ADDED | Event.PROPERTY_CHANGED | Event.PROPERTY_REMOVED);
n.getProperty(propertyName1).remove();
n.setProperty(propertyName1, v2);
testRootNode.getSession().save();
Event[] events = result.getEvents(DEFAULT_WAIT_TIMEOUT);
removeEventListener(result);
if (events.length == 1) {
checkPropertyChanged(events, new String[]{nodeName1 + "/" + propertyName1});
} else {
// prop remove and add event
assertEquals("Expected 2 events for a property type change.", 2, events.length);
int type = Event.PROPERTY_ADDED | Event.PROPERTY_REMOVED;
String path = testRoot + "/" + nodeName1 + "/" + propertyName1;
for (int i = 0; i < events.length; i++) {
assertTrue("Event is not of type PROPERTY_REMOVED or PROPERTY_ADDED", (events[i].getType() & type) > 0);
assertEquals("Path for event is wrong.", path, events[i].getPath());
}
}
}
}