Accept language header
를 통해 국제화 -i18n 을 해보자
만들어진 RestAPI 가 전세계 사용자들이 사용한다고 하였을 때 customize 하려면 어떻게 해야할까?
✔ : i18n(18개의 언어 Internationalization)
가령 다음과 같은 메시지를 반환하는 컨트롤러에 국제화를 적용해보고자 한다면?
Spring 과 부트에서 국제화를 다루는 정석적인 방법은 파일들을 messages.properties 로 정의하는것이다
다음과 같이 리소스 파일 안에 messages.properties 를 생성한다 (파일명이 틀리면 스프링이 인식하지 못함)
프로퍼티에서 적용한 메시지를 사용하고자 한다면 MessageSource
를 사용해주어야한다
이는 메시지의 파라미터화, 국제화를 지원하는 인터페이스이다
MessageSource
에서 메시지를 받아오고자 할 때 getMessage 를 통하여 메시지를 받아올 수 있다
Args : any variables to replace
Default Message : 기본 메시지
Locale : 사용자 로케일 정보
- 만약 사용자가 Accept header 를 주었다면 해당 로케일 정보가 locale 에 담김
Properties 에서 정의한 메시지가 잘 나오는 것을 확인
이제 다음 언어로 넘어가 국제화를 진행, messages_{i18n코드}.properties 를 생성
한글이 깨져요?
- Configuration을 통한 메시지소스의 set Encoding 을 UTF-8 으로 바꾸기
- 헤더의 Accept-Charset 을 통하여 UTF-8로 설정해보기
- 파일 인코딩 변경
- application.properties 에서 인코딩 설정하기
저의 경우 전역 인코딩은 UTF-8 로 되어있지만 프로퍼티 파일에 대한 인코딩은 ISO-8859-1 이여서 인코딩에 실패하였었던 것
만약 RestAPI 의 대폭적인 변화가 이루어졌을 때
→ 버전 업
버전 관리의 방법
1. url 의 변경 (ex: api/v1/user → api/v2/user)
2. Request Parameter
3. Header
4. Media Type
V2 에서는 Name 이라는 객체를 추가적으로 생성하여 성과 이름을 구분할 수 있도록 하였다
다음은 아마존에서 주로 사용하는 쿼리파라미터를 통한 버전관리이다
아니라면 하나의 컨트롤러에서 @RequestParam 을 통하여 version 값에 따른 변경을 해주어도 가능하다.
이 점은 취향 차이?
네가지 방법중 선택하게 될 때에 고려해야 할 점은 다음과 같다
🤔 완벽한 솔루션은 없지만 1 번이 좋지 않을까 한다. Uri pollution 이 최소화 되기도 하고 다른 문제들에 해당되지 않기 때문이다. 직관적이고 사용성이 좋다고나 할까?
✔ 다만 필수적으로 하나의 기업에서는 같은 버저닝을 통하여 일관성을 유지하여야 한다! 사용자에게도 개발자의 유지보수에도 이는 권장된다
하이퍼미디어를 애플리케이션의 상태를 관리하기 위한 메커니즘으로 사용
아니 대체 무슨 뜻…?
Rest API
를 사용하는 클라이언트가 전적으로 서버와 동적인 상호작용이 가능하도록 하는것부트에서는 간단하게 디펜던시를 추가하여 구현이 가능하다 (POM.XML)
데이터와 링크를 같이 보내기 위해서는 EntityModel
과 WebMvcLinkBuilder
를 이해할 필요가 있다
WebMvcLinkBuilder
의 methodOn
을 통하여 타겟 메서드를 가르키도록 하고 linkTo
를 이용하여 링크를 만들어준다
HAL 을 이용하여 HATEOS를 구현
자바 빈
을 JacksonConverter
를 통하여 json 객체
로 만들어 반환하게 되는데 만약 반환된 값중 제외시키고자 하는 필드가 있다면 어떻게 할까?
🤔 가령 id 와 같은 index 들은 사용자에게 제공하지 않는 편이 좋을 것이다 (보안상의 문제.. sql injection 이라도 하면 어쩌겠는가)
이러한 문제를 해결하기 위해 커스터 마이즈가 필요하다
다음과 같은 빈이 있을 때 @JsonIgnore 를 붙여준 필드는 제외시킬 수 있다
그렇다면 리스트에서는 어떨까?
리스트에서도 정상적으로 필드가 제외됨을 알 수 있다
@JsonIgnoreProperties
를 통하여 클래스에서 필드값들을 지정하여 제외시킬 수도 있다
같은 빈에 대하여 각각의 컨트롤러에서 다른 값들을 보내고 싶다면
가령 field1, 2, 3 가 있다했을 때
첫번째 컨트롤러에서는 field 2를 제외시키고
두번째 컨트롤러에서는 field 2와 3을 제외시키고 싶다면?
MappingJacksonValue
: json으로 매핑될때에 필터를 걸어 특정 필터만 매핑되도록 할 수 있음
MappingJacksonValue 를 통해 빈을 넣고 매핑할 객체를 지정
SimpleBeanPropertyFilter 를 통하여 제외 시키고자 하는 필드를 지정
FilterProvider 를 통해 세롭게 필터를 생성
MappingJacksonValue 에 생성한 필드를 세팅
JacksonConverter가 필터를 적용시켜 값을 반환