From 3aa393c3be0d4900066bae0265e7982243fe7036 Mon Sep 17 00:00:00 2001 From: Andrew Azores Date: Mon, 27 Nov 2023 12:41:33 -0500 Subject: [PATCH] test(snapshot): handle deletion request redirects for cleanup (#185) --- src/test/java/itest/SnapshotTest.java | 292 ++++++------------ .../java/itest/bases/StandardSelfTest.java | 33 ++ .../util/ITestCleanupFailedException.java | 2 + 3 files changed, 137 insertions(+), 190 deletions(-) diff --git a/src/test/java/itest/SnapshotTest.java b/src/test/java/itest/SnapshotTest.java index b08a6e41d..1f9d46125 100644 --- a/src/test/java/itest/SnapshotTest.java +++ b/src/test/java/itest/SnapshotTest.java @@ -29,17 +29,12 @@ import io.vertx.core.json.JsonObject; import io.vertx.ext.web.handler.HttpException; import itest.bases.StandardSelfTest; -import itest.util.ITestCleanupFailedException; import org.hamcrest.MatcherAssert; import org.hamcrest.Matchers; import org.junit.jupiter.api.Assertions; -import org.junit.jupiter.api.MethodOrderer.OrderAnnotation; -import org.junit.jupiter.api.Order; import org.junit.jupiter.api.Test; -import org.junit.jupiter.api.TestMethodOrder; @QuarkusTest -@TestMethodOrder(OrderAnnotation.class) public class SnapshotTest extends StandardSelfTest { static final String TEST_RECORDING_NAME = "someRecording"; @@ -54,7 +49,6 @@ String v2RequestUrl() { } @Test - @Order(1) void testPostV1ShouldHandleEmptySnapshot() throws Exception { // precondition, there should be no recordings before we start CompletableFuture preListRespFuture = new CompletableFuture<>(); @@ -101,7 +95,6 @@ void testPostV1ShouldHandleEmptySnapshot() throws Exception { } @Test - @Order(2) void testPostV2ShouldHandleEmptySnapshot() throws Exception { // precondition, there should be no recordings before we start CompletableFuture preListRespFuture = new CompletableFuture<>(); @@ -148,95 +141,56 @@ void testPostV2ShouldHandleEmptySnapshot() throws Exception { } @Test - @Order(3) void testPostV1ShouldCreateSnapshot() throws Exception { CompletableFuture snapshotName = new CompletableFuture<>(); - try { - // Create a recording - MultiMap form = MultiMap.caseInsensitiveMultiMap(); - form.add("recordingName", TEST_RECORDING_NAME); - form.add("duration", "5"); - form.add("events", "template=ALL"); - webClient - .extensions() - .post(String.format("%s/recordings", v1RequestUrl()), true, form, 5); + // Create a recording + MultiMap form = MultiMap.caseInsensitiveMultiMap(); + form.add("recordingName", TEST_RECORDING_NAME); + form.add("duration", "5"); + form.add("events", "template=ALL"); + webClient.extensions().post(String.format("%s/recordings", v1RequestUrl()), true, form, 5); - // Create a snapshot recording of all events at that time - webClient - .post(String.format("%s/snapshot", v1RequestUrl())) - .basicAuthentication("user", "pass") - .send( - ar -> { - if (assertRequestStatus(ar, snapshotName)) { - MatcherAssert.assertThat( - ar.result().statusCode(), Matchers.equalTo(200)); - MatcherAssert.assertThat( - ar.result() - .getHeader(HttpHeaders.CONTENT_TYPE.toString()), - Matchers.equalTo("text/plain;charset=UTF-8")); - snapshotName.complete(ar.result().bodyAsString()); - } - }); - - MatcherAssert.assertThat( - snapshotName.get(REQUEST_TIMEOUT_SECONDS, TimeUnit.SECONDS), - Matchers.matchesPattern(SNAPSHOT_NAME_PATTERN)); - - } finally { - // Clean up recording and snapshot - CompletableFuture deleteRecordingResponse = new CompletableFuture<>(); - webClient - .delete(String.format("%s/recordings/%s", v1RequestUrl(), TEST_RECORDING_NAME)) - .basicAuthentication("user", "pass") - .send( - ar -> { - if (assertRequestStatus(ar, deleteRecordingResponse)) { - deleteRecordingResponse.complete( - ar.result().bodyAsJsonObject()); - } - }); - - try { - deleteRecordingResponse.get(REQUEST_TIMEOUT_SECONDS, TimeUnit.SECONDS); - } catch (InterruptedException | ExecutionException e) { - throw new ITestCleanupFailedException( - String.format("Failed to delete target recording %s", TEST_RECORDING_NAME), - e); - } - - CompletableFuture deleteSnapshotResponse = new CompletableFuture<>(); + // Create a snapshot recording of all events at that time + webClient + .post(String.format("%s/snapshot", v1RequestUrl())) + .basicAuthentication("user", "pass") + .send( + ar -> { + if (assertRequestStatus(ar, snapshotName)) { + MatcherAssert.assertThat( + ar.result().statusCode(), Matchers.equalTo(200)); + MatcherAssert.assertThat( + ar.result().getHeader(HttpHeaders.CONTENT_TYPE.toString()), + Matchers.equalTo("text/plain;charset=UTF-8")); + snapshotName.complete(ar.result().bodyAsString()); + } + }); - webClient - .delete( - String.format( - "%s/recordings/%s", - v1RequestUrl(), - snapshotName.get(REQUEST_TIMEOUT_SECONDS, TimeUnit.SECONDS))) - .basicAuthentication("user", "pass") - .send( - ar -> { - if (assertRequestStatus(ar, deleteSnapshotResponse)) { - deleteSnapshotResponse.complete(ar.result().bodyAsJsonObject()); - } - }); + MatcherAssert.assertThat( + snapshotName.get(REQUEST_TIMEOUT_SECONDS, TimeUnit.SECONDS), + Matchers.matchesPattern(SNAPSHOT_NAME_PATTERN)); - try { - deleteSnapshotResponse.get(REQUEST_TIMEOUT_SECONDS, TimeUnit.SECONDS); - } catch (InterruptedException | ExecutionException e) { - throw new ITestCleanupFailedException( + // Clean up recording and snapshot + webClient + .extensions() + .delete( + String.format("%s/recordings/%s", v1RequestUrl(), TEST_RECORDING_NAME), + true, + 5); + webClient + .extensions() + .delete( String.format( - "Failed to delete snapshot %s", + "%s/recordings/%s", + v1RequestUrl(), snapshotName.get(REQUEST_TIMEOUT_SECONDS, TimeUnit.SECONDS)), - e); - } - } + true, + 5); } @Test - @Order(4) void testPostV1SnapshotThrowsWithNonExistentTarget() throws Exception { - CompletableFuture snapshotResponse = new CompletableFuture<>(); webClient .post("/api/v1/targets/notFound%2F9000/snapshot") @@ -255,126 +209,84 @@ void testPostV1SnapshotThrowsWithNonExistentTarget() throws Exception { } @Test - @Order(5) void testPostV2ShouldCreateSnapshot() throws Exception { CompletableFuture snapshotName = new CompletableFuture<>(); - try { - // Create a recording - MultiMap form = MultiMap.caseInsensitiveMultiMap(); - form.add("recordingName", TEST_RECORDING_NAME); - form.add("duration", "5"); - form.add("events", "template=ALL"); - webClient - .extensions() - .post(String.format("%s/recordings", v1RequestUrl()), true, form, 5); - - // Create a snapshot recording of all events at that time - CompletableFuture createResponse = new CompletableFuture<>(); - webClient - .post(String.format("%s/snapshot", v2RequestUrl())) - .basicAuthentication("user", "pass") - .send( - ar -> { - if (assertRequestStatus(ar, createResponse)) { - MatcherAssert.assertThat( - ar.result().statusCode(), Matchers.equalTo(201)); - MatcherAssert.assertThat( - ar.result() - .getHeader(HttpHeaders.CONTENT_TYPE.toString()), - Matchers.equalTo("application/json;charset=UTF-8")); - createResponse.complete(ar.result().bodyAsJsonObject()); - } - }); - - snapshotName.complete( - createResponse - .get(REQUEST_TIMEOUT_SECONDS, TimeUnit.SECONDS) - .getJsonObject("data") - .getJsonObject("result") - .getString("name")); + // Create a recording + MultiMap form = MultiMap.caseInsensitiveMultiMap(); + form.add("recordingName", TEST_RECORDING_NAME); + form.add("duration", "5"); + form.add("events", "template=ALL"); + webClient.extensions().post(String.format("%s/recordings", v1RequestUrl()), true, form, 5); - JsonObject json = createResponse.get(REQUEST_TIMEOUT_SECONDS, TimeUnit.SECONDS); - - MatcherAssert.assertThat( - json.getJsonObject("meta"), - Matchers.equalTo( - new JsonObject( - Map.of("type", "application/json", "status", "Created")))); - MatcherAssert.assertThat(json.getMap(), Matchers.hasKey("data")); - MatcherAssert.assertThat( - json.getJsonObject("data").getMap(), Matchers.hasKey("result")); - JsonObject result = json.getJsonObject("data").getJsonObject("result"); - MatcherAssert.assertThat(result.getString("state"), Matchers.equalTo("STOPPED")); - MatcherAssert.assertThat( - result.getLong("startTime"), - Matchers.lessThanOrEqualTo(Instant.now().toEpochMilli())); - MatcherAssert.assertThat( - result.getString("name"), - Matchers.equalTo(snapshotName.get(REQUEST_TIMEOUT_SECONDS, TimeUnit.SECONDS))); - MatcherAssert.assertThat(result.getLong("id"), Matchers.greaterThan(0L)); - MatcherAssert.assertThat( - result.getString("downloadUrl"), - Matchers.equalTo("/api/v3/activedownload/" + result.getLong("id"))); - MatcherAssert.assertThat( - result.getString("reportUrl"), - Matchers.equalTo("/api/v3/targets/1/reports/" + result.getLong("remoteId"))); - MatcherAssert.assertThat(result.getLong("expiry"), Matchers.nullValue()); - - } finally { - // Clean up recording and snapshot - CompletableFuture deleteRecordingResponse = new CompletableFuture<>(); - webClient - .delete(String.format("%s/recordings/%s", v1RequestUrl(), TEST_RECORDING_NAME)) - .basicAuthentication("user", "pass") - .send( - ar -> { - if (assertRequestStatus(ar, deleteRecordingResponse)) { - deleteRecordingResponse.complete( - ar.result().bodyAsJsonObject()); - } - }); + // Create a snapshot recording of all events at that time + CompletableFuture createResponse = new CompletableFuture<>(); + webClient + .post(String.format("%s/snapshot", v2RequestUrl())) + .basicAuthentication("user", "pass") + .send( + ar -> { + if (assertRequestStatus(ar, createResponse)) { + MatcherAssert.assertThat( + ar.result().statusCode(), Matchers.equalTo(201)); + MatcherAssert.assertThat( + ar.result().getHeader(HttpHeaders.CONTENT_TYPE.toString()), + Matchers.equalTo("application/json;charset=UTF-8")); + createResponse.complete(ar.result().bodyAsJsonObject()); + } + }); - try { - deleteRecordingResponse.get(REQUEST_TIMEOUT_SECONDS, TimeUnit.SECONDS); - } catch (InterruptedException | ExecutionException e) { - throw new ITestCleanupFailedException( - String.format("Failed to delete target recording %s", TEST_RECORDING_NAME), - e); - } + snapshotName.complete( + createResponse + .get(REQUEST_TIMEOUT_SECONDS, TimeUnit.SECONDS) + .getJsonObject("data") + .getJsonObject("result") + .getString("name")); - CompletableFuture deleteSnapshotResponse = new CompletableFuture<>(); + JsonObject json = createResponse.get(REQUEST_TIMEOUT_SECONDS, TimeUnit.SECONDS); - webClient - .delete( - String.format( - "%s/recordings/%s", - v1RequestUrl(), - snapshotName.get(REQUEST_TIMEOUT_SECONDS, TimeUnit.SECONDS))) - .basicAuthentication("user", "pass") - .send( - ar -> { - if (assertRequestStatus(ar, deleteSnapshotResponse)) { - deleteSnapshotResponse.complete(ar.result().bodyAsJsonObject()); - } - }); + MatcherAssert.assertThat( + json.getJsonObject("meta"), + Matchers.equalTo( + new JsonObject(Map.of("type", "application/json", "status", "Created")))); + MatcherAssert.assertThat(json.getMap(), Matchers.hasKey("data")); + MatcherAssert.assertThat(json.getJsonObject("data").getMap(), Matchers.hasKey("result")); + JsonObject result = json.getJsonObject("data").getJsonObject("result"); + MatcherAssert.assertThat(result.getString("state"), Matchers.equalTo("STOPPED")); + MatcherAssert.assertThat( + result.getLong("startTime"), + Matchers.lessThanOrEqualTo(Instant.now().toEpochMilli())); + MatcherAssert.assertThat( + result.getString("name"), + Matchers.equalTo(snapshotName.get(REQUEST_TIMEOUT_SECONDS, TimeUnit.SECONDS))); + MatcherAssert.assertThat(result.getLong("id"), Matchers.greaterThan(0L)); + MatcherAssert.assertThat( + result.getString("downloadUrl"), + Matchers.equalTo("/api/v3/activedownload/" + result.getLong("id"))); + MatcherAssert.assertThat( + result.getString("reportUrl"), + Matchers.equalTo("/api/v3/targets/1/reports/" + result.getLong("remoteId"))); + MatcherAssert.assertThat(result.getLong("expiry"), Matchers.nullValue()); - try { - deleteSnapshotResponse.get(REQUEST_TIMEOUT_SECONDS, TimeUnit.SECONDS); - } catch (InterruptedException | ExecutionException e) { - throw new ITestCleanupFailedException( + webClient + .extensions() + .delete( + String.format("%s/recordings/%s", v1RequestUrl(), TEST_RECORDING_NAME), + true, + 5); + webClient + .extensions() + .delete( String.format( - "Failed to delete snapshot %s", + "%s/recordings/%s", + v1RequestUrl(), snapshotName.get(REQUEST_TIMEOUT_SECONDS, TimeUnit.SECONDS)), - e); - } - } + true, + 5); } @Test - @Order(6) void testPostV2SnapshotThrowsWithNonExistentTarget() throws Exception { - CompletableFuture snapshotName = new CompletableFuture<>(); webClient .post("/api/v2/targets/notFound:9000/snapshot") diff --git a/src/test/java/itest/bases/StandardSelfTest.java b/src/test/java/itest/bases/StandardSelfTest.java index 18f43f04f..ed6e6026d 100644 --- a/src/test/java/itest/bases/StandardSelfTest.java +++ b/src/test/java/itest/bases/StandardSelfTest.java @@ -42,6 +42,7 @@ import io.vertx.ext.web.client.HttpResponse; import io.vertx.ext.web.codec.BodyCodec; import io.vertx.ext.web.handler.HttpException; +import itest.util.ITestCleanupFailedException; import itest.util.Utils; import itest.util.Utils.TestWebClient; import jakarta.ws.rs.core.HttpHeaders; @@ -49,6 +50,7 @@ import org.apache.commons.lang3.tuple.Pair; import org.apache.http.client.utils.URLEncodedUtils; import org.jboss.logging.Logger; +import org.junit.jupiter.api.AfterAll; import org.junit.jupiter.api.BeforeAll; public abstract class StandardSelfTest { @@ -68,6 +70,37 @@ public static void waitForDiscovery() { waitForDiscovery(0); } + @BeforeAll + public static void assertPreconditions() throws Exception { + assertNoRecordings(); + } + + @AfterAll + public static void assertPostconditions() throws Exception { + assertNoRecordings(); + } + + public static void assertNoRecordings() throws Exception { + CompletableFuture listFuture = new CompletableFuture<>(); + webClient + .get( + String.format( + "/api/v1/targets/%s/recordings", + getSelfReferenceConnectUrlEncoded())) + .basicAuthentication("user", "pass") + .send( + ar -> { + if (assertRequestStatus(ar, listFuture)) { + listFuture.complete(ar.result().bodyAsJsonArray()); + } + }); + JsonArray listResp = listFuture.get(REQUEST_TIMEOUT_SECONDS, TimeUnit.SECONDS); + if (!listResp.isEmpty()) { + throw new ITestCleanupFailedException( + String.format("Unexpected recordings:\n%s", listResp.encodePrettily())); + } + } + // @AfterAll public static void deleteSelfCustomTarget() { if (StringUtils.isBlank(selfCustomTargetLocation)) { diff --git a/src/test/java/itest/util/ITestCleanupFailedException.java b/src/test/java/itest/util/ITestCleanupFailedException.java index 6ce206704..adc907f2b 100644 --- a/src/test/java/itest/util/ITestCleanupFailedException.java +++ b/src/test/java/itest/util/ITestCleanupFailedException.java @@ -17,6 +17,8 @@ public class ITestCleanupFailedException extends Exception { + public ITestCleanupFailedException() {} + public ITestCleanupFailedException(String reason) { super(reason); }