/* * Copyright 2002-2016 the original author or 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 * * 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.springframework.http.server.reactive; import java.io.IOException; import java.net.InetSocketAddress; import java.net.URI; import java.nio.ByteBuffer; import java.util.Optional; import io.undertow.connector.ByteBufferPool; import io.undertow.connector.PooledByteBuffer; import io.undertow.server.HttpServerExchange; import io.undertow.server.handlers.Cookie; import io.undertow.util.HeaderValues; import org.xnio.ChannelListener; import org.xnio.channels.StreamSourceChannel; import reactor.core.publisher.Flux; import org.springframework.core.io.buffer.DataBuffer; import org.springframework.core.io.buffer.DataBufferFactory; import org.springframework.http.HttpCookie; import org.springframework.http.HttpHeaders; import org.springframework.http.HttpMethod; import org.springframework.util.Assert; import org.springframework.util.LinkedMultiValueMap; import org.springframework.util.MultiValueMap; import org.springframework.util.StringUtils; /** * Adapt {@link ServerHttpRequest} to the Undertow {@link HttpServerExchange}. * * @author Marek Hawrylczak * @author Rossen Stoyanchev * @since 5.0 */ public class UndertowServerHttpRequest extends AbstractServerHttpRequest { private final HttpServerExchange exchange; private final RequestBodyPublisher body; public UndertowServerHttpRequest(HttpServerExchange exchange, DataBufferFactory bufferFactory) { super(initUri(exchange), initHeaders(exchange)); this.exchange = exchange; this.body = new RequestBodyPublisher(exchange, bufferFactory); this.body.registerListeners(exchange); } private static URI initUri(HttpServerExchange exchange) { Assert.notNull(exchange, "HttpServerExchange is required."); String requestURL = exchange.getRequestURL(); String query = exchange.getQueryString(); String requestUriAndQuery = StringUtils.isEmpty(query) ? requestURL : requestURL + "?" + query; return URI.create(requestUriAndQuery); } private static HttpHeaders initHeaders(HttpServerExchange exchange) { HttpHeaders headers = new HttpHeaders(); for (HeaderValues values : exchange.getRequestHeaders()) { headers.put(values.getHeaderName().toString(), values); } return headers; } public HttpServerExchange getUndertowExchange() { return this.exchange; } @Override public HttpMethod getMethod() { return HttpMethod.valueOf(this.getUndertowExchange().getRequestMethod().toString()); } @Override protected MultiValueMap<String, HttpCookie> initCookies() { MultiValueMap<String, HttpCookie> cookies = new LinkedMultiValueMap<>(); for (String name : this.exchange.getRequestCookies().keySet()) { Cookie cookie = this.exchange.getRequestCookies().get(name); HttpCookie httpCookie = new HttpCookie(name, cookie.getValue()); cookies.add(name, httpCookie); } return cookies; } @Override public Optional<InetSocketAddress> getRemoteAddress() { return Optional.ofNullable(this.exchange.getSourceAddress()); } @Override public Flux<DataBuffer> getBody() { return Flux.from(this.body); } private static class RequestBodyPublisher extends AbstractListenerReadPublisher<DataBuffer> { private final StreamSourceChannel channel; private final DataBufferFactory bufferFactory; private final ByteBufferPool byteBufferPool; private PooledByteBuffer pooledByteBuffer; public RequestBodyPublisher(HttpServerExchange exchange, DataBufferFactory bufferFactory) { this.channel = exchange.getRequestChannel(); this.bufferFactory = bufferFactory; this.byteBufferPool = exchange.getConnection().getByteBufferPool(); } private void registerListeners(HttpServerExchange exchange) { exchange.addExchangeCompleteListener((ex, next) -> { onAllDataRead(); next.proceed(); }); this.channel.getReadSetter().set((ChannelListener<StreamSourceChannel>) c -> onDataAvailable()); this.channel.getCloseSetter().set((ChannelListener<StreamSourceChannel>) c -> onAllDataRead()); this.channel.resumeReads(); } @Override protected void checkOnDataAvailable() { onDataAvailable(); } @Override protected DataBuffer read() throws IOException { if (this.pooledByteBuffer == null) { this.pooledByteBuffer = this.byteBufferPool.allocate(); } ByteBuffer byteBuffer = this.pooledByteBuffer.getBuffer(); int read = this.channel.read(byteBuffer); if (logger.isTraceEnabled()) { logger.trace("read:" + read); } if (read > 0) { byteBuffer.flip(); return this.bufferFactory.wrap(byteBuffer); } else if (read == -1) { onAllDataRead(); } return null; } @Override public void onAllDataRead() { if (this.pooledByteBuffer != null && this.pooledByteBuffer.isOpen()) { this.pooledByteBuffer.close(); } super.onAllDataRead(); } } }