Polishing and refactoring

See gh-28395
This commit is contained in:
rstoyanchev 2022-04-29 16:25:48 +01:00
parent 38bf0776a1
commit d91b840d0e
5 changed files with 157 additions and 170 deletions

View File

@ -187,9 +187,9 @@ public final class HttpServiceProxyFactory {
private List<HttpServiceArgumentResolver> initArgumentResolvers(ConversionService conversionService) { private List<HttpServiceArgumentResolver> initArgumentResolvers(ConversionService conversionService) {
List<HttpServiceArgumentResolver> resolvers = new ArrayList<>(this.customResolvers); List<HttpServiceArgumentResolver> resolvers = new ArrayList<>(this.customResolvers);
resolvers.add(new HttpMethodArgumentResolver());
resolvers.add(new PathVariableArgumentResolver(conversionService));
resolvers.add(new RequestHeaderArgumentResolver(conversionService)); resolvers.add(new RequestHeaderArgumentResolver(conversionService));
resolvers.add(new PathVariableArgumentResolver(conversionService));
resolvers.add(new HttpMethodArgumentResolver());
return resolvers; return resolvers;
} }

View File

@ -63,28 +63,28 @@ public class PathVariableArgumentResolver implements HttpServiceArgumentResolver
return false; return false;
} }
if (Map.class.isAssignableFrom(parameter.getParameterType())) { Class<?> parameterType = parameter.getParameterType();
boolean required = (annotation.required() && !Optional.class.isAssignableFrom(parameterType));
if (Map.class.isAssignableFrom(parameterType)) {
if (argument != null) { if (argument != null) {
Assert.isInstanceOf(Map.class, argument); Assert.isInstanceOf(Map.class, argument);
((Map<Object, ?>) argument).forEach((key, value) -> ((Map<String, ?>) argument).forEach((key, value) ->
addUriParameter(key, value, annotation.required(), requestValues)); addUriParameter(key, value, required, requestValues));
} }
} }
else { else {
String name = StringUtils.hasText(annotation.value()) ? annotation.value() : annotation.name(); String name = StringUtils.hasText(annotation.value()) ? annotation.value() : annotation.name();
name = StringUtils.hasText(name) ? name : parameter.getParameterName(); name = StringUtils.hasText(name) ? name : parameter.getParameterName();
Assert.notNull(name, "Failed to determine path variable name for parameter: " + parameter); Assert.notNull(name, "Failed to determine path variable name for parameter: " + parameter);
addUriParameter(name, argument, annotation.required(), requestValues); addUriParameter(name, argument, required, requestValues);
} }
return true; return true;
} }
private void addUriParameter( private void addUriParameter(
Object name, @Nullable Object value, boolean required, HttpRequestValues.Builder requestValues) { String name, @Nullable Object value, boolean required, HttpRequestValues.Builder requestValues) {
String stringName = this.conversionService.convert(name, String.class);
Assert.notNull(stringName, "Missing path variable name");
if (value instanceof Optional) { if (value instanceof Optional) {
value = ((Optional<?>) value).orElse(null); value = ((Optional<?>) value).orElse(null);
@ -95,15 +95,15 @@ public class PathVariableArgumentResolver implements HttpServiceArgumentResolver
} }
if (value == null) { if (value == null) {
Assert.isTrue(!required, "Missing required path variable '" + stringName + "'"); Assert.isTrue(!required, "Missing required path variable '" + name + "'");
return; return;
} }
if (logger.isTraceEnabled()) { if (logger.isTraceEnabled()) {
logger.trace("Resolved path variable '" + stringName + "' to " + value); logger.trace("Resolved path variable '" + name + "' to " + value);
} }
requestValues.setUriVariable(stringName, (String) value); requestValues.setUriVariable(name, (String) value);
} }
} }

View File

