
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가 필터를 적용시켜 값을 반환
