From 6f4815dc28201cdd6252dc268a1cc6d751219117 Mon Sep 17 00:00:00 2001 From: Witalij Berdinskich Date: Tue, 12 Oct 2021 22:45:04 +0300 Subject: [PATCH 1/2] [mock] verifyTimes with RequestKey parameter --- mock/src/main/java/feign/mock/MockClient.java | 22 ++++++++---- .../test/java/feign/mock/MockClientTest.java | 35 +++++++++++++------ 2 files changed, 41 insertions(+), 16 deletions(-) diff --git a/mock/src/main/java/feign/mock/MockClient.java b/mock/src/main/java/feign/mock/MockClient.java index f7d084e47..5c8f636bb 100644 --- a/mock/src/main/java/feign/mock/MockClient.java +++ b/mock/src/main/java/feign/mock/MockClient.java @@ -1,5 +1,5 @@ /** - * Copyright 2012-2020 The Feign Authors + * Copyright 2012-2021 The Feign Authors * * Licensed 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 @@ -219,20 +219,31 @@ public MockClient noContent(HttpMethod method, String url) { } public Request verifyOne(HttpMethod method, String url) { - return verifyTimes(method, url, 1).get(0); + return verifyOne(RequestKey.builder(method, url).build()); } public List verifyTimes(final HttpMethod method, final String url, final int times) { + return verifyTimes(RequestKey.builder(method, url).build(), times); + } + + public void verifyNever(HttpMethod method, String url) { + verifyNever(RequestKey.builder(method, url).build()); + } + + public Request verifyOne(RequestKey requestKey) { + return verifyTimes(requestKey, 1).get(0); + } + + public List verifyTimes(final RequestKey requestKey, final int times) { if (times < 0) { throw new IllegalArgumentException("times must be a non negative number"); } if (times == 0) { - verifyNever(method, url); + verifyNever(requestKey); return Collections.emptyList(); } - RequestKey requestKey = RequestKey.builder(method, url).build(); if (!requests.containsKey(requestKey)) { throw new VerificationAssertionError("Wanted: '%s' but never invoked! Got: %s", requestKey, requests.keySet()); @@ -248,8 +259,7 @@ public List verifyTimes(final HttpMethod method, final String url, fina return result; } - public void verifyNever(HttpMethod method, String url) { - RequestKey requestKey = RequestKey.builder(method, url).build(); + public void verifyNever(RequestKey requestKey) { if (requests.containsKey(requestKey)) { throw new VerificationAssertionError("Do not wanted: '%s' but was invoked!", requestKey); } diff --git a/mock/src/test/java/feign/mock/MockClientTest.java b/mock/src/test/java/feign/mock/MockClientTest.java index c462a02cc..0b8abcad2 100644 --- a/mock/src/test/java/feign/mock/MockClientTest.java +++ b/mock/src/test/java/feign/mock/MockClientTest.java @@ -1,5 +1,5 @@ /** - * Copyright 2012-2020 The Feign Authors + * Copyright 2012-2021 The Feign Authors * * Licensed 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 @@ -26,16 +26,10 @@ import java.lang.reflect.Type; import java.util.List; import javax.net.ssl.HttpsURLConnection; +import feign.*; import org.hamcrest.Matchers; import org.junit.Before; import org.junit.Test; -import feign.Body; -import feign.Feign; -import feign.FeignException; -import feign.Param; -import feign.Request; -import feign.RequestLine; -import feign.Response; import feign.codec.DecodeException; import feign.codec.Decoder; import feign.gson.GsonDecoder; @@ -56,6 +50,7 @@ List contributors(@Param("client_id") String clientId, List patchContributors(@Param("owner") String owner, @Param("repo") String repo); @RequestLine("POST /repos/{owner}/{repo}/contributors") + @Headers({"Content-Type: application/json"}) @Body("%7B\"login\":\"{login}\",\"type\":\"{type}\"%7D") Contributor create(@Param("owner") String owner, @Param("repo") String repo, @@ -97,14 +92,21 @@ public Object decode(Response response, Type type) public void setup() throws IOException { try (InputStream input = getClass().getResourceAsStream("/fixtures/contributors.json")) { byte[] data = toByteArray(input); + RequestKey postContributorKey = + RequestKey.builder(HttpMethod.POST, "/repos/netflix/feign/contributors") + .headers(RequestHeaders.builder() + .add("Content-Length", "55") + .add("Content-Type", "application/json") + .build()) + .body("{\"login\":\"velo_at_github\",\"type\":\"preposterous hacker\"}") + .build(); mockClient = new MockClient(); github = Feign.builder().decoder(new AssertionDecoder(new GsonDecoder())) .client(mockClient.ok(HttpMethod.GET, "/repos/netflix/feign/contributors", data) .ok(HttpMethod.GET, "/repos/netflix/feign/contributors?client_id=55") .ok(HttpMethod.GET, "/repos/netflix/feign/contributors?client_id=7 7", new ByteArrayInputStream(data)) - .ok(HttpMethod.POST, "/repos/netflix/feign/contributors", - "{\"login\":\"velo\",\"contributions\":0}") + .ok(postContributorKey, "{\"login\":\"velo\",\"contributions\":0}") .noContent(HttpMethod.PATCH, "/repos/velo/feign-mock/contributors") .add(HttpMethod.GET, "/repos/netflix/feign/contributors?client_id=1234567890", HttpsURLConnection.HTTP_NOT_FOUND) @@ -154,6 +156,15 @@ public void paramsEncoding() { @Test public void verifyInvocation() { + RequestKey testRequestKey = + RequestKey.builder(HttpMethod.POST, "/repos/netflix/feign/contributors") + .headers(RequestHeaders.builder() + .add("Content-Length", "55") + .add("Content-Type", "application/json") + .build()) + .body("{\"login\":\"velo_at_github\",\"type\":\"preposterous hacker\"}") + .build(); + Contributor contribution = github.create("netflix", "feign", "velo_at_github", "preposterous hacker"); // making sure it received a proper response @@ -164,7 +175,11 @@ public void verifyInvocation() { List results = mockClient.verifyTimes(HttpMethod.POST, "/repos/netflix/feign/contributors", 1); assertThat(results, hasSize(1)); + results = mockClient.verifyTimes(testRequestKey, 1); + assertThat(results, hasSize(1)); + + assertThat(mockClient.verifyOne(testRequestKey).body(), notNullValue()); byte[] body = mockClient.verifyOne(HttpMethod.POST, "/repos/netflix/feign/contributors") .body(); assertThat(body, notNullValue()); From 89d7815a416f31d731e6238d5469376052bce5be Mon Sep 17 00:00:00 2001 From: Witalij Berdinskich Date: Wed, 13 Oct 2021 10:58:53 +0300 Subject: [PATCH 2/2] Re-implement verify* methods with RequestKey parameters and stricter equals check for headers and body --- mock/src/main/java/feign/mock/MockClient.java | 56 +++++++++++++++---- .../test/java/feign/mock/MockClientTest.java | 44 +++++++++++++++ 2 files changed, 88 insertions(+), 12 deletions(-) diff --git a/mock/src/main/java/feign/mock/MockClient.java b/mock/src/main/java/feign/mock/MockClient.java index 5c8f636bb..13ee540ed 100644 --- a/mock/src/main/java/feign/mock/MockClient.java +++ b/mock/src/main/java/feign/mock/MockClient.java @@ -219,31 +219,24 @@ public MockClient noContent(HttpMethod method, String url) { } public Request verifyOne(HttpMethod method, String url) { - return verifyOne(RequestKey.builder(method, url).build()); - } - - public List verifyTimes(final HttpMethod method, final String url, final int times) { - return verifyTimes(RequestKey.builder(method, url).build(), times); - } - - public void verifyNever(HttpMethod method, String url) { - verifyNever(RequestKey.builder(method, url).build()); + return verifyTimes(method, url, 1).get(0); } public Request verifyOne(RequestKey requestKey) { return verifyTimes(requestKey, 1).get(0); } - public List verifyTimes(final RequestKey requestKey, final int times) { + public List verifyTimes(final HttpMethod method, final String url, final int times) { if (times < 0) { throw new IllegalArgumentException("times must be a non negative number"); } if (times == 0) { - verifyNever(requestKey); + verifyNever(method, url); return Collections.emptyList(); } + RequestKey requestKey = RequestKey.builder(method, url).build(); if (!requests.containsKey(requestKey)) { throw new VerificationAssertionError("Wanted: '%s' but never invoked! Got: %s", requestKey, requests.keySet()); @@ -259,12 +252,51 @@ public List verifyTimes(final RequestKey requestKey, final int times) { return result; } - public void verifyNever(RequestKey requestKey) { + public List verifyTimes(RequestKey requestKey, final int times) { + if (times < 0) { + throw new IllegalArgumentException("times must be a non negative number"); + } + + if (times == 0) { + verifyNever(requestKey); + return Collections.emptyList(); + } + + List result = null; + for (Map.Entry> request : requests.entrySet()) { + if (request.getKey().equalsExtended(requestKey)) { + result = request.getValue(); + } + } + if (result == null) { + throw new VerificationAssertionError("Wanted: '%s' but never invoked! Got: %s", requestKey, + requests.keySet()); + } + + if (result.size() != times) { + throw new VerificationAssertionError("Wanted: '%s' to be invoked: '%s' times but got: '%s'!", + requestKey, + times, result.size()); + } + + return result; + } + + public void verifyNever(HttpMethod method, String url) { + RequestKey requestKey = RequestKey.builder(method, url).build(); if (requests.containsKey(requestKey)) { throw new VerificationAssertionError("Do not wanted: '%s' but was invoked!", requestKey); } } + public void verifyNever(RequestKey requestKey) { + for (RequestKey recorderRequestKey : requests.keySet()) { + if (recorderRequestKey.equalsExtended(requestKey)) { + throw new VerificationAssertionError("Do not wanted: '%s' but was invoked!", requestKey); + } + } + } + /** * To be called in an @After method: * diff --git a/mock/src/test/java/feign/mock/MockClientTest.java b/mock/src/test/java/feign/mock/MockClientTest.java index 0b8abcad2..3ea208eea 100644 --- a/mock/src/test/java/feign/mock/MockClientTest.java +++ b/mock/src/test/java/feign/mock/MockClientTest.java @@ -13,6 +13,7 @@ */ package feign.mock; +import static feign.Util.UTF_8; import static feign.Util.toByteArray; import static org.hamcrest.MatcherAssert.assertThat; import static org.hamcrest.Matchers.containsString; @@ -94,6 +95,7 @@ public void setup() throws IOException { byte[] data = toByteArray(input); RequestKey postContributorKey = RequestKey.builder(HttpMethod.POST, "/repos/netflix/feign/contributors") + .charset(UTF_8) .headers(RequestHeaders.builder() .add("Content-Length", "55") .add("Content-Type", "application/json") @@ -193,9 +195,51 @@ public void verifyInvocation() { @Test public void verifyNone() { + RequestKey testRequestKey; github.create("netflix", "feign", "velo_at_github", "preposterous hacker"); mockClient.verifyTimes(HttpMethod.POST, "/repos/netflix/feign/contributors", 1); + testRequestKey = + RequestKey.builder(HttpMethod.POST, "/repos/netflix/feign/contributors") + .charset(UTF_8) + .headers(RequestHeaders.builder() + .add("Content-Length", "55") + .add("Content-Type", "application/json") + .build()) + // body is not equal + .body("{\"login\":\"velo[at]github\",\"type\":\"preposterous hacker\"}") + .build(); + try { + mockClient.verifyOne(testRequestKey); + fail(); + } catch (VerificationAssertionError e) { + assertThat(e.getMessage(), containsString("Wanted")); + assertThat(e.getMessage(), containsString("POST")); + assertThat(e.getMessage(), containsString("/repos/netflix/feign/contributors")); + } + mockClient.verifyNever(testRequestKey); + + testRequestKey = + RequestKey.builder(HttpMethod.POST, "/repos/netflix/feign/contributors") + .charset(UTF_8) + .headers(RequestHeaders.builder() + .add("Content-Length", "55") + .add("Content-Type", "application/json") + // headers are not equal + .add("X-Header", "qwerty") + .build()) + .body("{\"login\":\"velo_at_github\",\"type\":\"preposterous hacker\"}") + .build(); + try { + mockClient.verifyOne(testRequestKey); + fail(); + } catch (VerificationAssertionError e) { + assertThat(e.getMessage(), containsString("Wanted")); + assertThat(e.getMessage(), containsString("POST")); + assertThat(e.getMessage(), containsString("/repos/netflix/feign/contributors")); + } + mockClient.verifyNever(testRequestKey); + try { mockClient.verifyTimes(HttpMethod.POST, "/repos/netflix/feign/contributors", 0); fail();