blob: 93a9b17e0bde438580422149c3f655ce99fe020f [file] [log] [blame]
/*
* 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.lucene.index;
import java.io.IOException;
import java.nio.charset.StandardCharsets;
import java.util.Collections;
import java.util.LinkedList;
import java.util.List;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicInteger;
import org.apache.lucene.store.Directory;
import org.apache.lucene.util.LuceneTestCase;
import org.apache.lucene.util.StringHelper;
import org.apache.lucene.util.TestUtil;
import org.apache.lucene.util.Version;
public class TestMergePolicy extends LuceneTestCase {
public void testWaitForOneMerge() throws IOException, InterruptedException {
try (Directory dir = newDirectory()) {
MergePolicy.MergeSpecification ms = createRandomMergeSpecification(dir, 1 + random().nextInt(10));
for (MergePolicy.OneMerge m : ms.merges) {
assertFalse(m.hasCompletedSuccessfully().isPresent());
}
Thread t = new Thread(() -> {
try {
for (MergePolicy.OneMerge m : ms.merges) {
m.close(true, false, mr -> {});
}
} catch (IOException e) {
throw new AssertionError(e);
}
});
t.start();
assertTrue(ms.await(100, TimeUnit.HOURS));
for (MergePolicy.OneMerge m : ms.merges) {
assertTrue(m.hasCompletedSuccessfully().get());
}
t.join();
}
}
public void testTimeout() throws IOException, InterruptedException {
try (Directory dir = newDirectory()) {
MergePolicy.MergeSpecification ms = createRandomMergeSpecification(dir, 3);
for (MergePolicy.OneMerge m : ms.merges) {
assertFalse(m.hasCompletedSuccessfully().isPresent());
}
Thread t = new Thread(() -> {
try {
ms.merges.get(0).close(true, false, mr -> {});
} catch (IOException e) {
throw new AssertionError(e);
}
});
t.start();
assertFalse(ms.await(10, TimeUnit.MILLISECONDS));
assertFalse(ms.merges.get(1).hasCompletedSuccessfully().isPresent());
t.join();
}
}
public void testTimeoutLargeNumberOfMerges() throws IOException, InterruptedException {
try (Directory dir = newDirectory()) {
MergePolicy.MergeSpecification ms = createRandomMergeSpecification(dir, 10000);
for (MergePolicy.OneMerge m : ms.merges) {
assertFalse(m.hasCompletedSuccessfully().isPresent());
}
AtomicInteger i = new AtomicInteger(0);
AtomicBoolean stop = new AtomicBoolean(false);
Thread t = new Thread(() -> {
while (stop.get() == false) {
try {
ms.merges.get(i.getAndIncrement()).close(true, false, mr -> {});
Thread.sleep(1);
} catch (IOException | InterruptedException e) {
throw new AssertionError(e);
}
}
});
t.start();
assertFalse(ms.await(10, TimeUnit.MILLISECONDS));
stop.set(true);
t.join();
for (int j = 0; j < ms.merges.size(); j++) {
if (j < i.get()) {
assertTrue(ms.merges.get(j).hasCompletedSuccessfully().get());
} else {
assertFalse(ms.merges.get(j).hasCompletedSuccessfully().isPresent());
}
}
}
}
public void testFinishTwice() throws IOException {
try (Directory dir = newDirectory()) {
MergePolicy.MergeSpecification spec = createRandomMergeSpecification(dir, 1);
MergePolicy.OneMerge oneMerge = spec.merges.get(0);
oneMerge.close(true, false, mr -> {});
expectThrows(IllegalStateException.class, () -> oneMerge.close(false, false, mr -> {}));
}
}
public void testTotalMaxDoc() throws IOException {
try (Directory dir = newDirectory()) {
MergePolicy.MergeSpecification spec = createRandomMergeSpecification(dir, 1);
int docs = 0;
MergePolicy.OneMerge oneMerge = spec.merges.get(0);
for (SegmentCommitInfo info : oneMerge.segments) {
docs += info.info.maxDoc();
}
assertEquals(docs, oneMerge.totalMaxDoc);
}
}
private static MergePolicy.MergeSpecification createRandomMergeSpecification(Directory dir, int numMerges) {
MergePolicy.MergeSpecification ms = new MergePolicy.MergeSpecification();
for (int ii = 0; ii < numMerges; ++ii) {
final SegmentInfo si = new SegmentInfo(
dir, // dir
Version.LATEST, // version
Version.LATEST, // min version
TestUtil.randomSimpleString(random()), // name
random().nextInt(1000), // maxDoc
random().nextBoolean(), // isCompoundFile
null, // codec
Collections.emptyMap(), // diagnostics
TestUtil.randomSimpleString(// id
random(),
StringHelper.ID_LENGTH,
StringHelper.ID_LENGTH).getBytes(StandardCharsets.US_ASCII),
Collections.emptyMap(), // attributes
null /* indexSort */);
final List<SegmentCommitInfo> segments = new LinkedList<SegmentCommitInfo>();
segments.add(new SegmentCommitInfo(si, 0, 0, 0, 0, 0, StringHelper.randomId()));
ms.add(new MergePolicy.OneMerge(segments));
}
return ms;
}
}