Using spring-mvc
annotations:
- How can I define an
@FeignClient
that canPOST
form-url-encoded
?
Using spring-mvc
annotations:
@FeignClient
that can POST
form-url-encoded
?Use FormEncoder for Feign:
And your Feign configuration can look like this:
class CoreFeignConfiguration {
@Autowired
private ObjectFactory<HttpMessageConverters> messageConverters
@Bean
@Primary
@Scope(SCOPE_PROTOTYPE)
Encoder feignFormEncoder() {
new FormEncoder(new SpringEncoder(this.messageConverters))
}
}
Then, the client can be mapped like this:
@FeignClient(name = 'client', url = 'localhost:9080', path ='/rest',
configuration = CoreFeignConfiguration)
interface CoreClient {
@RequestMapping(value = '/business', method = POST,
consumes = MediaType.APPLICATION_FORM_URLENCODED)
@Headers('Content-Type: application/x-www-form-urlencoded')
void activate(Map<String, ?> formParams)
}
Full Java code with a simplified version of kazuar solution, works with Spring Boot:
import java.util.Map;
import feign.codec.Encoder;
import feign.form.spring.SpringFormEncoder;
import org.springframework.beans.factory.ObjectFactory;
import org.springframework.boot.autoconfigure.http.HttpMessageConverters;
import org.springframework.cloud.openfeign.FeignClient;
import org.springframework.cloud.openfeign.support.SpringEncoder;
import org.springframework.context.annotation.Bean;
import org.springframework.http.ResponseEntity;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestBody;
import static org.springframework.http.MediaType.APPLICATION_FORM_URLENCODED_VALUE;
@FeignClient(name = "srv", url = "http://s.com")
public interface Client {
@PostMapping(value = "/form", consumes = APPLICATION_FORM_URLENCODED_VALUE)
void login(@RequestBody Map<String, ?> form);
class Configuration {
@Bean
Encoder feignFormEncoder(ObjectFactory<HttpMessageConverters> converters) {
return new SpringFormEncoder(new SpringEncoder(converters));
}
}
}
Dependency:
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-openfeign</artifactId>
</dependency>
@Bean
for Decoder
return new GsonDecoder();
–
Boeotia @Bean Gson upbeatGson() { return new GsonBuilder().setFieldNamingPolicy(FieldNamingPolicy.LOWER_CASE_WITH_DASHES).create();}
- hence I mentioned. Otherwise, if the JSONDecoder - default version works, there is no need. –
Boeotia @RequestParam
still go to the URL, only @RequestBody Map<String, ?>
seems to be working. –
Earshot Map<String, ?>
do work, and putting an POJO entity with the proper fields there works as well. –
Incision @FeignClient(configuration = Client.Configuration.class)
for this to work. The answer should be edited –
Sackett Just to complement accepted answer, one can also use POJO instead of Map<String, ?>
in order to pass form parameters to feign client:
@FeignClient(configuration = CustomConfig.class)
interface Client {
@PostMapping(
path = "/some/path",
consumes = MediaType.APPLICATION_FORM_URLENCODED_VALUE)
void postComment(CommentFormDto formDto);
...
}
...
@Configuration
class CustomConfig {
@Bean
Encoder formEncoder() {
return new feign.form.FormEncoder();
}
}
...
class CommentFormDto {
private static String willNotBeSerialized;
private final Integer alsoWillNotBeSerialized;
@feign.form.FormProperty("author_id")
private Long authorId;
private String message;
@feign.form.FormProperty("ids[]")
private List<Long> ids;
/* getters and setters omitted for brevity */
}
That will result in request with body that looks something like this:
author_id=42&message=somemessage&ids[]=1&ids[]=2
@FormProperty
annotation allows to set custom field names; please note that static or final fields of POJO, along with inherited ones, will not be serialized as form content.
for Kotlin:
import org.springframework.http.MediaType
@FeignClient(configuration = [CustomConfig::class])
interface Client {
@PostMapping(
path = "/some/path",
consumes = [MediaType.APPLICATION_FORM_URLENCODED_VALUE])
postComment(CommentFormDto formDto): responseDto
...
}
...
import feign.form.FormEncoder
@Configuration
class CustomConfig {
@Bean
fun formEncoder(): FormEncoder {
return FormEncoder()
}
}
...
import feign.form.FormProperty
data class CommentFormDto (
@FormProperty("author_id")
var authorId: Long
@FormProperty("ids[]")
var ids: List<Long>
)
@FormProperty
is what I was looking for! +1 –
Postman @FormProperty
instead @FeignProperty
, but I can't edit it. –
Drongo Just an additional contribution... It is possible to use a Spring abstraction, the org.springframework.util.MultiValueMap
, which has no other dependencies (only spring-cloud-starter-openfeign).
@PostMapping(value = "/your/path/here",
consumes = MediaType.APPLICATION_FORM_URLENCODED_VALUE)
YourDtoResponse formUrlEncodedEndpoint(MultiValueMap<String, Object> params);
The syntax of this useful data structure is quite simple, as shown below:
MultiValueMap<String, Object> params = new LinkedMultiValueMap<>();
params.add("param1", "Value 1");
params.add("param2", 2);
This worked for me
@FeignClient(name = "${feign.repository.name}", url = "${feign.repository.url}")
public interface LoginRepository {
@PostMapping(value = "${feign.repository.endpoint}", consumes = APPLICATION_FORM_URLENCODED_VALUE)
LoginResponse signIn(@RequestBody Map<String, ?> form);
}
form is Map<String, Object> loginCredentials = new HashMap<>();
You must use FormEncoder
in Feign encoder for url-form-encoded data in POST.
Include the dependency to your app:
Maven:
<dependency>
<groupId>io.github.openfeign.form</groupId>
<artifactId>feign-form</artifactId>
<version>3.8.0</version>
</dependency>
Add FormEncoder to your Feign.Builder like so:
SomeFeign sample = Feign.builder()
.encoder(new FormEncoder(new JacksonEncoder()))
.target(SomeFeign.class, "http://sample.test.org");
In the Feign interface
@RequestLine("POST /submit/form")
@Headers("Content-Type: application/x-www-form-urlencoded")
void from (@Param("field1") String field1, @Param("field2") String field2);
Ref for more info: https://github.com/OpenFeign/feign-form
Tested in Kotlin: What worked for me was the following:
@Configuration
class FeignFormConfiguration {
@Bean
fun multipartFormEncoder(): Encoder {
return SpringFormEncoder(SpringEncoder {
HttpMessageConverters(
RestTemplate().messageConverters
)
})
}
}
@FeignClient(
value = "client",
url = "localhost:9091",
configuration = [FeignFormConfiguration::class]
)
interface CoreClient {
@RequestMapping(
method = [RequestMethod.POST],
value = ["/"],
consumes = [MediaType.APPLICATION_FORM_URLENCODED_VALUE],
produces = ["application/json"]
)
fun callback(@RequestBody form: Map<String, *>): AnyDTO?
}
// ----- other code --------
@Autowired
private lateinit var coreClient: CoreClient
fun methodName() {
coreClient.callback(form)
}
// ----- other code --------
For the Feign.Builder, mine worked without the JacksonEncoder, just the Feign FormEncoder:
Add FormEncoder to your Feign.Builder:
SomeFeign sample = Feign.builder()
.encoder(new FormEncoder()) <==difference here
.target(SomeFeign.class, "http://sample.test.org");
The feign dependencies I added to pom.xml:
<dependency>
<groupId>io.github.openfeign</groupId>
<artifactId>feign-core</artifactId>
<version>11.8</version>
</dependency>
<dependency>
<groupId>io.github.openfeign</groupId>
<artifactId>feign-jackson</artifactId>
<version>11.8</version>
</dependency>
<dependency>
<groupId>io.github.openfeign.form</groupId>
<artifactId>feign-form</artifactId>
<version>3.8.0</version>
</dependency>
Parent in pom.xml is:
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>2.6.2</version>
<relativePath/> <!-- lookup parent from repository -->
</parent>
The feign interface as Ramanan gave:
@RequestLine("POST /submit/form")
@Headers("Content-Type: application/x-www-form-urlencoded")
void from (@Param("field1") String field1, @Param("field2") String field2);
public class CoreFeignConfiguration {
@Autowired
private ObjectFactory<HttpMessageConverters> messageConverters;
@Bean
@Primary
@Scope("prototype")
Encoder feignFormEncoder() {
return new FormEncoder(new SpringEncoder(this.messageConverters));
}
@Bean
Logger.Level feignLoggerLevel() {
return Logger.Level.FULL;
}
@Bean
Logger logger() {
return new MyLogger();
}
private static class MyLogger extends Logger {
@Override
protected void log(String s, String s1, Object... objects) {
System.out.println(String.format(s + s1, objects)); // Change me!
}
}
}
In interface @FeignClient(name = "resource-service1", url = "NOT_USED", configuration = CoreFeignConfiguration.class)
public interface TestFc {
@RequestMapping( method=RequestMethod.POST,consumes = "application/x-www-form-urlencoded")
//@Headers("Content-Type: application/x-www-form-urlencoded")
ResponseEntity<String> test(URI baseUrl,
Map<String,?> request);
}
© 2022 - 2024 — McMap. All rights reserved.
Map<String, ?> formParams
, the question mark is required. – Superhuman