@ -19,9 +19,7 @@ package org.springframework.web.service.invoker;
import java.util.Arrays; import java.util.Arrays;
import java.util.Collection; import java.util.Collection;
import java.util.Map; import java.util.Map;
import java.util.Objects;
import java.util.Optional; import java.util.Optional;
import java.util.stream.Stream;
import org.apache.commons.logging.Log; import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory; import org.apache.commons.logging.LogFactory;
@ -30,17 +28,26 @@ import org.springframework.core.MethodParameter;
import org.springframework.core.convert.ConversionService; import org.springframework.core.convert.ConversionService;
import org.springframework.lang.Nullable; import org.springframework.lang.Nullable;
import org.springframework.util.Assert; import org.springframework.util.Assert;
import org.springframework.util.ObjectUtils;
import org.springframework.util.StringUtils; import org.springframework.util.StringUtils;
import org.springframework.web.bind.annotation.RequestHeader; import org.springframework.web.bind.annotation.RequestHeader;
import org.springframework.web.bind.annotation.ValueConstants; import org.springframework.web.bind.annotation.ValueConstants;
/** /**
* An implementation of {@link HttpServiceArgumentResolver} that resolves * {@link HttpServiceArgumentResolver} for {@link RequestHeader @RequestHeader}
* request headers based on method arguments annotated * annotated arguments.
* with {@link RequestHeader}. {@code null} values are allowed only *
* if {@link RequestHeader#required()} is {@code true}. {@code null} * <p>The argument may be:
* values are replaced with {@link RequestHeader#defaultValue()} if it * <ul>
* is not equal to {@link ValueConstants#DEFAULT_NONE}. * <li>{@code Map} or {@link org.springframework.util.MultiValueMap} with
* multiple headers and value(s).
* <li>{@code Collection} or an array of header values.
* <li>An individual header value.
* </ul>
*
* <p>Individual header values may be Strings or Objects to be converted to
* String values through the configured {@link ConversionService}.
* *
* @author Olga Maciaszek-Sharma * @author Olga Maciaszek-Sharma
* @since 6.0 * @since 6.0
@ -51,89 +58,91 @@ public class RequestHeaderArgumentResolver implements HttpServiceArgumentResolve
private final ConversionService conversionService; private final ConversionService conversionService;
public RequestHeaderArgumentResolver(ConversionService conversionService) { public RequestHeaderArgumentResolver(ConversionService conversionService) {
Assert.notNull(conversionService, "ConversionService is required"); Assert.notNull(conversionService, "ConversionService is required");
this.conversionService = conversionService; this.conversionService = conversionService;
} }
@Override
public boolean resolve(@Nullable Object argument, MethodParameter parameter,
HttpRequestValues.Builder requestValues) {
RequestHeader annotation = parameter.getParameterAnnotation(RequestHeader.class);
if (annotation == null) { @SuppressWarnings("unchecked")
@Override
public boolean resolve(
@Nullable Object argument, MethodParameter parameter, HttpRequestValues.Builder requestValues) {
RequestHeader annot = parameter.getParameterAnnotation(RequestHeader.class);
if (annot == null) {
return false; return false;
} }
if (Map.class.isAssignableFrom(parameter.getParameterType())) { Class<?> parameterType = parameter.getParameterType();
boolean required = (annot.required() && !Optional.class.isAssignableFrom(parameterType));
Object defaultValue = (ValueConstants.DEFAULT_NONE.equals(annot.defaultValue()) ? null : annot.defaultValue());
if (Map.class.isAssignableFrom(parameterType)) {
if (argument != null) { if (argument != null) {
Assert.isInstanceOf(Map.class, argument); Assert.isInstanceOf(Map.class, argument);
((Map<?, ?>) argument).forEach((key, value) -> ((Map<String, ?>) argument).forEach((key, value) ->
addRequestHeader(key, value, annotation.required(), annotation.defaultValue(), addHeader(key, value, false, defaultValue, requestValues));
requestValues));
} }
} }
else { else {
String name = StringUtils.hasText(annotation.value()) ? String name = StringUtils.hasText(annot.value()) ? annot.value() : annot.name();
annotation.value() : annotation.name();
name = StringUtils.hasText(name) ? name : parameter.getParameterName(); name = StringUtils.hasText(name) ? name : parameter.getParameterName();
Assert.notNull(name, "Failed to determine request header name for parameter: " + parameter); Assert.notNull(name, "Failed to determine request header name for parameter: " + parameter);
addRequestHeader(name, argument, annotation.required(), annotation.defaultValue(), addHeader(name, argument, required, defaultValue, requestValues);
requestValues);
} }
return true; return true;
} }
private void addRequestHeader( private void addHeader(
Object name, @Nullable Object value, boolean required, String defaultValue, String name, @Nullable Object value, boolean required, @Nullable Object defaultValue,
HttpRequestValues.Builder requestValues) { HttpRequestValues.Builder requestValues) {
String stringName = this.conversionService.convert(name, String.class); value = (ObjectUtils.isArray(value) ? Arrays.asList((Object[]) value) : value);
Assert.notNull(stringName, "Failed to convert request header name '" + if (value instanceof Collection<?> elements) {
name + "' to String"); boolean hasValue = false;
for (Object element : elements) {
if (element != null) {
hasValue = true;
addHeaderValue(name, element, false, requestValues);
}
}
if (hasValue) {
return;
}
value = null;
}
if (value instanceof Optional) { if (value instanceof Optional<?> optionalValue) {
value = ((Optional<?>) value).orElse(null); value = optionalValue.orElse(null);
}
if (value == null && defaultValue != null) {
value = defaultValue;
}
addHeaderValue(name, value, required, requestValues);
}
private void addHeaderValue(
String name, @Nullable Object value, boolean required, HttpRequestValues.Builder requestValues) {
if (!(value instanceof String)) {
value = this.conversionService.convert(value, String.class);
} }
if (value == null) { if (value == null) {
if (!ValueConstants.DEFAULT_NONE.equals(defaultValue)) { Assert.isTrue(!required, "Missing required header '" + name + "'");
value = defaultValue; return;
}
else {
Assert.isTrue(!required, "Missing required request header '" + stringName + "'");
return;
}
} }
String[] headerValues = toStringArray(value);
if (logger.isTraceEnabled()) { if (logger.isTraceEnabled()) {
logger.trace("Resolved request header '" + stringName + "' to list of values: " + logger.trace("Resolved header '" + name + ":" + value + "'");
String.join(", ", headerValues)); }
}
requestValues.addHeader(stringName, headerValues); requestValues.addHeader(name, (String) value);
}
private String[] toStringArray(Object value) {
return toValueStream(value)
.filter(Objects::nonNull)
.map(headerElement -> headerElement instanceof String
? (String) headerElement :
this.conversionService.convert(headerElement, String.class))
.filter(Objects::nonNull)
.toArray(String[]::new);
}
private Stream<?> toValueStream(Object value) {
if (value instanceof Object[]) {
return Arrays.stream((Object[]) value);
}
if (value instanceof Collection<?>) {
return ((Collection<?>) value).stream();
}
return Stream.of(value);
} }
} }

View File

@ -135,12 +135,6 @@ class PathVariableArgumentResolverTests {
.isThrownBy(() -> this.service.executeOptionalValueMap(Map.of("id", Optional.empty()))); .isThrownBy(() -> this.service.executeOptionalValueMap(Map.of("id", Optional.empty())));
} }
@Test
void shouldResolvePathVariableNameFromObjectMapKey() {
this.service.executeValueMapWithObjectKey(Map.of(Boolean.TRUE, "true"));
assertPathVariable("true", "true");
}
@SuppressWarnings("SameParameterValue") @SuppressWarnings("SameParameterValue")
private void assertPathVariable(String name, @Nullable String expectedValue) { private void assertPathVariable(String name, @Nullable String expectedValue) {
assertThat(getActualUriVariables().get(name)).isEqualTo(expectedValue); assertThat(getActualUriVariables().get(name)).isEqualTo(expectedValue);
@ -184,9 +178,6 @@ class PathVariableArgumentResolverTests {
@GetExchange @GetExchange
void executeValueMap(@Nullable @PathVariable Map<String, String> map); void executeValueMap(@Nullable @PathVariable Map<String, String> map);
@GetExchange
void executeValueMapWithObjectKey(@Nullable @PathVariable Map<Object, String> map);
@GetExchange @GetExchange
void executeOptionalValueMap(@PathVariable Map<String, Optional<String>> map); void executeOptionalValueMap(@PathVariable Map<String, Optional<String>> map);
} }

View File

@ -31,8 +31,9 @@ import org.springframework.web.service.annotation.GetExchange;
import static org.assertj.core.api.Assertions.assertThat; import static org.assertj.core.api.Assertions.assertThat;
import static org.assertj.core.api.Assertions.assertThatIllegalArgumentException; import static org.assertj.core.api.Assertions.assertThatIllegalArgumentException;
/** /**
* Tests for {@link RequestHeaderArgumentResolver}. * Unit tests for {@link RequestHeaderArgumentResolver}.
* *
* @author Olga Maciaszek-Sharma * @author Olga Maciaszek-Sharma
*/ */
@ -42,104 +43,94 @@ class RequestHeaderArgumentResolverTests {
private final Service service = this.clientAdapter.createService(Service.class); private final Service service = this.clientAdapter.createService(Service.class);
@Test @Test
void shouldResolveSingleValueRequestHeader() { void stringHeader() {
this.service.executeString("test"); this.service.executeString("test");
assertRequestHeaders("id", "test"); assertRequestHeaders("id", "test");
} }
@Test @Test
void shouldResolveRequestHeaderWithNameFromAnnotationName() { void objectHeader() {
this.service.executeNamed("test");
assertRequestHeaders("id", "test");
}
@Test
void shouldResolveRequestHeaderNameFromValue() {
this.service.executeNamedWithValue("test");
assertRequestHeaders("test", "test");
}
@Test
void shouldResolveObjectValueRequestHeader() {
this.service.execute(Boolean.TRUE); this.service.execute(Boolean.TRUE);
assertRequestHeaders("id", "true"); assertRequestHeaders("id", "true");
} }
@Test @Test
void shouldResolveListRequestHeader() { void namedHeader() {
this.service.executeNamed("test");
assertRequestHeaders("id", "test");
}
@Test
void listHeader() {
this.service.execute(List.of("test1", Boolean.TRUE, "test3")); this.service.execute(List.of("test1", Boolean.TRUE, "test3"));
assertRequestHeaders("id", "test1", "true", "test3"); assertRequestHeaders("multiValueHeader", "test1", "true", "test3");
} }
@Test @Test
void shouldResolveArrayRequestHeader() { void arrayHeader() {
this.service.execute("test1", Boolean.FALSE, "test3"); this.service.execute("test1", Boolean.FALSE, "test3");
assertRequestHeaders("id", "test1", "false", "test3"); assertRequestHeaders("multiValueHeader", "test1", "false", "test3");
} }
@Test @Test
void shouldResolveRequestHeadersFromMap() { void mapHeader() {
this.service.executeMap(Maps.of(Boolean.TRUE, "true", Boolean.FALSE, "false")); this.service.executeMap(Maps.of("header1", "true", "header2", "false"));
assertRequestHeaders("true", "true"); assertRequestHeaders("header1", "true");
assertRequestHeaders("false", "false"); assertRequestHeaders("header2", "false");
} }
@Test @Test
void shouldThrowExceptionWhenRequiredHeaderNull() { void mapHeaderNull() {
assertThatIllegalArgumentException()
.isThrownBy(() -> this.service.executeString(null));
}
@Test
void shouldIgnoreNullWhenHeaderNotRequired() {
this.service.executeNotRequired(null);
assertThat(getActualHeaders().get("id")).isNull();
}
@Test
void shouldIgnoreNullMapValue() {
this.service.executeMap(null); this.service.executeMap(null);
assertThat(getActualHeaders()).isEmpty(); assertThat(getActualHeaders()).isEmpty();
} }
@Test @Test
void shouldResolveRequestHeaderFromOptionalArgumentWithConversion() { void mapWithOptional() {
this.service.executeOptional(Optional.of(Boolean.TRUE));
assertRequestHeaders("id", "true");
}
@Test
void shouldResolveRequestHeaderFromOptionalArgument() {
this.service.executeOptional(Optional.of("test"));
assertRequestHeaders("id", "test");
}
@Test
void shouldThrowExceptionForEmptyOptional() {
assertThatIllegalArgumentException().isThrownBy(() -> this.service.execute(Optional.empty()));
}
@Test
void shouldIgnoreEmptyOptionalWhenNotRequired() {
this.service.executeOptionalNotRequired(Optional.empty());
assertThat(getActualHeaders().get("id")).isNull();
}
@Test
void shouldResolveRequestHeaderFromOptionalMapValue() {
this.service.executeOptionalMapValue(Map.of("id", Optional.of("test"))); this.service.executeOptionalMapValue(Map.of("id", Optional.of("test")));
assertRequestHeaders("id", "test"); assertRequestHeaders("id", "test");
} }
@Test @Test
void shouldReplaceNullValueWithDefaultWhenAvailable() { void nullHeaderRequired() {
assertThatIllegalArgumentException().isThrownBy(() -> this.service.executeString(null));
}
@Test
void nullHeaderNotRequired() {
this.service.executeNotRequired(null);
assertThat(getActualHeaders().get("id")).isNull();
}
@Test
void optional() {
this.service.executeOptional(Optional.of("test"));
assertRequestHeaders("id", "test");
}
@Test
void optionalWithConversion() {
this.service.executeOptional(Optional.of(Boolean.TRUE));
assertRequestHeaders("id", "true");
}
@Test
void optionalEmpty() {
this.service.executeOptional(Optional.empty());
assertThat(getActualHeaders().get("id")).isNull();
}
@Test
void defaultValueWithNull() {
this.service.executeWithDefaultValue(null); this.service.executeWithDefaultValue(null);
assertRequestHeaders("id", "default"); assertRequestHeaders("id", "default");
} }
@Test @Test
void shouldReplaceEmptyOptionalValueWithDefaultWhenAvailable() { void defaultValueWithOptional() {
this.service.executeOptionalWithDefaultValue(Optional.empty()); this.service.executeOptionalWithDefaultValue(Optional.empty());
assertRequestHeaders("id", "default"); assertRequestHeaders("id", "default");
} }
@ -152,47 +143,43 @@ class RequestHeaderArgumentResolverTests {
return this.clientAdapter.getRequestValues().getHeaders(); return this.clientAdapter.getRequestValues().getHeaders();
} }
@SuppressWarnings("OptionalUsedAsFieldOrParameterType") @SuppressWarnings("OptionalUsedAsFieldOrParameterType")
private interface Service { private interface Service {
@GetExchange @GetExchange
void executeString(@Nullable @RequestHeader String id); void executeString(@Nullable @RequestHeader String id);
@GetExchange
void executeNotRequired(@Nullable @RequestHeader(required = false) String id);
@GetExchange @GetExchange
void execute(@RequestHeader Object id); void execute(@RequestHeader Object id);
@GetExchange
void execute(@RequestHeader List<Object> id);
@GetExchange
void execute(@RequestHeader Object... id);
@GetExchange
void executeMap(@Nullable @RequestHeader Map<Object, String> id);
@GetExchange
void executeOptionalMapValue(@RequestHeader Map<Object, Optional<String>> id);
@GetExchange
void executeOptional(@RequestHeader Optional<Object> id);
@GetExchange
void executeOptionalNotRequired(@RequestHeader(required = false) Optional<String> id);
@GetExchange
void executeNamedWithValue(@Nullable @RequestHeader(name = "id", value = "test") String employeeId);
@GetExchange @GetExchange
void executeNamed(@RequestHeader(name = "id") String employeeId); void executeNamed(@RequestHeader(name = "id") String employeeId);
@GetExchange
void execute(@RequestHeader List<Object> multiValueHeader);
@GetExchange
void execute(@RequestHeader Object... multiValueHeader);
@GetExchange
void executeMap(@Nullable @RequestHeader Map<String, String> id);
@GetExchange
void executeOptionalMapValue(@RequestHeader Map<String, Optional<String>> headers);
@GetExchange
void executeNotRequired(@Nullable @RequestHeader(required = false) String id);
@GetExchange
void executeOptional(@RequestHeader Optional<Object> id);
@GetExchange @GetExchange
void executeWithDefaultValue(@Nullable @RequestHeader(defaultValue = "default") String id); void executeWithDefaultValue(@Nullable @RequestHeader(defaultValue = "default") String id);
@GetExchange @GetExchange
void executeOptionalWithDefaultValue(@Nullable @RequestHeader(defaultValue = "default") Optional<Object> id); void executeOptionalWithDefaultValue(@RequestHeader(defaultValue = "default") Optional<Object> id);
} }
} }