From 3f7e1501737330f0617d3684c72d774c7100553b Mon Sep 17 00:00:00 2001 From: Jens Wille Date: Fri, 26 Aug 2022 19:46:30 +0200 Subject: [PATCH 01/12] Extend `HttpOpener`. --- .../java/org/metafacture/io/HttpOpener.java | 141 ++++++++++++++++-- 1 file changed, 131 insertions(+), 10 deletions(-) diff --git a/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java b/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java index e69e8865c..22f3b2825 100644 --- a/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java +++ b/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java @@ -24,18 +24,21 @@ import org.metafacture.framework.annotations.Out; import org.metafacture.framework.helpers.DefaultObjectPipe; +import java.io.ByteArrayInputStream; import java.io.IOException; +import java.io.InputStream; import java.io.InputStreamReader; import java.io.Reader; +import java.io.SequenceInputStream; +import java.net.HttpURLConnection; import java.net.URL; -import java.net.URLConnection; import java.util.Arrays; import java.util.HashMap; import java.util.Map; import java.util.regex.Pattern; /** - * Opens a {@link URLConnection} and passes a reader to the receiver. + * Opens an {@link HttpURLConnection} and passes a reader to the receiver. * * @author Christoph Böhme * @author Jan Schnasse @@ -55,14 +58,50 @@ public final class HttpOpener extends DefaultObjectPipe headers = new HashMap<>(); + private Method method; + private String body; + private String errorPrefix; + private String url; + private boolean inputUsed; + + public enum Method { + + DELETE(false), + GET(false), + HEAD(false), + OPTIONS(false), + POST(true), + PUT(true), + TRACE(false); + + private final boolean inputAsBody; + + Method(final boolean inputAsBody) { + this.inputAsBody = inputAsBody; + } + + private boolean getInputAsBody() { + return inputAsBody; + } + + } + /** * Creates an instance of {@link HttpOpener}. */ public HttpOpener() { setAccept(ACCEPT_DEFAULT); setEncoding(ENCODING_DEFAULT); + setErrorPrefix(DEFAULT_PREFIX); + setMethod(DEFAULT_METHOD); + setUrl(INPUT_DESIGNATOR); } /** @@ -76,6 +115,15 @@ public void setAccept(final String accept) { setHeader(ACCEPT_HEADER, accept); } + /** + * Sets the HTTP request body. + * + * @param body the request body + */ + public void setBody(final String body) { + this.body = body; + } + /** * Sets the preferred encoding of the HTTP response. This value is in the * accept-charset header. Additonally, the encoding is used for reading the @@ -89,6 +137,15 @@ public void setEncoding(final String encoding) { setHeader(ENCODING_HEADER, encoding); } + /** + * Sets the error prefix. + * + * @param errorPrefix the error prefix + */ + public void setErrorPrefix(final String errorPrefix) { + this.errorPrefix = errorPrefix; + } + /** * Sets a request property, or multiple request properties separated by * {@code \n}. @@ -117,21 +174,85 @@ public void setHeader(final String key, final String value) { headers.put(key.toLowerCase(), value); } + /** + * Sets the HTTP request method. + * + * @param method the request method + */ + public void setMethod(final Method method) { + this.method = method; + } + + /** + * Sets the HTTP request URL. + * + * @param url the request URL + */ + public void setUrl(final String url) { + this.url = url; + } + @Override - public void process(final String urlStr) { + public void process(final String input) { try { - final URL url = new URL(urlStr); - final URLConnection con = url.openConnection(); - headers.forEach(con::addRequestProperty); - String enc = con.getContentEncoding(); - if (enc == null) { - enc = headers.get(ENCODING_HEADER); + final String requestUrl = getInput(input, url); + final String requestBody = getInput(input, + body == null && method.getInputAsBody() ? INPUT_DESIGNATOR : body); + + final HttpURLConnection connection = + (HttpURLConnection) new URL(requestUrl).openConnection(); + + connection.setRequestMethod(method.name()); + headers.forEach(connection::addRequestProperty); + + if (requestBody != null) { + connection.setDoOutput(true); + connection.getOutputStream().write(requestBody.getBytes()); + } + + final InputStream errorStream = connection.getErrorStream(); + final InputStream inputStream; + + if (errorStream != null) { + if (errorPrefix != null) { + final InputStream errorPrefixStream = new ByteArrayInputStream(errorPrefix.getBytes()); + inputStream = new SequenceInputStream(errorPrefixStream, errorStream); + } + else { + inputStream = errorStream; + } } - getReceiver().process(new InputStreamReader(con.getInputStream(), enc)); + else { + inputStream = connection.getInputStream(); + } + + final String contentEncoding = getEncoding(connection.getContentEncoding()); + getReceiver().process(new InputStreamReader(inputStream, contentEncoding)); } catch (final IOException e) { throw new MetafactureException(e); } } + private String getInput(final String input, final String value) { + final String result; + + if (!INPUT_DESIGNATOR.equals(value)) { + result = value; + } + else if (inputUsed) { + result = null; + } + else { + inputUsed = true; + result = input; + } + + return result; + } + + private String getEncoding(final String contentEncoding) { + return contentEncoding != null ? contentEncoding : headers.get(ENCODING_HEADER); + } + } From 28d9db880d92864a691786a75ff2a57ffc315344 Mon Sep 17 00:00:00 2001 From: Pascal Christoph Date: Mon, 29 Aug 2022 09:27:35 +0200 Subject: [PATCH 02/12] Add content type header (#460) --- .../java/org/metafacture/io/HttpOpener.java | 19 +++++++++++++++---- 1 file changed, 15 insertions(+), 4 deletions(-) diff --git a/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java b/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java index 22f3b2825..9205537d1 100644 --- a/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java +++ b/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java @@ -1,5 +1,5 @@ /* - * Copyright 2013, 2014 Deutsche Nationalbibliothek + * Copyright 2013, 2022 Deutsche Nationalbibliothek et al * * Licensed under the Apache License, Version 2.0 the "License"; * you may not use this file except in compliance with the License. @@ -54,14 +54,14 @@ public final class HttpOpener extends DefaultObjectPipe headers = new HashMap<>(); @@ -98,6 +98,7 @@ private boolean getInputAsBody() { */ public HttpOpener() { setAccept(ACCEPT_DEFAULT); + setContentType(CONTENT_TYPE_DEFAULT); setEncoding(ENCODING_DEFAULT); setErrorPrefix(DEFAULT_PREFIX); setMethod(DEFAULT_METHOD); @@ -124,6 +125,16 @@ public void setBody(final String body) { this.body = body; } + /** + * Sets the HTTP content type header. This is a mime-type such as text/plain, + * text/html. The default is application/json. + * + * @param contentType mime-type to use for the HTTP contentType header + */ + public void setContentType(final String contentType) { + setHeader(CONTENT_TYPE_HEADER, contentType); + } + /** * Sets the preferred encoding of the HTTP response. This value is in the * accept-charset header. Additonally, the encoding is used for reading the From 6db8a3e4c3c3ec8773d24ef68a4a465bae91eb75 Mon Sep 17 00:00:00 2001 From: Pascal Christoph Date: Mon, 29 Aug 2022 15:20:20 +0200 Subject: [PATCH 03/12] Reformat (#460) - add */* as content type default --- .../src/main/java/org/metafacture/io/HttpOpener.java | 11 +++++------ 1 file changed, 5 insertions(+), 6 deletions(-) diff --git a/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java b/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java index 9205537d1..542e3de4c 100644 --- a/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java +++ b/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java @@ -52,14 +52,13 @@ public final class HttpOpener extends DefaultObjectPipe Date: Mon, 29 Aug 2022 15:32:23 +0200 Subject: [PATCH 04/12] Improve documentation (#460) --- .../src/main/java/org/metafacture/io/HttpOpener.java | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java b/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java index 542e3de4c..0884bf575 100644 --- a/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java +++ b/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java @@ -42,8 +42,9 @@ * * @author Christoph Böhme * @author Jan Schnasse + * @author Jens Wille */ -@Description("Opens an HTTP resource. Supports the setting of `Accept` and `Accept-Charset` as HTTP header fields, as well as generic headers (separated by `\\n`).") +@Description("Opens an HTTP resource. Supports the setting of `Accept` and `Accept-Charset` as HTTP header fields, as well as generic headers (separated by `\\n`). Default setting of header `Accept` and `Content-Type` is '*/*' and `Encoding` is 'UTF-8'; default for 'method' is 'GET' and 'ErrorPrefix' is '@-'.") @In(String.class) @Out(Reader.class) @FluxCommand("open-http") From 2a5ef5f4cb533524da003f943acf738d0fb0013a Mon Sep 17 00:00:00 2001 From: Pascal Christoph Date: Mon, 29 Aug 2022 15:35:16 +0200 Subject: [PATCH 05/12] Add CONTENT_TYPE_DEFAULT (#460) --- .../src/main/java/org/metafacture/io/HttpOpener.java | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java b/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java index 0884bf575..ce9aeab8c 100644 --- a/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java +++ b/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java @@ -55,6 +55,7 @@ public final class HttpOpener extends DefaultObjectPipe Date: Mon, 29 Aug 2022 16:38:14 +0200 Subject: [PATCH 06/12] Remove CONTENT_TYPE_DEFAULT (#460) --- .../src/main/java/org/metafacture/io/HttpOpener.java | 4 +--- 1 file changed, 1 insertion(+), 3 deletions(-) diff --git a/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java b/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java index ce9aeab8c..27dae4f2e 100644 --- a/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java +++ b/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java @@ -44,7 +44,7 @@ * @author Jan Schnasse * @author Jens Wille */ -@Description("Opens an HTTP resource. Supports the setting of `Accept` and `Accept-Charset` as HTTP header fields, as well as generic headers (separated by `\\n`). Default setting of header `Accept` and `Content-Type` is '*/*' and `Encoding` is 'UTF-8'; default for 'method' is 'GET' and 'ErrorPrefix' is '@-'.") +@Description("Opens an HTTP resource. Supports the setting of `Accept` and `Accept-Charset` as HTTP header fields, as well as generic headers (separated by `\\n`). Default setting of header `Accept` is '*/*' and `Encoding` is 'UTF-8'; default for 'method' is 'GET' and 'ErrorPrefix' is '@-'.") @In(String.class) @Out(Reader.class) @FluxCommand("open-http") @@ -55,7 +55,6 @@ public final class HttpOpener extends DefaultObjectPipe Date: Thu, 1 Sep 2022 18:12:15 +0200 Subject: [PATCH 07/12] Update documentation for `HttpOpener`. (#463) --- metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java b/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java index 27dae4f2e..3c1052b87 100644 --- a/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java +++ b/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java @@ -44,7 +44,7 @@ * @author Jan Schnasse * @author Jens Wille */ -@Description("Opens an HTTP resource. Supports the setting of `Accept` and `Accept-Charset` as HTTP header fields, as well as generic headers (separated by `\\n`). Default setting of header `Accept` is '*/*' and `Encoding` is 'UTF-8'; default for 'method' is 'GET' and 'ErrorPrefix' is '@-'.") +@Description("Opens an HTTP resource. Supports setting HTTP header fields `Accept`, `Accept-Charset` and `Content-Type`, as well as generic headers (separated by `\\n`). Defaults: request `method` = `GET`, request `url` = `@-` (input data), request `body` = `@-` (input data) if request method supports body and input data not already used, `Accept` header = `*/*`, `Accept-Charset` header (`encoding`) = `UTF-8`, `errorPrefix` = `ERROR: `.") @In(String.class) @Out(Reader.class) @FluxCommand("open-http") From aaefc6c0b345a85005d7153a852a515c71dfaf94 Mon Sep 17 00:00:00 2001 From: Jens Wille Date: Thu, 1 Sep 2022 18:12:54 +0200 Subject: [PATCH 08/12] Track response body support for `HttpOpener.Method`. (#463) --- .../java/org/metafacture/io/HttpOpener.java | 46 +++++++++++++------ 1 file changed, 31 insertions(+), 15 deletions(-) diff --git a/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java b/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java index 3c1052b87..7908a2d0f 100644 --- a/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java +++ b/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java @@ -73,22 +73,38 @@ public final class HttpOpener extends DefaultObjectPipe Date: Thu, 1 Sep 2022 18:19:17 +0200 Subject: [PATCH 09/12] Handle HTTP errors in `HttpOpener`. (#463) `HttpUrlConnection` throws `IOException` when trying to read `inputStream` instead of populating `errorStream`. --- .../java/org/metafacture/io/HttpOpener.java | 55 ++++++++++++++----- 1 file changed, 41 insertions(+), 14 deletions(-) diff --git a/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java b/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java index 7908a2d0f..40cd3eb2f 100644 --- a/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java +++ b/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java @@ -63,6 +63,9 @@ public final class HttpOpener extends DefaultObjectPipe headers = new HashMap<>(); private Method method; @@ -237,20 +240,8 @@ public void process(final String input) { } final InputStream errorStream = connection.getErrorStream(); - final InputStream inputStream; - - if (errorStream != null) { - if (errorPrefix != null) { - final InputStream errorPrefixStream = new ByteArrayInputStream(errorPrefix.getBytes()); - inputStream = new SequenceInputStream(errorPrefixStream, errorStream); - } - else { - inputStream = errorStream; - } - } - else { - inputStream = connection.getInputStream(); - } + final InputStream inputStream = errorStream != null ? + getErrorStream(errorStream) : getInputStream(connection); final String contentEncoding = getEncoding(connection.getContentEncoding()); getReceiver().process(new InputStreamReader(inputStream, contentEncoding)); @@ -277,6 +268,42 @@ else if (inputUsed) { return result; } + private InputStream getInputStream(final HttpURLConnection connection) throws IOException { + try { + return connection.getInputStream(); + } + catch (final IOException e) { + final int responseCode = connection.getResponseCode(); + if (responseCode >= SUCCESS_CODE_MIN && responseCode <= SUCCESS_CODE_MAX) { + throw e; + } + else { + final StringBuilder sb = new StringBuilder(String.valueOf(responseCode)); + + final String responseMessage = connection.getResponseMessage(); + if (responseMessage != null) { + sb.append(" - ").append(responseMessage); + } + + return getErrorStream(getInputStream(sb.toString())); + } + } + } + + private InputStream getInputStream(final String string) { + return new ByteArrayInputStream(string.getBytes()); + } + + private InputStream getErrorStream(final InputStream errorStream) { + if (errorPrefix != null) { + final InputStream errorPrefixStream = getInputStream(errorPrefix); + return new SequenceInputStream(errorPrefixStream, errorStream); + } + else { + return errorStream; + } + } + private String getEncoding(final String contentEncoding) { return contentEncoding != null ? contentEncoding : headers.get(ENCODING_HEADER); } From ea537d7b32ca4c24a2928288b2806b0daba46288 Mon Sep 17 00:00:00 2001 From: Jens Wille Date: Thu, 1 Sep 2022 18:31:48 +0200 Subject: [PATCH 10/12] Add unit tests for `HttpOpener`. (#463) --- metafacture-io/build.gradle | 2 + .../org/metafacture/io/HttpOpenerTest.java | 331 ++++++++++++++++++ 2 files changed, 333 insertions(+) create mode 100644 metafacture-io/src/test/java/org/metafacture/io/HttpOpenerTest.java diff --git a/metafacture-io/build.gradle b/metafacture-io/build.gradle index 3de8b53ce..e9947499a 100644 --- a/metafacture-io/build.gradle +++ b/metafacture-io/build.gradle @@ -23,7 +23,9 @@ dependencies { implementation 'commons-io:commons-io:2.5' implementation 'org.apache.commons:commons-compress:1.21' runtimeOnly 'org.tukaani:xz:1.6' + testImplementation 'com.github.tomakehurst:wiremock-jre8:2.33.2' testImplementation 'junit:junit:4.12' testImplementation 'org.mockito:mockito-core:2.5.5' testImplementation 'org.assertj:assertj-core:3.11.1' + testRuntimeOnly 'org.slf4j:slf4j-simple:1.7.21' } diff --git a/metafacture-io/src/test/java/org/metafacture/io/HttpOpenerTest.java b/metafacture-io/src/test/java/org/metafacture/io/HttpOpenerTest.java new file mode 100644 index 000000000..9d380200a --- /dev/null +++ b/metafacture-io/src/test/java/org/metafacture/io/HttpOpenerTest.java @@ -0,0 +1,331 @@ +/* + * Copyright 2013, 2022 Deutsche Nationalbibliothek et al + * + * 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 + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package org.metafacture.io; + +import org.metafacture.commons.ResourceUtil; +import org.metafacture.framework.ObjectReceiver; + +import com.github.tomakehurst.wiremock.client.MappingBuilder; +import com.github.tomakehurst.wiremock.client.ResponseDefinitionBuilder; +import com.github.tomakehurst.wiremock.client.WireMock; +import com.github.tomakehurst.wiremock.core.WireMockConfiguration; +import com.github.tomakehurst.wiremock.http.RequestMethod; +import com.github.tomakehurst.wiremock.junit.WireMockRule; +import com.github.tomakehurst.wiremock.matching.RequestPatternBuilder; +import com.github.tomakehurst.wiremock.matching.StringValuePattern; +import com.github.tomakehurst.wiremock.matching.UrlPattern; +import org.junit.Assert; +import org.junit.ComparisonFailure; +import org.junit.Rule; +import org.junit.Test; +import org.mockito.ArgumentCaptor; +import org.mockito.Captor; +import org.mockito.Mock; +import org.mockito.Mockito; +import org.mockito.junit.MockitoJUnit; +import org.mockito.junit.MockitoRule; + +import java.io.IOException; +import java.io.Reader; +import java.util.Arrays; +import java.util.function.BiConsumer; +import java.util.function.Consumer; + +/** + * Tests for class {@link HttpOpener}. + * + * @author Jens Wille + */ +public final class HttpOpenerTest { + + private static final String TEST_PATH = "/test/path"; + private static final String TEST_URL = "%s" + TEST_PATH; + + private static final String TEST_STRING = "test string"; + private static final StringValuePattern TEST_VALUE = WireMock.equalTo(TEST_STRING); + + private static final String TEST_ERROR = "400 - Bad Request"; + + private static final String REQUEST_BODY = "request body"; + private static final String RESPONSE_BODY = "response bödy"; // UTF-8 + + @Rule + public MockitoRule mockitoRule = MockitoJUnit.rule(); + + @Rule + public WireMockRule wireMockRule = new WireMockRule(WireMockConfiguration.wireMockConfig() + .jettyAcceptors(Runtime.getRuntime().availableProcessors()) + .dynamicPort()); + + @Mock + private ObjectReceiver receiver; + + @Captor + private ArgumentCaptor processedObject; + + @Test + public void shouldPerformGetRequestWithInputAsUrlByDefault() throws IOException { + shouldPerformRequest(TEST_URL, HttpOpener.Method.GET, (o, u) -> {}); + } + + @Test + public void shouldPerformGetRequestWithUrlParameter() throws IOException { + shouldPerformRequest(TEST_STRING, HttpOpener.Method.GET, (o, u) -> { + o.setUrl(u); + }); + } + + @Test + public void shouldPerformPostRequestWithInputAsUrl() throws IOException { + shouldPerformRequest(TEST_URL, HttpOpener.Method.POST, (o, u) -> { + o.setMethod(HttpOpener.Method.POST); + o.setBody(REQUEST_BODY); + }); + } + + @Test + public void shouldPerformPostRequestWithUrlParameter() throws IOException { + shouldPerformRequest(REQUEST_BODY, HttpOpener.Method.POST, (o, u) -> { + o.setMethod(HttpOpener.Method.POST); + o.setUrl(u); + }); + } + + @Test + public void shouldPerformPostRequestWithBodyParameter() throws IOException { + shouldPerformRequest(TEST_STRING, HttpOpener.Method.POST, (o, u) -> { + o.setMethod(HttpOpener.Method.POST); + o.setUrl(u); + o.setBody(REQUEST_BODY); + }); + } + + @Test + public void shouldPerformPostRequestInsteadOfGetWithBodyParameter() throws IOException { + shouldPerformRequest(TEST_URL, HttpOpener.Method.POST, (o, u) -> { + o.setMethod(HttpOpener.Method.GET); + o.setBody(REQUEST_BODY); + }); + } + + @Test + public void shouldPerformPostRequestInsteadOfGetWithInputAsBodyParameter() throws IOException { + shouldPerformRequest(REQUEST_BODY, HttpOpener.Method.POST, (o, u) -> { + o.setMethod(HttpOpener.Method.GET); + o.setUrl(u); + o.setBody("@-"); + }); + } + + @Test + public void shouldPerformGetRequestWithoutBodyWithAlreadyUsedInputAsBodyParameter() throws IOException { + shouldPerformRequest(TEST_URL, HttpOpener.Method.GET, (o, u) -> { + o.setBody("@-"); + }); + } + + @Test + public void shouldPerformPutRequestWithUrlParameter() throws IOException { + shouldPerformRequest(REQUEST_BODY, HttpOpener.Method.PUT, (o, u) -> { + o.setMethod(HttpOpener.Method.PUT); + o.setUrl(u); + }); + } + + @Test + public void shouldPerformPutRequestWithBodyParameter() throws IOException { + shouldPerformRequest(TEST_STRING, HttpOpener.Method.PUT, (o, u) -> { + o.setMethod(HttpOpener.Method.PUT); + o.setUrl(u); + o.setBody(REQUEST_BODY); + }); + } + + @Test + public void shouldPerformDeleteRequestWithUrlParameter() throws IOException { + shouldPerformRequest(REQUEST_BODY, HttpOpener.Method.DELETE, (o, u) -> { + o.setMethod(HttpOpener.Method.DELETE); + o.setUrl(u); + }); + } + + @Test + public void shouldPerformHeadRequestWithUrlParameter() throws IOException { + shouldPerformRequest(REQUEST_BODY, HttpOpener.Method.HEAD, (o, u) -> { + o.setMethod(HttpOpener.Method.HEAD); + o.setUrl(u); + }); + } + + @Test + public void shouldPerformOptionsRequestWithUrlParameter() throws IOException { + shouldPerformRequest(REQUEST_BODY, HttpOpener.Method.OPTIONS, (o, u) -> { + o.setMethod(HttpOpener.Method.OPTIONS); + o.setUrl(u); + }); + } + + @Test + public void shouldPerformTraceRequestWithUrlParameter() throws IOException { + shouldPerformRequest(REQUEST_BODY, HttpOpener.Method.TRACE, (o, u) -> { + o.setMethod(HttpOpener.Method.TRACE); + o.setUrl(u); + }); + } + + @Test + public void shouldPerformGetRequestWithAcceptParameter() throws IOException { + shouldPerformRequest(TEST_URL, HttpOpener.Method.GET, (o, u) -> { + o.setAccept(TEST_STRING); + }, "Accept"); + } + + @Test + public void shouldPerformGetRequestWithSingleValuedHeaderParameter() throws IOException { + shouldPerformRequest(TEST_URL, HttpOpener.Method.GET, (o, u) -> { + o.setHeader("x-api-key: " + TEST_STRING); + }, "x-api-key"); + } + + @Test + public void shouldPerformGetRequestWithMultiValuedHeaderParameter() throws IOException { + shouldPerformRequest(TEST_URL, HttpOpener.Method.GET, (o, u) -> { + o.setHeader("x-api-key: " + TEST_STRING + "\nx-other-header: " + TEST_STRING); + }, "x-api-key", "x-other-header"); + } + + @Test + public void shouldPerformGetRequestWithMultipledHeaderParameters() throws IOException { + shouldPerformRequest(TEST_URL, HttpOpener.Method.GET, (o, u) -> { + o.setHeader("x-api-key: " + TEST_STRING); + o.setHeader("x-other-header: " + TEST_STRING); + }, "x-api-key", "x-other-header"); + } + + @Test + public void shouldPerformPostRequestWithContentTypeParameter() throws IOException { + shouldPerformRequest(REQUEST_BODY, HttpOpener.Method.POST, (o, u) -> { + o.setMethod(HttpOpener.Method.POST); + o.setUrl(u); + o.setContentType(TEST_STRING); + }, "Content-Type"); + } + + @Test + public void shouldPerformPostRequestWithEncodingParameter() throws IOException { + final String encoding = "ISO-8859-1"; + final String header = "Accept-Charset"; + final StringValuePattern value = WireMock.equalTo(encoding); + + try { + shouldPerformRequest(REQUEST_BODY, HttpOpener.Method.POST, (o, u) -> { + o.setMethod(HttpOpener.Method.POST); + o.setUrl(u); + o.setEncoding(encoding); + }, s -> s.withHeader(header, value), q -> q.withHeader(header, value), null); + } + catch (final ComparisonFailure e) { + Assert.assertEquals("expected: but was:", e.getMessage()); + } + } + + @Test + public void shouldPerformPostRequestWithEncodingParameterAndContentEncodingResponseHeader() throws IOException { + final String encoding = "ISO-8859-1"; + final String header = "Accept-Charset"; + final StringValuePattern value = WireMock.equalTo(encoding); + + shouldPerformRequest(REQUEST_BODY, HttpOpener.Method.POST, (o, u) -> { + o.setMethod(HttpOpener.Method.POST); + o.setUrl(u); + o.setEncoding(encoding); + }, + s -> s.withHeader(header, value), + q -> q.withHeader(header, value), + r -> r.withHeader("Content-Encoding", "UTF-8") + ); + } + + @Test + public void shouldPerformGetRequestWithErrorResponse() throws IOException { + shouldPerformRequest(TEST_URL, HttpOpener.Method.GET, (o, u) -> {}, + null, null, WireMock.badRequest().withBody(RESPONSE_BODY), "ERROR: " + TEST_ERROR); + } + + @Test + public void shouldPerformGetRequestWithErrorResponseAndErrorPrefixParameter() throws IOException { + shouldPerformRequest(TEST_URL, HttpOpener.Method.GET, (o, u) -> o.setErrorPrefix(TEST_STRING), + null, null, WireMock.badRequest().withBody(RESPONSE_BODY), TEST_STRING + TEST_ERROR); + } + + @Test + public void shouldPerformGetRequestWithErrorResponseAndWithoutErrorPrefixParameter() throws IOException { + shouldPerformRequest(TEST_URL, HttpOpener.Method.GET, (o, u) -> o.setErrorPrefix(null), + null, null, WireMock.badRequest().withBody(RESPONSE_BODY), TEST_ERROR); + } + + private void shouldPerformRequest(final String input, final HttpOpener.Method method, final BiConsumer consumer, final String... headers) throws IOException { + shouldPerformRequest(input, method, consumer, + s -> Arrays.stream(headers).forEach(h -> s.withHeader(h, TEST_VALUE)), + q -> Arrays.stream(headers).forEach(h -> q.withHeader(h, TEST_VALUE)), null); + } + + private void shouldPerformRequest(final String input, final HttpOpener.Method method, final BiConsumer consumer, final Consumer stubConsumer, final Consumer requestConsumer, final Consumer responseConsumer) throws IOException { + final ResponseDefinitionBuilder response = WireMock.ok().withBody(RESPONSE_BODY); + if (responseConsumer != null) { + responseConsumer.accept(response); + } + + shouldPerformRequest(input, method, + consumer, stubConsumer, requestConsumer, + response, method.getResponseHasBody() ? RESPONSE_BODY : ""); + } + + private void shouldPerformRequest(final String input, final HttpOpener.Method method, final BiConsumer consumer, final Consumer stubConsumer, final Consumer requestConsumer, final ResponseDefinitionBuilder response, final String responseBody) throws IOException { + final String baseUrl = wireMockRule.baseUrl(); + final String url = String.format(TEST_URL, baseUrl); + + final String methodName = method.name(); + final UrlPattern urlPattern = WireMock.urlPathEqualTo(TEST_PATH); + + final HttpOpener opener = new HttpOpener(); + opener.setReceiver(receiver); + consumer.accept(opener, url); + + final MappingBuilder stub = WireMock.request(methodName, urlPattern).willReturn(response); + if (stubConsumer != null) { + stubConsumer.accept(stub); + } + + final RequestPatternBuilder request = new RequestPatternBuilder(RequestMethod.fromString(methodName), urlPattern) + .withRequestBody(method.getRequestHasBody() ? WireMock.equalTo(REQUEST_BODY) : WireMock.absent()); + if (requestConsumer != null) { + requestConsumer.accept(request); + } + + WireMock.stubFor(stub); + + opener.process(String.format(input, baseUrl)); + opener.closeStream(); + + WireMock.verify(request); + + Mockito.verify(receiver).process(processedObject.capture()); + Assert.assertEquals(responseBody, ResourceUtil.readAll(processedObject.getValue())); + } + +} From 75c7ef52e079fda3e131ff5f2e6ce978f2f680d6 Mon Sep 17 00:00:00 2001 From: Jens Wille Date: Fri, 2 Sep 2022 18:57:48 +0200 Subject: [PATCH 11/12] Update documentation for `HttpOpener`. (#463) --- .../java/org/metafacture/io/HttpOpener.java | 81 ++++++++++++------- 1 file changed, 51 insertions(+), 30 deletions(-) diff --git a/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java b/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java index 40cd3eb2f..1f098f4b1 100644 --- a/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java +++ b/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java @@ -50,18 +50,22 @@ @FluxCommand("open-http") public final class HttpOpener extends DefaultObjectPipe> { - private static final Pattern HEADER_FIELD_SEPARATOR = Pattern.compile("\n"); - private static final Pattern HEADER_VALUE_SEPARATOR = Pattern.compile(":"); + public static final String ACCEPT_DEFAULT = "*/*"; + public static final String ACCEPT_HEADER = "accept"; + public static final String CONTENT_TYPE_HEADER = "content-type"; + public static final String DEFAULT_PREFIX = "ERROR: "; + public static final String ENCODING_DEFAULT = "UTF-8"; + public static final String ENCODING_HEADER = "accept-charset"; + public static final String INPUT_DESIGNATOR = "@-"; - private static final String ACCEPT_DEFAULT = "*/*"; - private static final String ACCEPT_HEADER = "accept"; - private static final String CONTENT_TYPE_HEADER = "content-type"; - private static final String DEFAULT_PREFIX = "ERROR: "; - private static final String ENCODING_DEFAULT = "UTF-8"; - private static final String ENCODING_HEADER = "accept-charset"; - private static final String INPUT_DESIGNATOR = "@-"; + public static final String DEFAULT_METHOD_NAME = "GET"; + public static final Method DEFAULT_METHOD = Method.valueOf(DEFAULT_METHOD_NAME); - private static final Method DEFAULT_METHOD = Method.GET; + public static final String HEADER_FIELD_SEPARATOR = "\n"; + public static final String HEADER_VALUE_SEPARATOR = ":"; + + private static final Pattern HEADER_FIELD_SEPARATOR_PATTERN = Pattern.compile(HEADER_FIELD_SEPARATOR); + private static final Pattern HEADER_VALUE_SEPARATOR_PATTERN = Pattern.compile(HEADER_VALUE_SEPARATOR); private static final int SUCCESS_CODE_MIN = 200; private static final int SUCCESS_CODE_MAX = 399; @@ -124,18 +128,27 @@ public HttpOpener() { } /** - * Sets the HTTP accept header value. This is a mime-type such as text/plain - * or text/html. The default value of the accept is */* which means - * any mime-type. + * Sets the HTTP {@value ACCEPT_HEADER} header value. This is a MIME type + * such as {@code text/plain} or {@code application/json}. The default + * value for the accept header is {@value ACCEPT_DEFAULT} which means + * any MIME type. * - * @param accept mime-type to use for the HTTP accept header + * @param accept MIME type to use for the HTTP accept header */ public void setAccept(final String accept) { setHeader(ACCEPT_HEADER, accept); } /** - * Sets the HTTP request body. + * Sets the HTTP request body. The default value for the request body is + * {@value INPUT_DESIGNATOR} if the {@link #setMethod(Method) request + * method} accepts a request body, which means it will use the {@link + * #process(String) input data} data as request body if the input has + * not already been used; otherwise, no request body will be set by + * default. + * + *

If a request body has been set, but the request method does not + * accept a body, the method may be changed to {@code POST}. * * @param body the request body */ @@ -144,20 +157,20 @@ public void setBody(final String body) { } /** - * Sets the HTTP content type header. This is a mime-type such as text/plain, - * text/html. The default is application/json. + * Sets the HTTP {@value CONTENT_TYPE_HEADER} header value. This is a + * MIME type such as {@code text/plain} or {@code application/json}. * - * @param contentType mime-type to use for the HTTP contentType header + * @param contentType MIME type to use for the HTTP content-type header */ public void setContentType(final String contentType) { setHeader(CONTENT_TYPE_HEADER, contentType); } /** - * Sets the preferred encoding of the HTTP response. This value is in the - * accept-charset header. Additonally, the encoding is used for reading the - * HTTP resonse if it does not specify an encoding. The default value for - * the encoding is UTF-8. + * Sets the HTTP {@value ENCODING_HEADER} header value. This is the + * preferred encoding for the HTTP response. Additionally, the encoding + * is used for reading the HTTP response if it does not specify a content + * encoding. The default for the encoding is {@value ENCODING_DEFAULT}. * * @param encoding name of the encoding used for the accept-charset HTTP * header @@ -167,7 +180,8 @@ public void setEncoding(final String encoding) { } /** - * Sets the error prefix. + * Sets the error prefix. The default error prefix is + * {@value DEFAULT_PREFIX}. * * @param errorPrefix the error prefix */ @@ -176,14 +190,18 @@ public void setErrorPrefix(final String errorPrefix) { } /** - * Sets a request property, or multiple request properties separated by - * {@code \n}. + * Sets a request property (header), or multiple request properties + * separated by {@value HEADER_FIELD_SEPARATOR}. Header name and value + * are separated by {@value HEADER_VALUE_SEPARATOR}. The header name is + * case-insensitive. * * @param header request property line + * + * @see #setHeader(String, String) */ public void setHeader(final String header) { - Arrays.stream(HEADER_FIELD_SEPARATOR.split(header)).forEach(h -> { - final String[] parts = HEADER_VALUE_SEPARATOR.split(h, 2); + Arrays.stream(HEADER_FIELD_SEPARATOR_PATTERN.split(header)).forEach(h -> { + final String[] parts = HEADER_VALUE_SEPARATOR_PATTERN.split(h, 2); if (parts.length == 2) { setHeader(parts[0], parts[1].trim()); } @@ -194,7 +212,7 @@ public void setHeader(final String header) { } /** - * Sets a request property. + * Sets a request property (header). The header name is case-insensitive. * * @param key request property key * @param value request property value @@ -204,7 +222,8 @@ public void setHeader(final String key, final String value) { } /** - * Sets the HTTP request method. + * Sets the HTTP request method. The default request method is + * {@value DEFAULT_METHOD_NAME}. * * @param method the request method */ @@ -213,7 +232,9 @@ public void setMethod(final Method method) { } /** - * Sets the HTTP request URL. + * Sets the HTTP request URL. The default value for the request URL is + * {@value INPUT_DESIGNATOR}, which means it will use the {@link + * #process(String) input data} as request URL. * * @param url the request URL */ From b672731fc918f42291a09464dff28e211afa9d6d Mon Sep 17 00:00:00 2001 From: Jens Wille Date: Tue, 6 Sep 2022 19:11:20 +0200 Subject: [PATCH 12/12] Fix error stream handling for `HttpOpener`. (#463) Only read `errorStream` _after_ reading `inputStream` failed. (Thanks to @dr0i for the hint!) Drops use of response code range to determine failure handling. (864f0da) --- .../java/org/metafacture/io/HttpOpener.java | 30 +++++-------------- .../org/metafacture/io/HttpOpenerTest.java | 8 ++--- 2 files changed, 10 insertions(+), 28 deletions(-) diff --git a/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java b/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java index 1f098f4b1..5de3724bb 100644 --- a/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java +++ b/metafacture-io/src/main/java/org/metafacture/io/HttpOpener.java @@ -67,9 +67,6 @@ public final class HttpOpener extends DefaultObjectPipe headers = new HashMap<>(); private Method method; @@ -260,11 +257,9 @@ public void process(final String input) { connection.getOutputStream().write(requestBody.getBytes()); } - final InputStream errorStream = connection.getErrorStream(); - final InputStream inputStream = errorStream != null ? - getErrorStream(errorStream) : getInputStream(connection); - + final InputStream inputStream = getInputStream(connection); final String contentEncoding = getEncoding(connection.getContentEncoding()); + getReceiver().process(new InputStreamReader(inputStream, contentEncoding)); } catch (final IOException e) { @@ -294,30 +289,19 @@ private InputStream getInputStream(final HttpURLConnection connection) throws IO return connection.getInputStream(); } catch (final IOException e) { - final int responseCode = connection.getResponseCode(); - if (responseCode >= SUCCESS_CODE_MIN && responseCode <= SUCCESS_CODE_MAX) { - throw e; + final InputStream errorStream = connection.getErrorStream(); + if (errorStream != null) { + return getErrorStream(errorStream); } else { - final StringBuilder sb = new StringBuilder(String.valueOf(responseCode)); - - final String responseMessage = connection.getResponseMessage(); - if (responseMessage != null) { - sb.append(" - ").append(responseMessage); - } - - return getErrorStream(getInputStream(sb.toString())); + throw e; } } } - private InputStream getInputStream(final String string) { - return new ByteArrayInputStream(string.getBytes()); - } - private InputStream getErrorStream(final InputStream errorStream) { if (errorPrefix != null) { - final InputStream errorPrefixStream = getInputStream(errorPrefix); + final InputStream errorPrefixStream = new ByteArrayInputStream(errorPrefix.getBytes()); return new SequenceInputStream(errorPrefixStream, errorStream); } else { diff --git a/metafacture-io/src/test/java/org/metafacture/io/HttpOpenerTest.java b/metafacture-io/src/test/java/org/metafacture/io/HttpOpenerTest.java index 9d380200a..f86cb19c0 100644 --- a/metafacture-io/src/test/java/org/metafacture/io/HttpOpenerTest.java +++ b/metafacture-io/src/test/java/org/metafacture/io/HttpOpenerTest.java @@ -58,8 +58,6 @@ public final class HttpOpenerTest { private static final String TEST_STRING = "test string"; private static final StringValuePattern TEST_VALUE = WireMock.equalTo(TEST_STRING); - private static final String TEST_ERROR = "400 - Bad Request"; - private static final String REQUEST_BODY = "request body"; private static final String RESPONSE_BODY = "response bödy"; // UTF-8 @@ -263,19 +261,19 @@ public void shouldPerformPostRequestWithEncodingParameterAndContentEncodingRespo @Test public void shouldPerformGetRequestWithErrorResponse() throws IOException { shouldPerformRequest(TEST_URL, HttpOpener.Method.GET, (o, u) -> {}, - null, null, WireMock.badRequest().withBody(RESPONSE_BODY), "ERROR: " + TEST_ERROR); + null, null, WireMock.badRequest().withBody(RESPONSE_BODY), "ERROR: " + RESPONSE_BODY); } @Test public void shouldPerformGetRequestWithErrorResponseAndErrorPrefixParameter() throws IOException { shouldPerformRequest(TEST_URL, HttpOpener.Method.GET, (o, u) -> o.setErrorPrefix(TEST_STRING), - null, null, WireMock.badRequest().withBody(RESPONSE_BODY), TEST_STRING + TEST_ERROR); + null, null, WireMock.badRequest().withBody(RESPONSE_BODY), TEST_STRING + RESPONSE_BODY); } @Test public void shouldPerformGetRequestWithErrorResponseAndWithoutErrorPrefixParameter() throws IOException { shouldPerformRequest(TEST_URL, HttpOpener.Method.GET, (o, u) -> o.setErrorPrefix(null), - null, null, WireMock.badRequest().withBody(RESPONSE_BODY), TEST_ERROR); + null, null, WireMock.badRequest().withBody(RESPONSE_BODY), RESPONSE_BODY); } private void shouldPerformRequest(final String input, final HttpOpener.Method method, final BiConsumer consumer, final String... headers) throws IOException {