Spring Cloud OpenFeign无法创建动态查询参数

问题描述 投票:4回答:2

Spring cloud openFeign无法创建动态查询参数。它抛出异常,因为SpringMvcContract尝试查找不存在的RequestParam值属性。

java.lang.IllegalStateException: RequestParam.value() was empty on parameter 0


@RequestMapping(method = RequestMethod.GET, value = "/orders")
Pageable<Order> searchOrder2(@RequestParam CustomObject customObject);

我尝试使用@QueryMap而不是@RequestParam,但@QueryMap不会生成查询参数。

顺便说一句@RequestParam Map<String, Object> params方法参数工作正常,生成动态查询参数。

但是我想使用自定义对象,假装客户端可以从对象的属性生成动态查询参数。

谢谢

spring spring-mvc spring-boot spring-cloud spring-cloud-feign
2个回答
4
投票

来自Spring Cloud OpenFeign Docs

Spring Cloud OpenFeign提供等效的@SpringQueryMap注释,用于将POJO或Map参数注释为查询参数映射

所以你的代码应该是:

@RequestMapping(method = RequestMethod.GET, value = "/orders")
Pageable<Order> searchOrder2(@SpringQueryMap @ModelAttribute CustomObject customObject);

3
投票

spring-cloud-starter-feign有一个open issue用于支持pojo对象作为请求参数。因此我使用了一个请求拦截器,它从feign方法获取对象,并从其字段创建url的查询部分。感谢@charlesvhe

public class DynamicQueryRequestInterceptor implements RequestInterceptor {

private static final Logger LOGGER = LoggerFactory.getLogger(DynamicQueryRequestInterceptor.class);

private static final String EMPTY = "";

@Autowired
private ObjectMapper objectMapper;

@Override
public void apply(RequestTemplate template) {
    if ("GET".equals(template.method()) && Objects.nonNull(template.body())) {
        try {
            JsonNode jsonNode = objectMapper.readTree(template.body());
            template.body(null);

            Map<String, Collection<String>> queries = new HashMap<>();
            buildQuery(jsonNode, EMPTY, queries);
            template.queries(queries);
        } catch (IOException e) {
            LOGGER.error("IOException occurred while try to create http query");
        }
    }
}

private void buildQuery(JsonNode jsonNode, String path, Map<String, Collection<String>> queries) {
    if (!jsonNode.isContainerNode()) {
        if (jsonNode.isNull()) {
            return;
        }
        Collection<String> values = queries.computeIfAbsent(path, k -> new ArrayList<>());
        values.add(jsonNode.asText());
        return;
    }
    if (jsonNode.isArray()) {
        Iterator<JsonNode> it = jsonNode.elements();
        while (it.hasNext()) {
            buildQuery(it.next(), path, queries);
        }
    } else {
        Iterator<Map.Entry<String, JsonNode>> it = jsonNode.fields();
        while (it.hasNext()) {
            Map.Entry<String, JsonNode> entry = it.next();
            if (StringUtils.hasText(path)) {
                buildQuery(entry.getValue(), path + "." + entry.getKey(), queries);
            } else {
                buildQuery(entry.getValue(), entry.getKey(), queries);
            }
        }
    }
}

}

© www.soinside.com 2019 - 2024. All rights reserved.