From 8210e08cc9b6563fc0c8e479ce1f26d5e7e8c81c Mon Sep 17 00:00:00 2001 From: Sasha Sheikin Date: Tue, 29 Oct 2024 00:27:28 +0100 Subject: [PATCH] Prevent ghost containers after retries MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit reportLeakedContainers adapted from https://github.com/trinodb/trino/pull/20297 https://github.com/trinodb/trino/pull/21280 Co-authored-by: Piotr Findeisen Co-authored-by: Jan Waś --- .../containers/GenericContainer.java | 1 + .../containers/GenericContainerTest.java | 52 +++++++++++++++++++ 2 files changed, 53 insertions(+) diff --git a/core/src/main/java/org/testcontainers/containers/GenericContainer.java b/core/src/main/java/org/testcontainers/containers/GenericContainer.java index 52dede8792e..b2436b77286 100644 --- a/core/src/main/java/org/testcontainers/containers/GenericContainer.java +++ b/core/src/main/java/org/testcontainers/containers/GenericContainer.java @@ -550,6 +550,7 @@ private void tryStart() { } else { logger().error("There are no stdout/stderr logs available for the failed container"); } + stop(); } throw new ContainerLaunchException("Could not create/start container", e); diff --git a/core/src/test/java/org/testcontainers/containers/GenericContainerTest.java b/core/src/test/java/org/testcontainers/containers/GenericContainerTest.java index 0ba06dd4ca8..362e478ef34 100644 --- a/core/src/test/java/org/testcontainers/containers/GenericContainerTest.java +++ b/core/src/test/java/org/testcontainers/containers/GenericContainerTest.java @@ -5,6 +5,7 @@ import com.github.dockerjava.api.DockerClient; import com.github.dockerjava.api.command.InspectContainerResponse; import com.github.dockerjava.api.command.InspectContainerResponse.ContainerState; +import com.github.dockerjava.api.model.Container; import com.github.dockerjava.api.model.ExposedPort; import com.github.dockerjava.api.model.Info; import com.github.dockerjava.api.model.Ports; @@ -28,20 +29,45 @@ import org.testcontainers.utility.DockerImageName; import org.testcontainers.utility.MountableFile; +import java.time.Duration; import java.util.Arrays; import java.util.List; import java.util.Map; +import java.util.Optional; import java.util.concurrent.TimeUnit; import java.util.function.Predicate; import java.util.stream.Collectors; +import static com.google.common.base.MoreObjects.toStringHelper; +import static com.google.common.collect.ImmutableList.toImmutableList; +import static java.lang.String.format; +import static java.util.Arrays.asList; +import static java.util.Collections.singletonMap; +import static java.util.stream.Collectors.joining; import static org.assertj.core.api.Assertions.assertThat; import static org.assertj.core.api.Assertions.assertThatThrownBy; import static org.assertj.core.api.Assertions.catchThrowable; import static org.assertj.core.api.Assumptions.assumeThat; +import static org.testcontainers.containers.wait.strategy.Wait.forLogMessage; public class GenericContainerTest { + @Test + public void testStartupTimeoutWithAttemptsNotLeakingContainers() + { + try ( + GenericContainer container = new GenericContainer<>(TestImages.TINY_IMAGE) + .withStartupAttempts(3) + .waitingFor(forLogMessage("this text does not exist in logs", 1) + .withStartupTimeout(Duration.ofMillis(1))) + .withCommand("tail", "-f", "/dev/null"); + ) { + assertThatThrownBy(container::start) + .hasStackTraceContaining("Retry limit hit with exception"); + } + assertThat(reportLeakedContainers()).isEmpty(); + } + @Test public void shouldReportOOMAfterWait() { Info info = DockerClientFactory.instance().client().infoCmd().exec(); @@ -273,6 +299,32 @@ public void shouldRespectWaitStrategy() { } } + private static Optional reportLeakedContainers() + { + @SuppressWarnings("resource") // Throws when close is attempted, as this is a global instance. + DockerClient dockerClient = DockerClientFactory.lazyClient(); + + List containers = dockerClient.listContainersCmd() + .withLabelFilter(singletonMap(DockerClientFactory.TESTCONTAINERS_SESSION_ID_LABEL, DockerClientFactory.SESSION_ID)) + // ignore status "exited" - for example, failed containers after using `withStartupAttempts()` + .withStatusFilter(asList("created", "restarting", "running", "paused")) + .exec() + .stream() + .collect(toImmutableList()); + + if (containers.isEmpty()) { + return Optional.empty(); + } + + return Optional.of(format("Leaked containers: %s", containers.stream() + .map(container -> toStringHelper("container") + .add("id", container.getId()) + .add("image", container.getImage()) + .add("imageId", container.getImageId()) + .toString()) + .collect(joining(", ", "[", "]")))); + } + static class NoopStartupCheckStrategy extends StartupCheckStrategy { @Override