From 131bd8b30d9bc2885470301327720b2edcbe7d9e Mon Sep 17 00:00:00 2001 From: baranowb Date: Fri, 9 Dec 2022 11:16:20 +0100 Subject: [PATCH] [UNDERTOW-1735] add reason-phrase handler and make response-code obey doc contract of one-by-one execution --- .../java/io/undertow/UndertowMessages.java | 3 + .../server/handlers/ReasonPhraseHandler.java | 66 +++++++++++ .../server/handlers/ResponseCodeHandler.java | 28 ++++- .../server/handlers/ResponseHandler.java | 109 ++++++++++++++++++ .../builder/ReasonPhraseHandlerBuilder.java | 68 +++++++++++ .../builder/ResponseCodeHandlerBuilder.java | 2 +- .../builder/ResponseHandlerBuilder.java | 83 +++++++++++++ ...tow.server.handlers.builder.HandlerBuilder | 2 + .../predicate/PredicateParsingTestCase.java | 1 + .../handlers/PredicatedHandlersTestCase.java | 100 +++++++++++++++- 10 files changed, 459 insertions(+), 3 deletions(-) create mode 100644 core/src/main/java/io/undertow/server/handlers/ReasonPhraseHandler.java create mode 100644 core/src/main/java/io/undertow/server/handlers/ResponseHandler.java create mode 100644 core/src/main/java/io/undertow/server/handlers/builder/ReasonPhraseHandlerBuilder.java create mode 100644 core/src/main/java/io/undertow/server/handlers/builder/ResponseHandlerBuilder.java diff --git a/core/src/main/java/io/undertow/UndertowMessages.java b/core/src/main/java/io/undertow/UndertowMessages.java index 664618f620..0eb653dfb8 100644 --- a/core/src/main/java/io/undertow/UndertowMessages.java +++ b/core/src/main/java/io/undertow/UndertowMessages.java @@ -655,4 +655,7 @@ public interface UndertowMessages { @Message(id = 210, value = "Buffer content underflow for exchange '%s', buffer '%s'") IOException bufferUnderflow(HttpServerExchange exchange, ByteBuffer buf); + @Message(id = 211, value = "Exchange '%s' already has body or is blocking.") + IOException exhangeBlockingOrBlocking(HttpServerExchange e); + } diff --git a/core/src/main/java/io/undertow/server/handlers/ReasonPhraseHandler.java b/core/src/main/java/io/undertow/server/handlers/ReasonPhraseHandler.java new file mode 100644 index 0000000000..7ea9655a9f --- /dev/null +++ b/core/src/main/java/io/undertow/server/handlers/ReasonPhraseHandler.java @@ -0,0 +1,66 @@ +/* + * JBoss, Home of Professional Open Source. + * Copyright 2023 Red Hat, Inc., and individual contributors + * as indicated by the @author tags. + * + * 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 io.undertow.server.handlers; + +import io.undertow.UndertowLogger; +import io.undertow.server.HttpHandler; +import io.undertow.server.HttpServerExchange; + +/** + * A handler which simply sets a response code. + * + * @author Bartosz Baranowski + */ +public final class ReasonPhraseHandler implements HttpHandler { + + private static final boolean debugEnabled; + + static { + debugEnabled = UndertowLogger.PREDICATE_LOGGER.isDebugEnabled(); + } + + private final String reasonPhrase; + + private final HttpHandler next; + /** + * Construct a new instance. + * + * @param reasonPhrase the reason phrase to be set in status line + */ + public ReasonPhraseHandler(final HttpHandler next, final String reasonPhrase) { + this.next = next; + this.reasonPhrase = reasonPhrase; + } + + @Override + public void handleRequest(final HttpServerExchange exchange) throws Exception { + exchange.setReasonPhrase(reasonPhrase); + if(debugEnabled) { + UndertowLogger.PREDICATE_LOGGER.debugf("Reason phrase set to [%s] for %s.", this.reasonPhrase, exchange); + } + if(next != null) { + next.handleRequest(exchange); + } + } + + @Override + public String toString() { + return "reason-phrase( " + this.reasonPhrase + " )"; + } +} diff --git a/core/src/main/java/io/undertow/server/handlers/ResponseCodeHandler.java b/core/src/main/java/io/undertow/server/handlers/ResponseCodeHandler.java index e4b4a20591..19f43354ad 100644 --- a/core/src/main/java/io/undertow/server/handlers/ResponseCodeHandler.java +++ b/core/src/main/java/io/undertow/server/handlers/ResponseCodeHandler.java @@ -64,13 +64,36 @@ public final class ResponseCodeHandler implements HttpHandler { private final int responseCode; + private HttpHandler next; /** * Construct a new instance. * * @param responseCode the response code to set + * @param next next handler */ - public ResponseCodeHandler(final int responseCode) { + public ResponseCodeHandler(final HttpHandler next, final int responseCode) { + assert responseCode > 99; + assert responseCode < 600; this.responseCode = responseCode; + this.next = next; + } + + /** + * Construct a new instance. + * + * @param responseCode the response code to set + * @param next next handler + */ + public ResponseCodeHandler(final int responseCode) { + this(null,responseCode); + } + + public HttpHandler getNext() { + return next; + } + + public void setNext(HttpHandler next) { + this.next = next; } @Override @@ -79,6 +102,9 @@ public void handleRequest(final HttpServerExchange exchange) throws Exception { if(debugEnabled) { UndertowLogger.PREDICATE_LOGGER.debugf("Response code set to [%s] for %s.", responseCode, exchange); } + if(next != null) { + next.handleRequest(exchange); + } } @Override diff --git a/core/src/main/java/io/undertow/server/handlers/ResponseHandler.java b/core/src/main/java/io/undertow/server/handlers/ResponseHandler.java new file mode 100644 index 0000000000..4d8b42a544 --- /dev/null +++ b/core/src/main/java/io/undertow/server/handlers/ResponseHandler.java @@ -0,0 +1,109 @@ +/* + * JBoss, Home of Professional Open Source. + * Copyright 2023 Red Hat, Inc., and individual contributors + * as indicated by the @author tags. + * + * 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 io.undertow.server.handlers; + +import io.undertow.UndertowLogger; +import io.undertow.UndertowMessages; +import io.undertow.server.HttpHandler; +import io.undertow.server.HttpServerExchange; +import io.undertow.util.HeaderMap; +import io.undertow.util.Headers; + +/** + * Class which handles set operations for response: code, reason phrase and potentially body and type. Status code is required + * parameter.
+ * The response handler allows to set response body as well.
+ * response(code=404, reason='dont like it') is roughly equivalent to reason-phrase('dont like it');response-code(404)"
+ * + * @author Bartosz Baranowski + */ +public class ResponseHandler implements HttpHandler { + + private static final String DEFAULT_BODY_TYPE = "text/html"; + private static final boolean debugEnabled; + + static { + debugEnabled = UndertowLogger.PREDICATE_LOGGER.isDebugEnabled(); + } + + private final String body; + private final String type; + private final int code; + private final String reason; + private HttpHandler chained; + + // TODO: review parsing/execution rules. For some reason without next, this particular handler does not ignore + // trailing handlers. + public ResponseHandler(final int code, final String reason) { + this(code, reason, null, null); + } + + public ResponseHandler(final int code, final String reason, final String body) { + this(code, reason, body, DEFAULT_BODY_TYPE); + } + + public ResponseHandler(final int code, final String reason, final String body, final String type) { + this.body = body; + this.type = type; + // toString only + this.code = code; + this.reason = reason; + if (reason != null) { + this.chained = new ReasonPhraseHandler(null, reason); + } + this.chained = new ResponseCodeHandler(this.chained, code); + } + + @Override + public void handleRequest(HttpServerExchange exchange) throws Exception { + this.chained.handleRequest(exchange); + if (this.body != null) { + final byte[] bodyBytes = this.body.getBytes("UTF-8"); + final HeaderMap responseHeaders = exchange.getResponseHeaders(); + if (responseHeaders.contains(Headers.CONTENT_LENGTH) || responseHeaders.contains(Headers.CONTENT_TYPE) || exchange.isBlocking()) { + //TODO: need user feedback + throw UndertowMessages.MESSAGES.exhangeBlockingOrBlocking(exchange); + } + responseHeaders.add(Headers.CONTENT_TYPE, this.type); + responseHeaders.add(Headers.CONTENT_LENGTH, bodyBytes.length); + exchange.startBlocking(); + if (exchange.isInIoThread()) { + exchange.dispatch(new HttpHandler() { + + @Override + public void handleRequest(HttpServerExchange exchange) throws Exception { + exchange.getOutputStream().write(bodyBytes); + } + }); + } else { + exchange.getOutputStream().write(bodyBytes); + } + + if (debugEnabled) { + UndertowLogger.PREDICATE_LOGGER.debugf("Respons body set to \n[%s]\nfor %s.", this.body, exchange); + } + } + } + + @Override + public String toString() { + return "response( code='" + code + "'" + ((this.reason != null) ? ", reason='" + this.reason + "'" : "") + "" + + ((this.body != null) ? ", type='" + this.type + "', body='" + this.body + "'" : "") + " )"; + } + +} diff --git a/core/src/main/java/io/undertow/server/handlers/builder/ReasonPhraseHandlerBuilder.java b/core/src/main/java/io/undertow/server/handlers/builder/ReasonPhraseHandlerBuilder.java new file mode 100644 index 0000000000..b0ea220131 --- /dev/null +++ b/core/src/main/java/io/undertow/server/handlers/builder/ReasonPhraseHandlerBuilder.java @@ -0,0 +1,68 @@ +/* + * JBoss, Home of Professional Open Source. + * Copyright 2023 Red Hat, Inc., and individual contributors + * as indicated by the @author tags. + * + * 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 io.undertow.server.handlers.builder; + +import io.undertow.server.HandlerWrapper; +import io.undertow.server.HttpHandler; +import io.undertow.server.handlers.ReasonPhraseHandler; + +import java.util.HashMap; +import java.util.HashSet; +import java.util.Map; +import java.util.Set; + +/** + * @author Bartosz Baranowski + */ +public class ReasonPhraseHandlerBuilder implements HandlerBuilder { + @Override + public String name() { + return "reason-phrase"; + } + + @Override + public Map> parameters() { + Map> parameters = new HashMap<>(); + parameters.put("value", String.class); + return parameters; + } + + @Override + public Set requiredParameters() { + final Set req = new HashSet<>(); + req.add("value"); + return req; + } + + @Override + public String defaultParameter() { + return "value"; + } + + @Override + public HandlerWrapper build(final Map config) { + final String value = (String) config.get("value"); + return new HandlerWrapper() { + @Override + public HttpHandler wrap(HttpHandler handler) { + return new ReasonPhraseHandler(handler, value); + } + }; + } +} diff --git a/core/src/main/java/io/undertow/server/handlers/builder/ResponseCodeHandlerBuilder.java b/core/src/main/java/io/undertow/server/handlers/builder/ResponseCodeHandlerBuilder.java index 30ae8d05a3..0387989fc4 100644 --- a/core/src/main/java/io/undertow/server/handlers/builder/ResponseCodeHandlerBuilder.java +++ b/core/src/main/java/io/undertow/server/handlers/builder/ResponseCodeHandlerBuilder.java @@ -61,7 +61,7 @@ public HandlerWrapper build(final Map config) { return new HandlerWrapper() { @Override public HttpHandler wrap(HttpHandler handler) { - return new ResponseCodeHandler(value); + return new ResponseCodeHandler(handler, value); } }; } diff --git a/core/src/main/java/io/undertow/server/handlers/builder/ResponseHandlerBuilder.java b/core/src/main/java/io/undertow/server/handlers/builder/ResponseHandlerBuilder.java new file mode 100644 index 0000000000..111f31d9f2 --- /dev/null +++ b/core/src/main/java/io/undertow/server/handlers/builder/ResponseHandlerBuilder.java @@ -0,0 +1,83 @@ +/* + * JBoss, Home of Professional Open Source. + * Copyright 2023 Red Hat, Inc., and individual contributors + * as indicated by the @author tags. + * + * 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 io.undertow.server.handlers.builder; + +import java.util.HashMap; +import java.util.HashSet; +import java.util.Map; +import java.util.Set; + +import io.undertow.server.HandlerWrapper; +import io.undertow.server.HttpHandler; +import io.undertow.server.handlers.ResponseHandler; + +/** + * @author Bartosz Baranowski + */ +public class ResponseHandlerBuilder implements HandlerBuilder { + @Override + public String name() { + return "response"; + } + + @Override + public Map> parameters() { + Map> parameters = new HashMap<>(); + parameters.put("code", Integer.class); + parameters.put("reason", String.class); + parameters.put("type", String.class); + parameters.put("body", String.class); + return parameters; + } + + @Override + public Set requiredParameters() { + final Set req = new HashSet<>(); + req.add("code"); + return req; + } + + @Override + public String defaultParameter() { + // default parameter - name(paramValue) not supported + return null; + } + + @Override + public HandlerWrapper build(final Map config) { + final Integer code = (Integer) config.get("code"); + final String reason = (String) config.get("reason"); + final String type = (String) config.get("type"); + final String body = (String) config.get("body"); + return new HandlerWrapper() { + @Override + public HttpHandler wrap(HttpHandler handler) { + if (body == null) { + return new ResponseHandler(code, reason); + } else { + if (type == null) { + return new ResponseHandler(code, reason, body); + } else { + return new ResponseHandler(code, reason, body, type); + } + } + } + }; + } +} diff --git a/core/src/main/resources/META-INF/services/io.undertow.server.handlers.builder.HandlerBuilder b/core/src/main/resources/META-INF/services/io.undertow.server.handlers.builder.HandlerBuilder index 0c2affdbe4..70f45c59c5 100644 --- a/core/src/main/resources/META-INF/services/io.undertow.server.handlers.builder.HandlerBuilder +++ b/core/src/main/resources/META-INF/services/io.undertow.server.handlers.builder.HandlerBuilder @@ -43,3 +43,5 @@ io.undertow.server.handlers.HttpContinueAcceptingHandler$Builder io.undertow.server.handlers.form.EagerFormParsingHandler$Builder io.undertow.server.handlers.SameSiteCookieHandler$Builder io.undertow.server.handlers.SetErrorHandler$Builder +io.undertow.server.handlers.builder.ReasonPhraseHandlerBuilder +io.undertow.server.handlers.builder.ResponseHandlerBuilder diff --git a/core/src/test/java/io/undertow/predicate/PredicateParsingTestCase.java b/core/src/test/java/io/undertow/predicate/PredicateParsingTestCase.java index dd7c1463a6..f7a59f5c30 100644 --- a/core/src/test/java/io/undertow/predicate/PredicateParsingTestCase.java +++ b/core/src/test/java/io/undertow/predicate/PredicateParsingTestCase.java @@ -141,4 +141,5 @@ private void expect(String string, boolean result1, boolean result2) { throw new RuntimeException("String " + string, ex); } } + } diff --git a/core/src/test/java/io/undertow/server/handlers/PredicatedHandlersTestCase.java b/core/src/test/java/io/undertow/server/handlers/PredicatedHandlersTestCase.java index e3c13ae3a1..576afecb6a 100644 --- a/core/src/test/java/io/undertow/server/handlers/PredicatedHandlersTestCase.java +++ b/core/src/test/java/io/undertow/server/handlers/PredicatedHandlersTestCase.java @@ -25,8 +25,11 @@ import io.undertow.server.handlers.builder.PredicatedHandlersParser; import io.undertow.testutils.DefaultServer; import io.undertow.testutils.HttpClientUtils; +import io.undertow.testutils.ProxyIgnore; import io.undertow.testutils.TestHttpClient; import io.undertow.util.StatusCodes; + +import org.apache.http.Header; import org.apache.http.HttpResponse; import org.apache.http.client.methods.HttpGet; import org.junit.Assert; @@ -242,4 +245,99 @@ public void handleRequest(HttpServerExchange exchange) throws Exception { } } -} + @Test @ProxyIgnore + public void testReasonPhrase() throws IOException { + DefaultServer.setRootHandler( + Handlers.predicates( + + PredicatedHandlersParser.parse( + "path('/test') -> reason-phrase('test-my-patience');response-code(480)", getClass().getClassLoader()), new HttpHandler() { + @Override + public void handleRequest(HttpServerExchange exchange) throws Exception { + exchange.getResponseSender().send(exchange.getRelativePath()); + } + })); + TestHttpClient client = new TestHttpClient(); + try { + HttpGet get = new HttpGet(DefaultServer.getDefaultServerURL() + "/test"); + + HttpResponse result = client.execute(get); + Assert.assertEquals("test-my-patience", result.getStatusLine().getReasonPhrase()); + Assert.assertEquals(480, result.getStatusLine().getStatusCode()); + + } finally { + client.getConnectionManager().shutdown(); + } + + } + + @Test @ProxyIgnore + public void testDefaultResponse() throws IOException { + DefaultServer.setRootHandler( + Handlers.predicates( + PredicatedHandlersParser.parse( + "path('/test') -> response(code='208', reason='test-my-patience', body='Dont-Touch')", getClass().getClassLoader()), new HttpHandler() { + @Override + public void handleRequest(HttpServerExchange exchange) throws Exception { + exchange.getResponseSender().send(exchange.getRelativePath()); + } + })); + TestHttpClient client = new TestHttpClient(); + try { + HttpGet get = new HttpGet(DefaultServer.getDefaultServerURL() + "/test"); + + HttpResponse result = client.execute(get); + Assert.assertEquals("test-my-patience", result.getStatusLine().getReasonPhrase()); + Assert.assertEquals(208, result.getStatusLine().getStatusCode()); + final String body = HttpClientUtils.readResponse(result); + Assert.assertEquals("Dont-Touch", body); + Header[] hdrs = result.getHeaders("Content-Length"); + Assert.assertNotNull(hdrs); + Assert.assertEquals(1, hdrs.length); + Assert.assertNotNull(hdrs[0]); + Assert.assertEquals(hdrs[0].getValue(), ""+"Dont-Touch".length()); + hdrs = result.getHeaders("Content-Type"); + Assert.assertNotNull(hdrs); + Assert.assertEquals(1, hdrs.length); + Assert.assertNotNull(hdrs[0]); + Assert.assertEquals(hdrs[0].getValue(), "text/html"); + } finally { + client.getConnectionManager().shutdown(); + } + } + + @Test @ProxyIgnore + public void testCustomTypeResponse() throws IOException { + DefaultServer.setRootHandler( + Handlers.predicates( + PredicatedHandlersParser.parse( + "path('/test') -> response(code='208', reason='test-my-patience', body='Dont-Touch', type='text/plain')", getClass().getClassLoader()), new HttpHandler() { + @Override + public void handleRequest(HttpServerExchange exchange) throws Exception { + exchange.getResponseSender().send(exchange.getRelativePath()); + } + })); + TestHttpClient client = new TestHttpClient(); + try { + HttpGet get = new HttpGet(DefaultServer.getDefaultServerURL() + "/test"); + + HttpResponse result = client.execute(get); + Assert.assertEquals("test-my-patience", result.getStatusLine().getReasonPhrase()); + Assert.assertEquals(208, result.getStatusLine().getStatusCode()); + final String body = HttpClientUtils.readResponse(result); + Assert.assertEquals("Dont-Touch", body); + Header[] hdrs = result.getHeaders("Content-Length"); + Assert.assertNotNull(hdrs); + Assert.assertEquals(1, hdrs.length); + Assert.assertNotNull(hdrs[0]); + Assert.assertEquals(hdrs[0].getValue(), ""+"Dont-Touch".length()); + hdrs = result.getHeaders("Content-Type"); + Assert.assertNotNull(hdrs); + Assert.assertEquals(1, hdrs.length); + Assert.assertNotNull(hdrs[0]); + Assert.assertEquals(hdrs[0].getValue(), "text/plain"); + } finally { + client.getConnectionManager().shutdown(); + } + } +} \ No newline at end of file