Author: mreutegg
Date: Wed Dec 12 15:34:06 2018
New Revision: 1848769
URL: http://svn.apache.org/viewvc?rev=1848769&view=rev
Log:
OAK-7956: Conflict may leave behind _collisions entry
Modified:
jackrabbit/oak/trunk/oak-store-document/src/main/java/org/apache/jackrabbit/oak/plugins/document/DocumentNodeStore.java
jackrabbit/oak/trunk/oak-store-document/src/test/java/org/apache/jackrabbit/oak/plugins/document/CollisionCleanupTest.java
Modified:
jackrabbit/oak/trunk/oak-store-document/src/main/java/org/apache/jackrabbit/oak/plugins/document/DocumentNodeStore.java
URL:
http://svn.apache.org/viewvc/jackrabbit/oak/trunk/oak-store-document/src/main/java/org/apache/jackrabbit/oak/plugins/document/DocumentNodeStore.java?rev=1848769&r1=1848768&r2=1848769&view=diff
==============================================================================
---
jackrabbit/oak/trunk/oak-store-document/src/main/java/org/apache/jackrabbit/oak/plugins/document/DocumentNodeStore.java
(original)
+++
jackrabbit/oak/trunk/oak-store-document/src/main/java/org/apache/jackrabbit/oak/plugins/document/DocumentNodeStore.java
Wed Dec 12 15:34:06 2018
@@ -191,6 +191,11 @@ public final class DocumentNodeStore
private int journalPushThreshold =
Integer.getInteger("oak.journalPushThreshold", 100000);
/**
+ * How many collision entries to collect in a single call.
+ */
+ private int collisionGarbageBatchSize =
Integer.getInteger("oak.documentMK.collisionGarbageBatchSize", 1000);
+
+ /**
* The document store without potentially lease checking wrapper.
*/
private final DocumentStore nonLeaseCheckingStore;
@@ -2025,7 +2030,7 @@ public final class DocumentNodeStore
long time = start;
// clean orphaned branches and collisions
cleanOrphanedBranches();
- cleanCollisions();
+ cleanRootCollisions();
long cleanTime = clock.getTime() - time;
time = clock.getTime();
// split documents (does not create new revisions)
@@ -2285,16 +2290,19 @@ public final class DocumentNodeStore
store.findAndUpdate(NODES, op);
}
}
-
- private void cleanCollisions() {
+
+ private void cleanRootCollisions() {
String id = Utils.getIdFromPath("/");
NodeDocument root = store.find(NODES, id);
- if (root == null) {
- return;
+ if (root != null) {
+ cleanCollisions(root, Integer.MAX_VALUE);
}
+ }
+
+ private void cleanCollisions(NodeDocument doc, int limit) {
RevisionVector head = getHeadRevision();
- Map<Revision, String> map = root.getLocalMap(NodeDocument.COLLISIONS);
- UpdateOp op = new UpdateOp(id, false);
+ Map<Revision, String> map = doc.getLocalMap(NodeDocument.COLLISIONS);
+ UpdateOp op = new UpdateOp(doc.getId(), false);
for (Revision r : map.keySet()) {
if (r.getClusterId() == clusterId) {
// remove collision if there is no active branch with
@@ -2304,11 +2312,15 @@ public final class DocumentNodeStore
if (branches.getBranchCommit(r) == null
&& !head.isRevisionNewer(r)) {
NodeDocument.removeCollision(op, r);
+ if (--limit <= 0) {
+ break;
+ }
}
}
}
if (op.hasChanges()) {
- LOG.debug("Removing collisions {}", op.getChanges().keySet());
+ LOG.debug("Removing collisions {} on {}",
+ op.getChanges().keySet(), doc.getId());
store.findAndUpdate(NODES, op);
}
}
@@ -2321,6 +2333,7 @@ public final class DocumentNodeStore
if (doc == null) {
continue;
}
+ cleanCollisions(doc, collisionGarbageBatchSize);
for (UpdateOp op : doc.split(this, head, binarySize)) {
NodeDocument before = null;
if (!op.isNew() ||
Modified:
jackrabbit/oak/trunk/oak-store-document/src/test/java/org/apache/jackrabbit/oak/plugins/document/CollisionCleanupTest.java
URL:
http://svn.apache.org/viewvc/jackrabbit/oak/trunk/oak-store-document/src/test/java/org/apache/jackrabbit/oak/plugins/document/CollisionCleanupTest.java?rev=1848769&r1=1848768&r2=1848769&view=diff
==============================================================================
---
jackrabbit/oak/trunk/oak-store-document/src/test/java/org/apache/jackrabbit/oak/plugins/document/CollisionCleanupTest.java
(original)
+++
jackrabbit/oak/trunk/oak-store-document/src/test/java/org/apache/jackrabbit/oak/plugins/document/CollisionCleanupTest.java
Wed Dec 12 15:34:06 2018
@@ -30,7 +30,6 @@ import org.apache.jackrabbit.oak.plugins
import org.apache.jackrabbit.oak.spi.state.NodeBuilder;
import org.junit.After;
import org.junit.Before;
-import org.junit.Ignore;
import org.junit.Rule;
import org.junit.Test;
import org.slf4j.Logger;
@@ -40,11 +39,13 @@ import static org.apache.jackrabbit.oak.
import static
org.apache.jackrabbit.oak.plugins.document.NodeDocument.COLLISIONS;
import static org.apache.jackrabbit.oak.plugins.document.TestUtils.merge;
import static org.hamcrest.Matchers.empty;
+import static org.hamcrest.Matchers.hasSize;
+import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertNotNull;
import static org.junit.Assert.assertThat;
+import static org.junit.Assert.fail;
// OAK-7956
-@Ignore("OAK-7956")
public class CollisionCleanupTest {
private static final Logger LOG =
LoggerFactory.getLogger(CollisionCleanupTest.class);
@@ -84,16 +85,83 @@ public class CollisionCleanupTest {
executor.invokeAll(tasks);
String id = Utils.getIdFromPath("/test");
+ ns1.addSplitCandidate(id);
+ ns1.runBackgroundOperations();
+ ns2.addSplitCandidate(id);
+ ns2.runBackgroundOperations();
+
DocumentStore store = ns1.getDocumentStore();
- store.invalidateCache(NODES, id);
NodeDocument doc = store.find(NODES, id);
assertNotNull(doc);
assertThat(doc.getValueMap(COLLISIONS).keySet(), empty());
}
+ @Test
+ public void batchCleanup() throws Exception {
+ Revision r = ns1.newRevision();
+ NodeBuilder b1 = ns1.getRoot().builder();
+ b1.child("test");
+ merge(ns1, b1);
+
+ String id = Utils.getIdFromPath("/test");
+ Revision other = new Revision(r.getTimestamp(), r.getCounter(),
r.getClusterId() + 1);
+ // add lots of old collisions
+ UpdateOp op = new UpdateOp(id, false);
+ for (int i = 0; i < 5000; i++) {
+ NodeDocument.addCollision(op, r, other);
+ r = new Revision(r.getTimestamp() - 1, 0, r.getClusterId());
+ }
+ NodeDocument doc = ns1.getDocumentStore().findAndUpdate(NODES, op);
+ assertNotNull(doc);
+
+ for (int i = 1; i <= 5; i++) {
+ // each background operation run will clean up 1000 collision
entries
+ ns1.addSplitCandidate(id);
+ ns1.runBackgroundOperations();
+
+ doc = ns1.getDocumentStore().find(NODES, id);
+ assertNotNull(doc);
+ assertEquals(5000 - i * 1000, doc.getLocalMap(COLLISIONS).size());
+ }
+ }
+
+ @Test
+ public void branchCollision() throws Exception {
+ NodeBuilder builder = ns1.getRoot().builder();
+ builder.child("conflict");
+ // trigger a branch commit
+ for (int i = 0; i < 200; i++) {
+ builder.child("n-" + i).setProperty("p", "v");
+ }
+
+ // this one wins and will create a collision marker
+ NodeBuilder b2 = ns1.getRoot().builder();
+ b2.child("conflict");
+ merge(ns1, b2);
+
+ NodeDocument doc = Utils.getRootDocument(ns1.getDocumentStore());
+ assertThat(doc.getLocalMap(COLLISIONS).keySet(), hasSize(1));
+
+ // must not clean up marker
+ ns1.addSplitCandidate(Utils.getIdFromPath("/"));
+ ns1.runBackgroundOperations();
+
+ doc = Utils.getRootDocument(ns1.getDocumentStore());
+ assertThat(doc.getLocalMap(COLLISIONS).keySet(), hasSize(1));
+
+ // must not be able to merge
+ try {
+ merge(ns1, builder);
+ fail("CommitFailedException expected");
+ } catch (CommitFailedException e) {
+ // expected
+ }
+ }
+
private DocumentNodeStore newDocumentNodeStore(int clusterId) {
DocumentNodeStore ns = builderProvider.newBuilder()
.setClusterId(clusterId).setAsyncDelay(0)
+ .setUpdateLimit(100)
.setDocumentStore(store).build();
// do not retry on conflicts
ns.setMaxBackOffMillis(0);