/*
* 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.core.integration.benchmark;
import java.io.File;
import java.util.Arrays;
import java.util.concurrent.atomic.AtomicLong;
import javax.jcr.ItemVisitor;
import javax.jcr.Node;
import javax.jcr.Property;
import javax.jcr.Repository;
import javax.jcr.RepositoryException;
import javax.jcr.Session;
import javax.jcr.SimpleCredentials;
import junit.framework.TestCase;
import org.apache.jackrabbit.commons.JcrUtils;
import org.apache.jackrabbit.core.TransientRepository;
/**
* Test case for
* <a href="https://issues.apache.org/jira/browse/JCR-2546">JCR-2546</a>.
* Note that this test takes a long time to finish and does not contain
* normal assertions, so it should only be invoked explicitly instead of
* being included in the normal test suite.
*/
public class ItemStateCacheSyncTest extends TestCase {
private File directory;
private Repository repository;
private Session session;
private Node root;
private volatile boolean run;
private AtomicLong counter = new AtomicLong();
protected void setUp() throws Exception {
directory = new File("target", "jackrabbit-sync-test-repo");
repository = new TransientRepository(directory);
session = repository.login(
new SimpleCredentials("admin", "admin".toCharArray()));
// Add a tree of 100k medium-sized (~10kB) nodes
String[] data = new String[1000];
Arrays.fill(data, "something");
root = session.getRootNode();
for (int i = 0; i < 1000; i++) {
Node a = root.addNode("a" + i);
for (int j = 0; j < 100; j++) {
Node b = a.addNode("b" + j);
b.setProperty("data", data);
}
session.save();
System.out.println((i + 1) * 1000 + " nodes created");
}
}
protected void tearDown() throws Exception {
for (int i = 0; i < 1000; i++) {
root.getNode("a" + i).remove();
session.save();
}
session.logout();
}
public void testCacheSync() throws Exception {
run = true;
Thread[] threads = new Thread[30];
for (int i = 0; i < threads.length; i++) {
threads[i] = new Thread(new Lookup());
threads[i].start();
}
// time for all threads to start and caches to warm up
Thread.sleep(3000);
long start, stop, count;
counter.set(0);
start = System.currentTimeMillis();
Thread.sleep(10000);
count = counter.get();
stop = System.currentTimeMillis();
System.out.println(
count * 1000 / (stop - start) + " lookups per second");
counter.set(0);
start = System.currentTimeMillis();
int i = 0;
do {
Node node = root.getNode("a" + (i++) % 1000);
for (Node ignore : JcrUtils.getChildNodes(node)) {
}
count = counter.get();
stop = System.currentTimeMillis();
} while (stop < start + 10000);
System.out.println(
count * 1000 / (stop - start) + " lookups per second"
+ " while traversing " + (i * 1000 * 1000) / (stop - start)
+ " nodes per second");
run = false;
for (i = 0; i < threads.length; i++) {
threads[i].join();
}
}
public class Lookup implements Runnable {
public void run() {
try {
Session session = repository.login();
try {
while (run) {
for (int i = 0; i < 100; i++) {
if (session.nodeExists("/a" + i)) {
session.getNode("/a" + 1);
}
counter.incrementAndGet();
}
}
} finally {
session.logout();
}
} catch (RepositoryException e) {
throw new RuntimeException(e);
}
}
}
public class Traverse implements ItemVisitor {
public void visit(Property property) {
}
public void visit(Node node) throws RepositoryException {
for (Node child : JcrUtils.getChildNodes(node)) {
child.accept(this);
}
}
}
}