From b0910bd8b43927f9dbac1e1dcf1f707e252c55a8 Mon Sep 17 00:00:00 2001 From: Lee Hinman Date: Thu, 25 Jun 2020 13:33:56 -0600 Subject: [PATCH] Fix negative limiting with fewer PARTIAL snapshots than minimum required (#58563) In SLM retention, when a minimum number of snapshots is required for retention, we prefer to remove the oldest snapshots first. To perform this, we limit one of the streams, in a rare case this can cause: ``` [mynode] error during snapshot retention task java.lang.IllegalArgumentException: -5 at java.util.stream.ReferencePipeline.limit(ReferencePipeline.java:469) ~[?:?] at org.elasticsearch.xpack.core.slm.SnapshotRetentionConfiguration.lambda$getSnapshotDeletionPredicate$6(SnapshotRetentionConfiguration.java:195) ~[?:?] at org.elasticsearch.xpack.slm.SnapshotRetentionTask.snapshotEligibleForDeletion(SnapshotRetentionTask.java:245) ~[?:?] at org.elasticsearch.xpack.slm.SnapshotRetentionTask$1.lambda$onResponse$0(SnapshotRetentionTask.java:163) ~[?:?] at java.util.stream.ReferencePipeline$2$1.accept(ReferencePipeline.java:176) ~[?:?] at java.util.ArrayList$ArrayListSpliterator.forEachRemaining(ArrayList.java:1624) ~[?:?] at java.util.stream.AbstractPipeline.copyInto(AbstractPipeline.java:484) ~[?:?] at java.util.stream.AbstractPipeline.wrapAndCopyInto(AbstractPipeline.java:474) ~[?:?] at java.util.stream.ReduceOps$ReduceOp.evaluateSequential(ReduceOps.java:913) ~[?:?] ``` When certain criteria are met. This commit fixes the negative limiting with `Math.max(0, ...)` and adds a unit test for the behavior. Resolves #58515 --- .../core/slm/SnapshotRetentionConfiguration.java | 2 +- .../slm/SnapshotRetentionConfigurationTests.java | 13 ++++++++++++- 2 files changed, 13 insertions(+), 2 deletions(-) diff --git a/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/slm/SnapshotRetentionConfiguration.java b/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/slm/SnapshotRetentionConfiguration.java index cb21bc1c02d5d..a6265cecb44a1 100644 --- a/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/slm/SnapshotRetentionConfiguration.java +++ b/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/slm/SnapshotRetentionConfiguration.java @@ -184,7 +184,7 @@ public Predicate getSnapshotDeletionPredicate(final List 1, null, 2, 2); @@ -243,6 +242,18 @@ public void testMostRecentSuccessfulTimestampIsUsed() { assertThat(conf.getSnapshotDeletionPredicate(infos).test(s4), equalTo(false)); } + public void testFewerSuccessesThanMinWithPartial() { + SnapshotRetentionConfiguration conf = new SnapshotRetentionConfiguration(() -> 1, TimeValue.timeValueSeconds(5), 10, 20); + SnapshotInfo s1 = makeInfo(1); + SnapshotInfo sP = makePartialInfo(2); + SnapshotInfo s2 = makeInfo(3); + + List infos = Arrays.asList(s1 , sP, s2); + assertThat(conf.getSnapshotDeletionPredicate(infos).test(s1), equalTo(false)); + assertThat(conf.getSnapshotDeletionPredicate(infos).test(sP), equalTo(false)); + assertThat(conf.getSnapshotDeletionPredicate(infos).test(s2), equalTo(false)); + } + private SnapshotInfo makeInfo(long startTime) { final Map meta = new HashMap<>(); meta.put(SnapshotLifecyclePolicy.POLICY_ID_METADATA_FIELD, REPO);