forked from OpenFeign/feign
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathMethodHandler.java
More file actions
142 lines (125 loc) · 5.2 KB
/
MethodHandler.java
File metadata and controls
142 lines (125 loc) · 5.2 KB
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
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
/*
* Copyright 2013 Netflix, Inc.
*
* 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 feign;
import com.google.common.base.Function;
import com.google.common.collect.ImmutableList;
import com.google.common.reflect.TypeToken;
import java.io.IOException;
import java.net.URI;
import javax.inject.Inject;
import javax.inject.Provider;
import feign.Request.Options;
import feign.codec.Decoder;
import feign.codec.ErrorDecoder;
import static com.google.common.base.Preconditions.checkNotNull;
import static com.google.common.net.HttpHeaders.LOCATION;
import static feign.FeignException.errorExecuting;
import static feign.FeignException.errorReading;
final class MethodHandler {
static class Factory {
private final Client client;
private final Provider<Retryer> retryer;
private final Wire wire;
@Inject Factory(Client client, Provider<Retryer> retryer, Wire wire) {
this.client = checkNotNull(client, "client");
this.retryer = checkNotNull(retryer, "retryer");
this.wire = checkNotNull(wire, "wire");
}
public MethodHandler create(Target<?> target, MethodMetadata md,
Function<Object[], RequestTemplate> buildTemplateFromArgs, Options options, Decoder decoder, ErrorDecoder errorDecoder) {
return new MethodHandler(target, client, retryer, wire, md, buildTemplateFromArgs, options, decoder, errorDecoder);
}
}
private final MethodMetadata metadata;
private final Target<?> target;
private final Client client;
private final Provider<Retryer> retryer;
private final Wire wire;
private final Function<Object[], RequestTemplate> buildTemplateFromArgs;
private final Options options;
private final Decoder decoder;
private final ErrorDecoder errorDecoder;
// cannot inject wildcards in dagger
@SuppressWarnings("rawtypes")
private MethodHandler(Target target, Client client, Provider<Retryer> retryer, Wire wire, MethodMetadata metadata,
Function<Object[], RequestTemplate> buildTemplateFromArgs, Options options, Decoder decoder, ErrorDecoder errorDecoder) {
this.target = checkNotNull(target, "target");
this.client = checkNotNull(client, "client for %s", target);
this.retryer = checkNotNull(retryer, "retryer for %s", target);
this.wire = checkNotNull(wire, "wire for %s", target);
this.metad
70BA
ata = checkNotNull(metadata, "metadata for %s", target);
this.buildTemplateFromArgs = checkNotNull(buildTemplateFromArgs, "metadata for %s", target);
this.options = checkNotNull(options, "options for %s", target);
this.decoder = checkNotNull(decoder, "decoder for %s", target);
this.errorDecoder = checkNotNull(errorDecoder, "errorDecoder for %s", target);
}
public Object invoke(Object[] argv) throws Throwable {
RequestTemplate template = buildTemplateFromArgs.apply(argv);
Retryer retryer = this.retryer.get();
while (true) {
try {
return executeAndDecode(metadata.configKey(), template, metadata.returnType());
} catch (RetryableException e) {
retryer.continueOrPropagate(e);
continue;
}
}
}
public Object executeAndDecode(String configKey, RequestTemplate template, TypeToken<?> returnType)
throws Throwable {
// create the request from a mutable copy of the input template.
Request request = target.apply(new RequestTemplate(template));
wire.wireRequest(target, request);
Response response = execute(request);
try {
response = wire.wireAndRebufferResponse(target, response);
if (response.status() >= 200 && response.status() < 300) {
if (returnType.getRawType().equals(Response.class)) {
return response;
} else if (returnType.getRawType() == URI.class && !response.body().isPresent()) {
ImmutableList<String> location = response.headers().get(LOCATION);
if (!location.isEmpty())
return URI.create(location.get(0));
} else if (returnType.getRawType() == void.class) {
return null;
}
return decoder.decode(configKey, response, returnType);
} else {
return errorDecoder.decode(configKey, response, returnType);
}
} catch (Throwable e) {
ensureBodyClosed(response);
if (IOException.class.isInstance(e))
throw errorReading(request, response, IOException.class.cast(e));
throw e;
}
}
private void ensureBodyClosed(Response response) {
if (response.body().isPresent()) {
try {
response.body().get().close();
} catch (IOException ignored) { // NOPMD
}
}
}
private Response execute(Request request) {
try {
return client.execute(request, options);
} catch (IOException e) {
throw errorExecuting(request, e);
}
}
}