Bài viết được sự cho phép của tác giả Giang Phan
Trong các bài viết trước chúng ta sử dụng thư viện Jersey client, OkHttp, Retrofit để gọi các RESTful API. Trong bài này, tôi sẽ giới thiệu với các bạn thư viện khác là Feign. Thư viện này giúp chúng ta dễ dàng hơn nữa trong phát triển ứng dụng Rest Client.
Giới thiệu Feign
Feign là một HTTP client cho Java, được phát triển bởi Netflix. Mục tiêu của Fiegn là giúp đơn giản hóa HTTP API Client.
Tương tự với các thư viện khác, Feign giúp bạn dễ dàng xử lý dữ liệu JSON hoặc XML sau đó phân tích cú pháp thành Plain Old Java Objects (POJOs). Tất cả các yêu cầu GET, POST, PUT, và DELETE đều có thể được thực thi.
Feign được xây dựng dựa trên một số thư viện mạnh mẽ và công cụ khác để xử lý các request/ response trên mạng bao gồm OkHttp, JAX-RS, Gson, Jackson, JAXB, Ribbon, Hystrix, SOAP, … Các bạn xem thêm các thư viện khác: https://mvnrepository.com/artifact/io.github.openfeign
Feign hỗ trợ một số tính năng mạnh mẽ khác như: Error Handling, Retry, hỗ rợ default method, static method với interface trong java 8.
Xem thêm tuyển dụng Java lương cao trên TopDev
Sử dụng Feign
Ý tưởng của Feign tương tự như Retrofit là sử dụng interface và các annotation để định nghĩa các phương thức request đến API. Với Retrofit, chúng ta còn gặp một chút phiền phức khi phải gọi xử lý Call<Respone>. Sử dụng Feign chúng ta sẽ không thấy sự khác biệt giữa call các method trong interface thông thường và call thông qua Feign.
Để sử dụng Feign chúng ta thực hiện các bước sau:
- Một class object tương ứng với JSON/ XML data.
- Một interface dùng để định nghĩa các các phương thức request đến API.
- Sử dụng Annotations để mô tả yêu cầu HTTP.
- Tạo một Feign.builder() để khởi tạo các phương thức trong interface đã được định nghĩa.
Các Annotations để mô tả yêu cầu HTTP
Request method
Mỗi phương thức phải có Annotation HTTP cung cấp request method và URL. Feign sử dụng @RequestLine để mô tả các thông tin này.
@RequestLine("GET /api/v1/users?sort=desc") List getUsers();
Header manipulation
Chúng ta có thể set thông tin static header bằng cách sử dụng annotation @Header ở mức method.
Đối với các kết hợp tham số truy vấn phức tạp, có thể sử dụng @HeaderMap.
@RequestLine("GET /api/v1/users?sort=desc") List getUsers(@HeaderMap Map<String, String> headers);
Url manipulation
URL request có thể được cập nhật tự động bằng cách sử dụng các khối thay thế và tham số trên phương thức.
Chúng ta có thể sử dụng URL 1 cách động dựa vào biến truyền vào, bằng cách sử dụng anotation @Path.
@RequestLine("GET /api/v1/users/{id}") User getUser(@Path("id") int userId);
Đối với các kết hợp tham số truy vấn phức tạp, có thể sử dụng @QueryMap.
@RequestLine("GET /api/v1/users?page=1&limit=10&sortBy=createdAt&order=desc") List getUsers(@QueryMap Map<String, String> options);
Request body
Một đối tượng có thể được chỉ định để sử dụng làm phần thân yêu cầu HTTP với Annotation @Body.
@RequestLine("POST /api/v1/users") @Headers("Content-Type: application/json") @Body("%7B"username": "{gpcoder}", "password": "{gpcoder}"%7D") Call createUser(@Body User user);
Form encoded and Multipart
Để gửi dữ liệu form: application/x-www-form-urlencoded và multipart/form-data chúng ta cần sử dụng thêm thư viện feign-form.
Chúng ta cần cung cấp Content-Type trong @Headers và field name trong @Field.
@RequestLine("POST /api/v1/auth") @Headers("Content-Type: application/x-www-form-urlencoded") Call getToken(@Field("username") String username, @Field("password") String password);
Các yêu cầu multipart được sử dụng khi @Multipart xuất hiện trên phương thức. Các phần được khai báo bằng cách sử dụng @Part. @Multipart thường được sử dụng để truyền tải file.
@Multipart @POST("/api/v1/files/upload") Call uploadFile(@Part("uploadFile") RequestBody uploadFile, @Part("description") RequestBody description); @RequestLine("POST /send_photo") Headers("Content-Type: multipart/form-data") void sendPhoto (@Param("isPublic") Boolean isPublic, @Param("photo") FormData photo);
Ví dụ CRUD Restful Client với Feign
Tạo project
Tạo maven project và khai báo dependency sau trong file pom.xml.
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
|
< project xmlns = "http://maven.apache.org/POM/4.0.0" xmlns:xsi = "http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation = "http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd" > < modelVersion >4.0.0</ modelVersion > < groupId >com.gpcoder</ groupId > < artifactId >RestfulClientWithFeignExample</ artifactId > < version >0.0.1-SNAPSHOT</ version > < packaging >jar</ packaging > < name >RestfulClientWithFeignExample</ name > < properties > < project.build.sourceEncoding >UTF-8</ project.build.sourceEncoding > < maven.compiler.source >1.8</ maven.compiler.source > < maven.compiler.target >1.8</ maven.compiler.target > < lombok.version >1.16.20</ lombok.version > </ properties > < dependencies > < dependency > < groupId >io.github.openfeign</ groupId > < artifactId >feign-okhttp</ artifactId > < version >10.2.3</ version > </ dependency > < dependency > < groupId >io.github.openfeign</ groupId > < artifactId >feign-gson</ artifactId > < version >10.2.3</ version > </ dependency > < dependency > < groupId >io.github.openfeign</ groupId > < artifactId >feign-slf4j</ artifactId > < version >10.2.3</ version > </ dependency > <!-- Support for encoding application/x-www-form-urlencoded and multipart/form-data form --> < dependency > < groupId >io.github.openfeign.form</ groupId > < artifactId >feign-form</ artifactId > < version >3.8.0</ version > </ dependency > < dependency > < groupId >org.projectlombok</ groupId > < artifactId >lombok</ artifactId > < version >${lombok.version}</ version > < scope >provided</ scope > </ dependency > </ dependencies > </ project > |
Trong project này, tôi sử dụng thư viện gson để convert request/ response data giữa client và server.
Tạo CRUD Restful Client với Feign
Trong ví dụ này, chúng ta sẽ gọi lại các Restful API chúng ta đã tạo ở bài viết trước “JWT – Token-based Authentication trong Jersey 2.x“.
Đầu tiên, chúng ta cần gọi API /auth để lấy token và sau đó chúng ta sẽ attach token này vào mỗi request để truy cập resource.
AuthService.java
OrderService.java
Tương tự Retrofit, chúng ta có thể sử dụng Interceptor – một tính năng của OkHttp để tự động thêm Authentication Token vào mỗi request, chúng ta không cần thêm nó một cách thủ công trong từng request.
Có một vài cách để gửi Authentication Header token lên server: sử dụng @Header, tạo custom Target, sử dụng Interceptor của Feign, sử dụng Interceptor của OkHttp.
Ví dụ bên dưới sử dụng 2 Interceptor:
- AuthInterceptor : thêm Token vào mỗi request.
- LoggingInterceptor : log trước khi gửi request và sau khi nhận response.
AuthInterceptor.java
Tương tự chúng ta sẽ tạo LoggingInterceptor:
LoggingInterceptor.java
Để sử dụng Interceptor, chúng ta cần đăng ký với Client thông qua phương thức addInterceptor() hoặc addNetworkInterceptor(). Chương trình bên dưới, tôi đăng ký AuthInterceptor ở mức Application và LoggingInterceptor cho cả 2 mức Application và Network.
Chúng ta sẽ tạo một lớp hỗ trợ cấu hình Feign, đăng ký các Interceptor và tạo instance của Feign service.
FeignClientCreator.java
Tiếp theo chúng ta sẽ sử dụng Feign service để call các API.
FeignClientExample.java
Như bạn thấy, Feign là một thư viện rất mạnh mẽ nhưng rất đơn giản để sử dụng. Hy vọng bài viết giúp ích cho các bạn, hẹn gặp lại ở các bài viết tiếp theo.
Tài liệu tham khảo:
- https://github.com/OpenFeign/feign/
- https://github.com/OpenFeign/feign-form
- https://www.ctolib.com/feign.html
Bài viết gốc được đăng tải tại gpcoder.com
Có thể bạn quan tâm:
- Giới thiệu Swagger – Công cụ document cho RESTfull APIs
- Xây dựng ứng dụng Client-Server với Socket trong Java
- Định nghĩa request URL trong ứng dụng Jakarta EE RESTful Web Services
Xem thêm công việc CNTT hấp dẫn trên TopDev