|
33 | 33 | import org.apache.lucene.index.NoMergePolicy;
|
34 | 34 | import org.apache.lucene.index.RandomIndexWriter;
|
35 | 35 | import org.apache.lucene.index.SegmentInfos;
|
| 36 | +import org.apache.lucene.index.SoftDeletesRetentionMergePolicy; |
36 | 37 | import org.apache.lucene.index.Term;
|
37 | 38 | import org.apache.lucene.search.IndexSearcher;
|
38 | 39 | import org.apache.lucene.search.MatchAllDocsQuery;
|
| 40 | +import org.apache.lucene.search.ScoreDoc; |
39 | 41 | import org.apache.lucene.search.TermQuery;
|
| 42 | +import org.apache.lucene.search.TopDocs; |
40 | 43 | import org.apache.lucene.search.Weight;
|
41 | 44 | import org.apache.lucene.store.Directory;
|
42 | 45 | import org.apache.lucene.store.MMapDirectory;
|
43 | 46 | import org.apache.lucene.store.MockDirectoryWrapper;
|
44 | 47 | import org.apache.lucene.util.Bits;
|
| 48 | +import org.elasticsearch.core.internal.io.IOUtils; |
45 | 49 | import org.elasticsearch.test.ESTestCase;
|
46 | 50 |
|
47 | 51 | import java.io.IOException;
|
| 52 | +import java.io.StringReader; |
48 | 53 | import java.util.ArrayList;
|
49 | 54 | import java.util.Collections;
|
50 | 55 | import java.util.HashSet;
|
|
53 | 58 | import java.util.concurrent.CountDownLatch;
|
54 | 59 | import java.util.concurrent.atomic.AtomicBoolean;
|
55 | 60 |
|
| 61 | +import static org.hamcrest.Matchers.equalTo; |
| 62 | + |
56 | 63 | public class LuceneTests extends ESTestCase {
|
57 | 64 | public void testWaitForIndex() throws Exception {
|
58 | 65 | final MockDirectoryWrapper dir = newMockDirectory();
|
@@ -406,4 +413,88 @@ public void testMMapHackSupported() throws Exception {
|
406 | 413 | // add assume's here if needed for certain platforms, but we should know if it does not work.
|
407 | 414 | assertTrue("MMapDirectory does not support unmapping: " + MMapDirectory.UNMAP_NOT_SUPPORTED_REASON, MMapDirectory.UNMAP_SUPPORTED);
|
408 | 415 | }
|
| 416 | + |
| 417 | + public void testWrapAllDocsLive() throws Exception { |
| 418 | + Directory dir = newDirectory(); |
| 419 | + IndexWriterConfig config = newIndexWriterConfig().setSoftDeletesField(Lucene.SOFT_DELETE_FIELD) |
| 420 | + .setMergePolicy(new SoftDeletesRetentionMergePolicy(Lucene.SOFT_DELETE_FIELD, MatchAllDocsQuery::new, newMergePolicy())); |
| 421 | + IndexWriter writer = new IndexWriter(dir, config); |
| 422 | + int numDocs = between(1, 10); |
| 423 | + Set<String> liveDocs = new HashSet<>(); |
| 424 | + for (int i = 0; i < numDocs; i++) { |
| 425 | + String id = Integer.toString(i); |
| 426 | + Document doc = new Document(); |
| 427 | + doc.add(new StringField("id", id, Store.YES)); |
| 428 | + writer.addDocument(doc); |
| 429 | + liveDocs.add(id); |
| 430 | + } |
| 431 | + for (int i = 0; i < numDocs; i++) { |
| 432 | + if (randomBoolean()) { |
| 433 | + String id = Integer.toString(i); |
| 434 | + Document doc = new Document(); |
| 435 | + doc.add(new StringField("id", "v2-" + id, Store.YES)); |
| 436 | + if (randomBoolean()) { |
| 437 | + doc.add(Lucene.newSoftDeleteField()); |
| 438 | + } |
| 439 | + writer.softUpdateDocument(new Term("id", id), doc, Lucene.newSoftDeleteField()); |
| 440 | + liveDocs.add("v2-" + id); |
| 441 | + } |
| 442 | + } |
| 443 | + try (DirectoryReader unwrapped = DirectoryReader.open(writer)) { |
| 444 | + DirectoryReader reader = Lucene.wrapAllDocsLive(unwrapped); |
| 445 | + assertThat(reader.numDocs(), equalTo(liveDocs.size())); |
| 446 | + IndexSearcher searcher = new IndexSearcher(reader); |
| 447 | + Set<String> actualDocs = new HashSet<>(); |
| 448 | + TopDocs topDocs = searcher.search(new MatchAllDocsQuery(), Integer.MAX_VALUE); |
| 449 | + for (ScoreDoc scoreDoc : topDocs.scoreDocs) { |
| 450 | + actualDocs.add(reader.document(scoreDoc.doc).get("id")); |
| 451 | + } |
| 452 | + assertThat(actualDocs, equalTo(liveDocs)); |
| 453 | + } |
| 454 | + IOUtils.close(writer, dir); |
| 455 | + } |
| 456 | + |
| 457 | + public void testWrapLiveDocsNotExposeAbortedDocuments() throws Exception { |
| 458 | + Directory dir = newDirectory(); |
| 459 | + IndexWriterConfig config = newIndexWriterConfig().setSoftDeletesField(Lucene.SOFT_DELETE_FIELD) |
| 460 | + .setMergePolicy(new SoftDeletesRetentionMergePolicy(Lucene.SOFT_DELETE_FIELD, MatchAllDocsQuery::new, newMergePolicy())); |
| 461 | + IndexWriter writer = new IndexWriter(dir, config); |
| 462 | + int numDocs = between(1, 10); |
| 463 | + List<String> liveDocs = new ArrayList<>(); |
| 464 | + for (int i = 0; i < numDocs; i++) { |
| 465 | + String id = Integer.toString(i); |
| 466 | + Document doc = new Document(); |
| 467 | + doc.add(new StringField("id", id, Store.YES)); |
| 468 | + if (randomBoolean()) { |
| 469 | + doc.add(Lucene.newSoftDeleteField()); |
| 470 | + } |
| 471 | + writer.addDocument(doc); |
| 472 | + liveDocs.add(id); |
| 473 | + } |
| 474 | + int abortedDocs = between(1, 10); |
| 475 | + for (int i = 0; i < abortedDocs; i++) { |
| 476 | + try { |
| 477 | + Document doc = new Document(); |
| 478 | + doc.add(new StringField("id", "aborted-" + i, Store.YES)); |
| 479 | + StringReader reader = new StringReader(""); |
| 480 | + doc.add(new TextField("other", reader)); |
| 481 | + reader.close(); // mark the indexing hit non-aborting error |
| 482 | + writer.addDocument(doc); |
| 483 | + fail("index should have failed"); |
| 484 | + } catch (Exception ignored) { } |
| 485 | + } |
| 486 | + try (DirectoryReader unwrapped = DirectoryReader.open(writer)) { |
| 487 | + DirectoryReader reader = Lucene.wrapAllDocsLive(unwrapped); |
| 488 | + assertThat(reader.maxDoc(), equalTo(numDocs + abortedDocs)); |
| 489 | + assertThat(reader.numDocs(), equalTo(liveDocs.size())); |
| 490 | + IndexSearcher searcher = new IndexSearcher(reader); |
| 491 | + List<String> actualDocs = new ArrayList<>(); |
| 492 | + TopDocs topDocs = searcher.search(new MatchAllDocsQuery(), Integer.MAX_VALUE); |
| 493 | + for (ScoreDoc scoreDoc : topDocs.scoreDocs) { |
| 494 | + actualDocs.add(reader.document(scoreDoc.doc).get("id")); |
| 495 | + } |
| 496 | + assertThat(actualDocs, equalTo(liveDocs)); |
| 497 | + } |
| 498 | + IOUtils.close(writer, dir); |
| 499 | + } |
409 | 500 | }
|
0 commit comments