+++ /dev/null
-package org.apache.lucene.index;
-
-/**
- * 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.
- */
-
-import java.io.IOException;
-
-import org.apache.lucene.analysis.MockAnalyzer;
-import org.apache.lucene.document.Document;
-import org.apache.lucene.document.Field;
-import org.apache.lucene.store.Directory;
-import org.apache.lucene.store.MockDirectoryWrapper;
-import org.apache.lucene.store.RAMDirectory;
-import org.apache.lucene.util.English;
-import org.apache.lucene.util.LuceneTestCase;
-
-public class TestTransactions extends LuceneTestCase {
-
- private static volatile boolean doFail;
-
- private class RandomFailure extends MockDirectoryWrapper.Failure {
- @Override
- public void eval(MockDirectoryWrapper dir) throws IOException {
- if (TestTransactions.doFail && random.nextInt() % 10 <= 3)
- throw new IOException("now failing randomly but on purpose");
- }
- }
-
- private static abstract class TimedThread extends Thread {
- volatile boolean failed;
- private static float RUN_TIME_MSEC = atLeast(500);
- private TimedThread[] allThreads;
-
- abstract public void doWork() throws Throwable;
-
- TimedThread(TimedThread[] threads) {
- this.allThreads = threads;
- }
-
- @Override
- public void run() {
- final long stopTime = System.currentTimeMillis() + (long) (RUN_TIME_MSEC);
-
- try {
- do {
- if (anyErrors()) break;
- doWork();
- } while (System.currentTimeMillis() < stopTime);
- } catch (Throwable e) {
- System.out.println(Thread.currentThread() + ": exc");
- e.printStackTrace(System.out);
- failed = true;
- }
- }
-
- private boolean anyErrors() {
- for(int i=0;i<allThreads.length;i++)
- if (allThreads[i] != null && allThreads[i].failed)
- return true;
- return false;
- }
- }
-
- private class IndexerThread extends TimedThread {
- Directory dir1;
- Directory dir2;
- Object lock;
- int nextID;
-
- public IndexerThread(Object lock, Directory dir1, Directory dir2, TimedThread[] threads) {
- super(threads);
- this.lock = lock;
- this.dir1 = dir1;
- this.dir2 = dir2;
- }
-
- @Override
- public void doWork() throws Throwable {
-
- IndexWriter writer1 = new IndexWriter(
- dir1,
- newIndexWriterConfig(TEST_VERSION_CURRENT, new MockAnalyzer(random)).
- setMaxBufferedDocs(3).
- setMergeScheduler(new ConcurrentMergeScheduler()).
- setMergePolicy(newLogMergePolicy(2))
- );
- ((ConcurrentMergeScheduler) writer1.getConfig().getMergeScheduler()).setSuppressExceptions();
-
- // Intentionally use different params so flush/merge
- // happen @ different times
- IndexWriter writer2 = new IndexWriter(
- dir2,
- newIndexWriterConfig( TEST_VERSION_CURRENT, new MockAnalyzer(random)).
- setMaxBufferedDocs(2).
- setMergeScheduler(new ConcurrentMergeScheduler()).
- setMergePolicy(newLogMergePolicy(3))
- );
- ((ConcurrentMergeScheduler) writer2.getConfig().getMergeScheduler()).setSuppressExceptions();
-
- update(writer1);
- update(writer2);
-
- TestTransactions.doFail = true;
- try {
- synchronized(lock) {
- try {
- writer1.prepareCommit();
- } catch (Throwable t) {
- writer1.rollback();
- writer2.rollback();
- return;
- }
- try {
- writer2.prepareCommit();
- } catch (Throwable t) {
- writer1.rollback();
- writer2.rollback();
- return;
- }
-
- writer1.commit();
- writer2.commit();
- }
- } finally {
- TestTransactions.doFail = false;
- }
-
- writer1.close();
- writer2.close();
- }
-
- public void update(IndexWriter writer) throws IOException {
- // Add 10 docs:
- for(int j=0; j<10; j++) {
- Document d = new Document();
- int n = random.nextInt();
- d.add(newField("id", Integer.toString(nextID++), Field.Store.YES, Field.Index.NOT_ANALYZED));
- d.add(newField("contents", English.intToEnglish(n), Field.Store.NO, Field.Index.ANALYZED));
- writer.addDocument(d);
- }
-
- // Delete 5 docs:
- int deleteID = nextID-1;
- for(int j=0; j<5; j++) {
- writer.deleteDocuments(new Term("id", ""+deleteID));
- deleteID -= 2;
- }
- }
- }
-
- private static class SearcherThread extends TimedThread {
- Directory dir1;
- Directory dir2;
- Object lock;
-
- public SearcherThread(Object lock, Directory dir1, Directory dir2, TimedThread[] threads) {
- super(threads);
- this.lock = lock;
- this.dir1 = dir1;
- this.dir2 = dir2;
- }
-
- @Override
- public void doWork() throws Throwable {
- IndexReader r1, r2;
- synchronized(lock) {
- r1 = IndexReader.open(dir1, true);
- r2 = IndexReader.open(dir2, true);
- }
- if (r1.numDocs() != r2.numDocs())
- throw new RuntimeException("doc counts differ: r1=" + r1.numDocs() + " r2=" + r2.numDocs());
- r1.close();
- r2.close();
- }
- }
-
- public void initIndex(Directory dir) throws Throwable {
- IndexWriter writer = new IndexWriter(dir, newIndexWriterConfig( TEST_VERSION_CURRENT, new MockAnalyzer(random)));
- for(int j=0; j<7; j++) {
- Document d = new Document();
- int n = random.nextInt();
- d.add(newField("contents", English.intToEnglish(n), Field.Store.NO, Field.Index.ANALYZED));
- writer.addDocument(d);
- }
- writer.close();
- }
-
- public void testTransactions() throws Throwable {
- // we cant use non-ramdir on windows, because this test needs to double-write.
- MockDirectoryWrapper dir1 = new MockDirectoryWrapper(random, new RAMDirectory());
- MockDirectoryWrapper dir2 = new MockDirectoryWrapper(random, new RAMDirectory());
- dir1.setPreventDoubleWrite(false);
- dir2.setPreventDoubleWrite(false);
- dir1.failOn(new RandomFailure());
- dir2.failOn(new RandomFailure());
-
- initIndex(dir1);
- initIndex(dir2);
-
- TimedThread[] threads = new TimedThread[3];
- int numThread = 0;
-
- IndexerThread indexerThread = new IndexerThread(this, dir1, dir2, threads);
- threads[numThread++] = indexerThread;
- indexerThread.start();
-
- SearcherThread searcherThread1 = new SearcherThread(this, dir1, dir2, threads);
- threads[numThread++] = searcherThread1;
- searcherThread1.start();
-
- SearcherThread searcherThread2 = new SearcherThread(this, dir1, dir2, threads);
- threads[numThread++] = searcherThread2;
- searcherThread2.start();
-
- for(int i=0;i<numThread;i++)
- threads[i].join();
-
- for(int i=0;i<numThread;i++)
- assertTrue(!threads[i].failed);
- dir1.close();
- dir2.close();
- }
-}