1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
|
// Copyright 2017 Yahoo Holdings. Licensed under the terms of the Apache 2.0 license. See LICENSE in the project root.
package com.yahoo.vespa.clustercontroller.apputil.communication.http;
import com.yahoo.container.jdisc.LoggingRequestHandler;
import com.yahoo.container.logging.AccessLog;
import com.yahoo.jdisc.HeaderFields;
import com.yahoo.jdisc.Response;
import com.yahoo.jdisc.handler.CompletionHandler;
import com.yahoo.text.Utf8;
import com.yahoo.vespa.clustercontroller.utils.communication.http.HttpRequest;
import com.yahoo.vespa.clustercontroller.utils.communication.http.HttpRequestHandler;
import com.yahoo.vespa.clustercontroller.utils.communication.http.HttpResult;
import org.apache.commons.io.IOUtils;
import java.io.*;
import java.time.Duration;
import java.util.concurrent.Executor;
import java.util.logging.Logger;
/**
* Note. This class is tested through apache http instance test, using this as other endpoint.
* @author humbe
* @author Harald Musum
* @author Vegard Sjonfjell
*/
public class JDiscHttpRequestHandler extends LoggingRequestHandler {
private static final Logger log = Logger.getLogger(JDiscHttpRequestHandler.class.getName());
private final HttpRequestHandler requestHandler;
public JDiscHttpRequestHandler(HttpRequestHandler handler, Executor executor, AccessLog accessLog) {
super(executor, accessLog);
this.requestHandler = handler;
}
static class EmptyCompletionHandler implements CompletionHandler {
@Override
public void completed() { }
@Override
public void failed(Throwable throwable) { }
}
@Override
public com.yahoo.container.jdisc.HttpResponse handle(com.yahoo.container.jdisc.HttpRequest request) {
final HttpRequest legacyRequest = new HttpRequest();
final com.yahoo.jdisc.http.HttpRequest jDiscRequest = request.getJDiscRequest();
legacyRequest.setHost(request.getUri().getHost());
setOperation(legacyRequest, request.getMethod());
legacyRequest.setPort(request.getUri().getPort());
legacyRequest.setPath(request.getUri().getPath());
copyPostData(request, legacyRequest);
copyRequestHeaders(legacyRequest, jDiscRequest);
copyParameters(legacyRequest, jDiscRequest);
legacyRequest.setTimeout(Duration.ofMinutes(60).toMillis());
try {
final HttpResult result = requestHandler.handleRequest(legacyRequest);
log.fine("Got result " + result.toString(true));
return copyResponse(result);
} catch (Exception e) {
log.warning("Caught exception while handling request: " + e.getMessage());
return new com.yahoo.container.jdisc.HttpResponse(500) {
@Override
public void render(OutputStream outputStream) throws IOException {
outputStream.write(Utf8.toBytes(e.getMessage()));
}
};
}
}
static HttpRequest setOperation(HttpRequest request, com.yahoo.jdisc.http.HttpRequest.Method method) {
switch (method) {
case GET: return request.setHttpOperation(HttpRequest.HttpOp.GET);
case POST: return request.setHttpOperation(HttpRequest.HttpOp.POST);
case PUT: return request.setHttpOperation(HttpRequest.HttpOp.PUT);
case DELETE: return request.setHttpOperation(HttpRequest.HttpOp.DELETE);
default: throw new IllegalStateException("Unhandled method " + method);
}
}
private com.yahoo.container.jdisc.HttpResponse copyResponse(final HttpResult result) {
return new com.yahoo.container.jdisc.HttpResponse(result.getHttpReturnCode()) {
@Override
public void render(OutputStream outputStream) throws IOException {
outputStream.write(Utf8.toBytes(result.getContent().toString()));
}
@Override
public void complete(){
copyResponseHeaders(result, getJdiscResponse());
}
};
}
private void copyPostData(com.yahoo.container.jdisc.HttpRequest request, HttpRequest legacyRequest) {
try {
legacyRequest.setPostContent(IOUtils.toString(request.getData(), "UTF-8"));
} catch (IOException e) {
throw new RuntimeException(e);
}
}
private static void copyParameters(HttpRequest legacyRequest, com.yahoo.jdisc.http.HttpRequest jDiscRequest) {
for (String key : jDiscRequest.parameters().keySet()) {
for (String value : jDiscRequest.parameters().get(key)) {
legacyRequest.addUrlOption(key, value);
}
}
}
private static void copyRequestHeaders(HttpRequest legacyRequest, com.yahoo.jdisc.http.HttpRequest jDiscRequest) {
for (String key : jDiscRequest.headers().keySet()) {
for (String value : jDiscRequest.headers().get(key)) {
legacyRequest.addHttpHeader(key, value);
}
}
}
private static HeaderFields copyResponseHeaders(HttpResult result, Response response) {
HeaderFields headers = new HeaderFields();
for (HttpRequest.KeyValuePair keyValuePair : result.getHeaders()) {
response.headers().put((keyValuePair.getKey()), keyValuePair.getValue());
}
return headers;
}
}
|