/*
* Copyright 2013 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 ratpack.test.http.internal;
import com.google.common.collect.ImmutableMultimap;
import com.google.common.collect.Lists;
import com.google.common.collect.Maps;
import com.google.common.net.HostAndPort;
import io.netty.handler.codec.http.cookie.ClientCookieDecoder;
import io.netty.handler.codec.http.cookie.ClientCookieEncoder;
import io.netty.handler.codec.http.cookie.Cookie;
import ratpack.exec.ExecController;
import ratpack.exec.internal.DefaultExecController;
import ratpack.func.Action;
import ratpack.func.Function;
import ratpack.http.HttpUrlBuilder;
import ratpack.http.client.HttpClient;
import ratpack.http.client.ReceivedResponse;
import ratpack.http.client.RequestSpec;
import ratpack.http.client.internal.DelegatingRequestSpec;
import ratpack.http.internal.HttpHeaderConstants;
import ratpack.test.ApplicationUnderTest;
import ratpack.test.http.TestHttpClient;
import ratpack.test.internal.BlockingHttpClient;
import ratpack.test.internal.TestByteBufAllocators;
import ratpack.util.Exceptions;
import java.net.URI;
import java.net.URISyntaxException;
import java.time.Duration;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import static ratpack.util.Exceptions.uncheck;
public class DefaultTestHttpClient implements TestHttpClient {
private final ApplicationUnderTest applicationUnderTest;
private final BlockingHttpClient client = new BlockingHttpClient();
private final Action<? super RequestSpec> defaultRequestConfig;
private final Map<String, List<Cookie>> cookies = Maps.newLinkedHashMap();
private Action<? super RequestSpec> request = Action.noop();
private Action<? super ImmutableMultimap.Builder<String, Object>> params = Action.noop();
private ReceivedResponse response;
public DefaultTestHttpClient(ApplicationUnderTest applicationUnderTest, Action<? super RequestSpec> defaultRequestConfig) {
this.applicationUnderTest = applicationUnderTest;
this.defaultRequestConfig = defaultRequestConfig;
}
@Override
public ApplicationUnderTest getApplicationUnderTest() {
return applicationUnderTest;
}
@Override
public TestHttpClient requestSpec(Action<? super RequestSpec> requestAction) {
request = requestAction;
return this;
}
@Override
public TestHttpClient params(Action<? super ImmutableMultimap.Builder<String, Object>> params) {
this.params = params;
return this;
}
@Override
public void resetRequest() {
request = Action.noop();
cookies.clear();
}
@Override
public ReceivedResponse getResponse() {
return response;
}
@Override
public ReceivedResponse head() {
return head("");
}
@Override
public ReceivedResponse head(String path) {
return request(path, RequestSpec::head);
}
@Override
public ReceivedResponse options() {
return options("");
}
@Override
public ReceivedResponse options(String path) {
return request(path, RequestSpec::options);
}
@Override
public String optionsText() {
return optionsText("");
}
@Override
public String optionsText(String path) {
return options(path).getBody().getText();
}
@Override
public ReceivedResponse get() {
return get("");
}
@Override
public ReceivedResponse get(String path) {
return request(path, RequestSpec::get);
}
@Override
public String getText() {
return getText("");
}
@Override
public String getText(String path) {
return get(path).getBody().getText();
}
@Override
public ReceivedResponse post() {
return post("");
}
@Override
public ReceivedResponse post(String path) {
return request(path, RequestSpec::post);
}
@Override
public String postText() {
return postText("");
}
@Override
public String postText(String path) {
post(path);
return response.getBody().getText();
}
@Override
public ReceivedResponse put() {
return put("");
}
@Override
public ReceivedResponse put(String path) {
return request(path, RequestSpec::put);
}
@Override
public String putText() {
return putText("");
}
@Override
public String putText(String path) {
return put(path).getBody().getText();
}
@Override
public ReceivedResponse patch() {
return patch("");
}
@Override
public ReceivedResponse patch(String path) {
return request(path, RequestSpec::patch);
}
@Override
public String patchText() {
return patchText("");
}
@Override
public String patchText(String path) {
return patch(path).getBody().getText();
}
@Override
public ReceivedResponse delete() {
return delete("");
}
@Override
public ReceivedResponse delete(String path) {
return request(path, RequestSpec::delete);
}
@Override
public String deleteText() {
return deleteText("");
}
@Override
public String deleteText(String path) {
return delete(path).getBody().getText();
}
@Override
public ReceivedResponse request(Action<? super RequestSpec> requestAction) {
return request("", requestAction);
}
@Override
public ReceivedResponse request(String path, Action<? super RequestSpec> requestAction) {
try (ExecController execController = new DefaultExecController(2)) {
URI uri = builder(path).params(params).build();
try (HttpClient httpClient = httpClient()) {
response = client.request(httpClient, uri, execController, Duration.ofMinutes(60), requestSpec -> {
final RequestSpec decorated = new CookieHandlingRequestSpec(requestSpec);
decorated.get();
defaultRequestConfig.execute(decorated);
request.execute(decorated);
requestAction.execute(decorated);
int port = uri.getPort() > 0 ? uri.getPort() : 80;
requestSpec.getHeaders().add(HttpHeaderConstants.HOST, HostAndPort.fromParts(uri.getHost(), port).toString());
});
}
} catch (Throwable throwable) {
throw uncheck(throwable);
}
extractCookies(response);
return response;
}
private HttpClient httpClient() {
return Exceptions.uncheck(() -> HttpClient.of(s -> s
.byteBufAllocator(TestByteBufAllocators.LEAKING_UNPOOLED_HEAP)
.maxContentLength(Integer.MAX_VALUE)
.poolSize(8)
));
}
private void applyCookies(RequestSpec requestSpec) {
List<Cookie> requestCookies = getCookies(requestSpec.getUri().getPath());
String encodedCookie = requestCookies.isEmpty() ? "" : ClientCookieEncoder.STRICT.encode(requestCookies);
requestSpec.getHeaders().add(HttpHeaderConstants.COOKIE, encodedCookie);
}
private void extractCookies(ReceivedResponse response) {
List<String> cookieHeaders = response.getHeaders().getAll("Set-Cookie");
for (String cookieHeader : cookieHeaders) {
Cookie decodedCookie = ClientCookieDecoder.STRICT.decode(cookieHeader);
if (decodedCookie != null) {
if (decodedCookie.value() == null || decodedCookie.value().isEmpty()) {
// clear cookie with the given name, skip the other parameters (path, domain) in compare to
cookies.forEach((key, list) -> {
Iterator<Cookie> iter = list.listIterator();
while (iter.hasNext()) {
if (iter.next().name().equals(decodedCookie.name())) {
iter.remove();
}
}
});
} else {
String cookiePath = decodedCookie.path();
cookiePath = (cookiePath != null && !("".equals(cookiePath))) ? cookiePath : "/";
List<Cookie> pathCookies = cookies.get(cookiePath);
if (pathCookies == null) {
pathCookies = Lists.newLinkedList();
cookies.put(cookiePath, pathCookies);
}
if (pathCookies.contains(decodedCookie)) {
pathCookies.remove(decodedCookie);
}
pathCookies.add(decodedCookie);
}
}
}
}
private HttpUrlBuilder builder(String path) {
try {
URI basePath = new URI(path);
if (basePath.isAbsolute()) {
return HttpUrlBuilder.base(basePath);
} else {
path = path.startsWith("/") ? path.substring(1) : path;
return HttpUrlBuilder.base(new URI(applicationUnderTest.getAddress().toString() + path));
}
} catch (URISyntaxException e) {
throw uncheck(e);
}
}
public List<Cookie> getCookies(String path) {
List<Cookie> clonedList = Lists.newLinkedList();
if (path == null || "".equals(path) || "/".equals(path)) {
List<Cookie> list = cookies.get("/");
if (list != null) {
clonedList.addAll(list);
}
} else {
cookies.forEach((key, list) -> {
if ("/".equals(key)) {
clonedList.addAll(list);
} else if (path.startsWith(key)) {
clonedList.addAll(list);
}
});
}
return clonedList;
}
private class CookieHandlingRequestSpec extends DelegatingRequestSpec {
public CookieHandlingRequestSpec(RequestSpec delegate) {
super(delegate);
onRedirect(Function.constant(Action.noop()));
applyCookies(this);
}
@Override
public RequestSpec onRedirect(Function<? super ReceivedResponse, Action<? super RequestSpec>> function) {
return super.onRedirect(resp -> {
extractCookies(resp);
final Action<? super RequestSpec> userFunc = function.apply(response);
if (userFunc == null) {
return null;
} else {
return spec -> {
final RequestSpec decorated = new CookieHandlingRequestSpec(spec);
userFunc.execute(decorated);
};
}
});
}
}
}