ํ์คํ ์น๐ ๊ฐ๋ฐ์ ์ง๋ง์ ๐ง๐ฝโ๐ป
โ ์ธ๊ณต์ง๋ฅ ๊ด์ฌ ๐ค
Categories
-
โฃ
โถ COMPUTER_SCIENCE
๐: 7 -
โฃ
โถ WEB
๐: 3 -
โฃ
โถ ETC
๐: 3-
โ
โฃ
ETCS
๐: 10 -
โ
โฃ
SUBBRAIN ๊ฐ๋ฐ๊ธฐ
๐: 5 -
โ
โ
YOS ๊ฐ๋ฐ๊ธฐ
๐: 1
-
โ
โฃ
-
โ
โถ AI
๐: 9-
โฃ
AITOOLS
๐: 3 -
โฃ
CV
๐: 2 -
โฃ
DEEP_LEARNING
๐: 1 -
โฃ
DATA_VIS
๐: 2 -
โฃ
GRAPH
๐: 1 -
โฃ
LIGHTWEIGHT
๐: 1 -
โฃ
MATH
๐: 1 -
โฃ
NLP
๐: 3 -
โ
STRUCTURED_DATA
๐: 2
-
โฃ
Spring5-HATEOAS
- HATEOAS๋?
- ์ง์ ํ์ดํผ๋งํฌ ์ถ๊ฐํ๊ธฐ
- ๋ฆฌ์์ค ์ด์ ๋ธ๋ฌ๋ฅผ ํตํ ํ์ดํผ๋งํฌ ์ถ๊ฐ
Spring5-HATEOAS
_ ์ด๋ณด ์น ๊ฐ๋ฐ์๋ฅผ ์ํ ์คํ๋ง 5 ํ๋ก๊ทธ๋๋ฐ ์ ๋ฌธ _์ ์คํ๋ง ์ธ ์ก์ ์ ๋ด์ฉ์ ๋ฐํ์ผ๋ก ์ ๋ฆฌํ ๋ด์ฉ์ ๋๋ค.
HATEOAS๋?
๋ง์ฝ ๋ฐฑ์๋ API์ ๊ตฌ์กฐ๊ฐ ๋ฐ๋๊ฒ ๋๋ค๋ฉด, ํด๋ผ์ด์ธํธ ์ฝ๋๊ฐ ๊ณ ์ฅ์ด ๋ ๊ฒ์ด๋ค.
HATEOAS(Hypermedia As The Engine Of Application)๋ API์์ ์๋ต ์, ๋ฐํ๋๋ ๋ฆฌ์์ค์ ๊ด๋ จ๋ ํ์ดํผ๋งํฌ๋ฅผ ์ ๋ณด์ ํฌํจํ์ฌ ํด๋ผ์ด์ธํธ ์ธก์์ ์ด๋ฅผ ํตํด ๋ค์ ์์ฒญ์ ์ฒ๋ฆฌํ๋๋ก ํ๋ค.
HATEOAS๊ฐ JSON ์๋ต์ ํ์ดํผ๋งํฌ๋ฅผ ํฌํจ์ํค๋ ํ์์ HAL(Hypertext Application Language)์ด๋ผ๊ณ ํ๋ค.
// REST API
[
{
"id": 4,
"name": "Veg-Out",
"createdAt": "2018-01-31T20:15:53.219+0000",
"ingredients": [
{"id": "FLTO", "name": "Flour Tortilla", "type": "WRAP"},
{"id": "TMTO", "name": "Diced Tomatoes", "type": "VEGGIES"},
...
]
},
]
// HATEOAS API
{
"_embedded": {
"tacoResurceList": [
{
"name": "Veg-Out",
"createdAt": "2018-01-31T20:15:53.219+0000",
"ingredients": [
{
"name": "Flour Tortilla", "type": "WRAP",
"_links": {
"self": {"href": "http://localhost:8080/ingredieints/FLTO" }
}
},
{
"name": "Diced Tomatoes", "type": "VEGGIES",
"_links": {
"self": {"href": "http://localhost:8080/ingredieints/TMTO" }
}
},
],
"_links": {
"self": {"href": "http://localhost:8080/design/4"}
}
}
]
},
"_links": {
"recents": {
"href": "http://localhost:8080/design/recent"
}
}
}
๋ด๋ถ์ "_links"
์์ฑ์ ํฌํจํ๋ฉฐ, ๊ด๋ จ API๋ช
๊ณผ URL์ ํฌํจํ๋ฉฐ, ์ด๋ฅผ ํตํด ํด๋ผ์ด์ธํธ๊ฐ ์์ฒญ์ ์ํํ๋ค.
์คํ๋ง ๋ถํธ์์ HATEOAS๋ฅผ ๊ตฌ์ถํ๋๋ฐ ๋์์ด ๋๋ ๋ชจ๋๊ณผ ํด๋์ค, ๋ฆฌ์์ค ์ด์ ๋ธ๋ฌ๋ฅผ ์ ๊ณตํ๋ค.
- org.springframework.boot.spring-boot-starter-hateoas ๋ชจ๋์ ์ถ๊ฐํ๋ค.
์ง์ ํ์ดํผ๋งํฌ ์ถ๊ฐํ๊ธฐ
์คํ๋ง HATEOAS๋ ํ์ดํผ๋งํฌ๋ฅผ ๋จ์ผ ๋งํฌ๋ฅผ ์ฒ๋ฆฌํ๋ Resource ํ์ ๊ณผ ๋ฆฌ์์ค ์ปฌ๋ ์ ์ ์ฒ๋ฆฌํ๋ Resources ํ์ ์ผ๋ก ๋ค๋ฃฌ๋ค.
์ด๋ค์ ํตํด ์๋ต์ ํ์ดํผ๋งํฌ๋ฅผ ์ถ๊ฐํ ์ ์๋ค.
๋ค์์ ๊ฐ์ฒด๋ฅผ ๋ฆฌํดํด์ผํ ๋๋ Resources
ํ์
์ ๊ฐ์ผ Resource
ํ์
์ ๋ฆฌํดํด์ผ ํ๋ค.
add()
๋ฉ์๋๋ฅผ ํตํด recentTacos()
์ ๋งคํ URL์ ์ง์ด๋ฃ๋ ์ฝ๋์ด๋ค.
import static org.springframework.hateoas.mvc.ControllerLinkBuilder
import org.springframework.hateoas.Resources;
import org.springframework.hateoas.Resource;
@GetMapping(path="/recent", produces="application/hal+json")
public Resources<Resource<Taco>> recentTacos() {
PageRequest page = PageRequest.of(0, 12, Sort.by("createdAt").descending());
List<Taco> tacos = tacoRepo.findAll(page).getContent();
// ๊ฒฐ๊ณผ๋ฅผ ๋ฆฌ์์ค ํ์
์ผ๋ก ๊ฐ์ธ๊ธฐ
Resources<Resource<Taco>> recentResources = Resources.wrap(tacos);
recentResources.add(
// new Link("http://localhost:8080/design/recent", "recents") // ์๋ฏธ์๋ ํ๋์ฝ๋ฉ ๋ฒ์
ControllerLinkBuilder.linkTo(DesignTacoController.class)
.slash("recent")
.withRel("recents"));
return recentResources;
}
ControllerLinkBuilder
๋ฅผ ํตํด URL์ ํ๋์ฝ๋ฉํ์ง ์๊ณ ํธ์คํธ ์ด๋ฆ์ ์ ์ ์์ผ๋ฉฐ, ๊ด๋ จ ๋งํฌ์ ๋น๋๋ฅผ ๋์์ฃผ๋ ํธ๋ฆฌํ API๋ฅผ ์ ๊ณตํ๋ค.linkTo()
๋ฉ์๋๋ฅผ ํตํด ํ์ฌ ์ปจํธ๋กค๋ฌ์ ๊ฒฝ๋ก๋ฅผ ์ง์ (๊ธฐ๋ณธ ๊ฒฝ๋ก ์ถ๊ฐ)slash(๋ด์ฉ)
๋ฉ์๋๋/๋ด์ฉ
๋ฌธ์์ด์ URL์ ์ถ๊ฐํ๋ค.(๋งคํ ๊ฒฝ๋ก ์ถ๊ฐ)withRel(๋งํฌ๋ช )
์ ํตํด ํด๋น ํ์ดํผ๋งํฌ์ API๋ช ์ ์ง์ ํ๋ค.
ํ์ง๋ง ์๋์ฒ๋ผ ๋ ์งง๊ณ ๋งคํ ๊ฒฝ๋ก๊น์ง ์ง์ ์ถ๊ฐํ์ง ์์๋ ๋๋ ์ฝ๋๊ฐ ์ข๋ค.
import static org.springframework.hateoas.mvc.ControllerLinkBuilder.*
Resources<Resource<Taco>> recentResources = Resources.wrap(tacos); recentResources.add(
linkTo(methodOn(DesignTacoController.class).recentTacos())
.withRel("recents"));
methodOn(์ปจํธ๋กค๋ฌ)
์ ์ปจํธ๋กค๋ฌ ๋ด๋ถ์ ๋ฉ์๋๋ฅผ ํธ์ถํ ์ ์๊ฒ ํด์ค๋ค.
๋ฆฌ์์ค ์ด์ ๋ธ๋ฌ๋ฅผ ํตํ ํ์ดํผ๋งํฌ ์ถ๊ฐ
์ ๋ฐฉ๋ฒ์ ์ง๊ด์ ์ด์ง๋ง, ์ฌ๋ฌ ๊ฐ์ฒด๋ฅผ ์ฒ๋ฆฌํด์ผํ ๋ ๋งค๋ฒ ๋ฃจํ๋ฅผ ์คํํ๋ฏ๋ก ๋ฒ๊ฑฐ๋กญ๋ค.
๊ฐ ๊ฐ์ฒด์ ํ์ดํผ๋งํฌ๋ฅผ ์ถ๊ฐํ๊ธฐ ์ํด ์ผ์ผ์ด Resources.wrap()
์ ์คํํ๋ ๋์ , ResourceSupport
๊ฐ์ฒด๋ฅผ ์์๋ฐ๋ ๋ฆฌ์์ค ๊ฐ์ฒด๋ฅผ ์์ฑํด ์๋์ผ๋ก ํ์ดํผ๋งํฌ๋ฅผ ์ถ๊ฐํ ์ ์๋ค.
๋ฆฌ์์ค ๊ฐ์ฒด ์์ฑ
ResourceSupoort
ํด๋์ค๋ ๋ฆฌ์์ค์ ๊ด๋ จ๋ Link
๊ฐ์ฒด๋ค๊ณผ ์ด๊ฒ๋ค์ ๊ด๋ฆฌํ๋ ๋ฉ์๋๋ฅผ ๊ฐ์ง๊ณ ์๋ค.
์ด๋, ๊ธฐ์กด์ ๋๋ฉ์ธ ๊ฐ์ฒด์ ์ญํ ๋ ๋ฆฌ์์ค ๊ฐ์ฒด๊ฐ ๋์์ ํ ์ ์๋๋ก ๋ง๋ค ์ ์์ง๋ง, ์๋ก ์ธ๋ชจ์๋ ๋ฐ์ดํฐ๋ฅผ ๊ฐ์ง๊ณ ์์ผ๋ฏ๋ก ๋๋๋ ๊ฒ์ด ์ข์ ์ ์๋ค.
- ๋๋ฉ์ธ ๊ฐ์ฒด๋ ๋๋ฉ์ธ์ id ๊ฐ์ ๊ฐ์ง๊ณ ์์ง๋ง ๋งํฌ ๊ฐ์ฒด๋ ํ์์๋ค.
- ๋ฆฌ์์ค ๊ฐ์ฒด๋ ๋งํฌ ๊ฐ์ฒด๋ฅผ ๊ฐ์ง๊ณ ์์ง๋ง self ๋งํฌ๊ฐ ์๋ณ์ ์ญํ ์ ํ๋ฏ๋ก id ๊ฐ์ ํ์์๋ค.
package tacos.web.api;
import java.util.Date;
import java.util.List;
import org.springframework.hateoas.ResourceSupport;
import lombok.Getter;
import tacos.Taco; // ๋๋ฉ์ธ ๊ฐ์ฒด
public class TacoResource extends ResourceSupport {
// ๋๋ฉ์ธ ๊ฐ์ฒด์ ๋ฌ๋ฆฌ id๊ฐ์ ํ์์๋ค.
@Getter
private final String name;
@Getter
private final Date createdAt;
@Getter
private final List<IngredientResource> ingredients;
public TacoResource(Taco taco) { // Taco ๊ฐ์ฒด์์ ์์ฑ๊ฐ ๋ณต์ฌ
this.name = taco.getName();
this.createdAt = taco.getCreatedAt();
this.ingredients = taco.getIngredients();
}
}
์์ฑ์๋ฅผ ํตํด ์ผ๋ฐ ๋๋ฉ์ธ ๊ฐ์ฒด๋ฅผ ๋ฆฌ์์ค ๊ฐ์ฒด๋ก ๋ฐ๊พธ๋ ์ญํ ์ ํ๋ค.
๋ฆฌ์์ค ์ด์ ๋ธ๋ฌ ๊ตฌ์ฑ
ResourceAssemblerSupport
๋ฅผ ์์๋ฐ์ ๋ฆฌ์์ค ์ด์
๋ธ๋ฌ๋ ์ปจํธ๋กค๋ฌ์ ๋ฆฌ์์ค ๊ฐ์ฒด๋ฅผ ๋ฐ์ ํ์ดํผ๋งํฌ๋ฅผ ๋ง๋ค์ด๋ธ๋ค.
package tacos.web.api;
import org.springframework.hateoas.mvc.ResourceAssemblerSupport;
import tacos.Taco;
public class TacoResourceAssembler extends ResourceAssemblerSupport<Taco, TacoResource> {
// ์์ฑ์๋ฅผ ํตํด ํ์ดํผ๋งํฌ ์์ฑ
public TacoResourceAssembler() {
super(DesignTacoController.class, TacoResource.class);
}
// ๋๋ฉ์ธ ๊ฐ์ฒด๋ฅผ ๋ฆฌ์์ค ๊ฐ์ฒด๋ก ๋ง๋๋ ๋ฉ์๋
@Override
protected TacoResource instantiateResource(Taco taco) {
return new TacoResource(taco); // ๋ฆฌ์์ค ๊ฐ์ฒด๊ฐ ์ค๋ฒ๋ผ์ด๋ํ ์์ฑ์๋ฅผ ๊ฐ์ง ๊ฒฝ์ฐ ์ง์ ์ค๋ฒ๋ผ์ด๋ ํด์ค์ผํจ
}
// ์ธ์คํด์ค ์์ฑ + ํ์ดํผ๋งํฌ ์ถ๊ฐ ํ๋ ๋ฉ์๋
// ์์ instantiateResource ๋ฉ์๋๋ฅผ ์ด์ฉํ๋ค.
@Override
public TacoResource toResource(Taco taco) {
//์ฒซ๋ฒ์งธ ์ธ์๋ก api๋ฅผ ๋๋๋ ์๋ณ์๋ฅผ ์ค์ผํจ.
return createResourceWithId(taco.getId(), taco);
}
}
์ด๋ฅผ ์ด์ฉํด ๋ค์๊ณผ ๊ฐ์ด ๋ฉ์๋๋ฅผ ๋ณ๊ฒฝํ์.
@GetMapping(path="/recent", produces="application/hal+json")
public Resources<TacoResource> recentTacos() {
PageRequest page = PageRequest.of(0, 12, Sort.by("createdAt").descending());
List<Taco> tacos = tacoRepo.findAll(page).getContent();
List<TacoResource> tacoResources = new TacoResourceAssembler().toResources(tacos);
Resources<TacoResource> recentResources = new Resources<TacoResource>(tacoResources);
recentResources.add(
linkTo(methodOn(DesignTacoController.class)
.recentTacos())
.withRel("recents"));
return recentResources;
}
๋ฆฌํด ํ์
์ด Resources<Resource<Taco>>
๋์ Resources<TacoResource>
๋ก ๋ฐ๋์๋ค.
์์ ๋ฆฌ์์ค ์ด์
๋ธ๋ฌ๋ฅผ ํตํด ๋ฆฌ์์ค ๊ฐ์ฒด ๋ฆฌ์คํธ๋ฅผ ๋ง๋ ๋ค ์ด๋ฅผ ํตํด Resources
๊ฐ์ฒด๋ฅผ ๋ง๋ค์ด ์ถ๊ฐํ๋ค.
์ค์ฒฉ ๋ฆฌ์์ค ๊ฐ์ฒด ๊ตฌํ
์์ ๊ฐ ํ์ฝ ๊ฐ์ฒด์๋ ์์์ฌ(ingredient)๋ผ๋ ์ค์ฒฉ๋ ๋๋ฉ์ธ ๊ฐ์ฒด๊ฐ ์กด์ฌํ๋ค.
์ด๋ค์ ์ํด ์ถ๊ฐ๋ก ํ์ดํผ๋งํฌ๋ฅผ ์ถ๊ฐํ๋ ค๋ฉด ๋ค์์ ๋ฐ๋ฅด๋ฉด ๋๋ค.
- ์์์ฌ๋ฅผ ์ํ ๋ฆฌ์์ค ๊ฐ์ฒด, ๋ฆฌ์์ค ์ด์ ๋ธ๋ฌ ์์ฑ
IngredientResource.java
package tacos.web.api;
import org.springframework.hateoas.ResourceSupport;
import lombok.Getter;
import tacos.Ingredient;
import tacos.Ingredient.Type;
public class IngredientResource extends ResourceSupport {
@Getter
private String name;
@Getter
private Type type;
public IngredientResource(Ingredient ingredient) {
this.name = ingredient.getName();
this.type = ingredient.getType();
}
}
IngredientResourceAssembler.java
package tacos.web.api;
import org.springframework.hateoas.mvc.ResourceAssemblerSupport;
import tacos.Ingredient;
class IngredientResourceAssembler extends ResourceAssemblerSupport<Ingredient, IngredientResource> {
public IngredientResourceAssembler() {
super(IngredientController.class, IngredientResource.class);
}
@Override
public IngredientResource toResource(Ingredient ingredient) {
return createResourceWithId(ingredient.getId(), ingredient);
}
@Override
protected IngredientResource instantiateResource(Ingredient ingredient) {
return new IngredientResource(ingredient);
}
}
์์ ์์ฑํ๋ ๋ฐฉ์๊ณผ ๊ฐ์ ๋ฐฉ์์ด๋ค.
- ํด๋น ๋ฆฌ์์ค ๊ฐ์ฒด๋ฅผ ํฌํจํ๋ ์์ ๋ฆฌ์์ค ๊ฐ์ฒด ๋ณ๊ฒฝ
TacoResource.java
package tacos.web.api;
import java.util.Date;
import java.util.List;
import org.springframework.hateoas.ResourceSupport;
import lombok.Getter;
import tacos.Taco;
public class TacoResource extends ResourceSupport {
private static final IngredientResourceAssembler ingredientAssembler = new IngredientResourceAssembler();
@Getter
private final String name;
@Getter
private final Date createdAt;
@Getter
private final List<IngredientResource> ingredients;
public TacoResource(Taco taco) {
this.name = taco.getName();
this.createdAt = taco.getCreatedAt();
this.ingredients = ingredientAssembler.toResources(taco.getIngredients());
}
}
๊ฐ ๋๋ฉ์ธ ๊ฐ์ฒด์ ์์ฌ๋ฃ ๋ฆฌ์คํธ๋ฅผ ์์ฌ๋ฃ ์ด์ ๋ธ๋ฌ๋ฅผ ํตํด ์์ฑํ๋๋ก ๋ง๋ค์๋ค.
embedded
๊ด๊ณ ์ด๋ฆ ์ง๊ธฐ
์ฐ๋ฆฌ๊ฐ ๋ง๋ API๋ ๋ค์๊ณผ ๊ฐ์ด ์๋์ผ๋ก tacoRsourceList
๋ผ๋ ์์ฑ์ ๋ง๋ค์ด์ค๋ค.
{
"_embedded": {
"tacoResourceList": [ // ๋ฆฌ์์ค๊ฐ์ฒด๋ช
+ List๋ก ์ ํด์ง๋ค. (๊ธฐ๋ณธ๊ฐ)
{
"name": "Veg-Out",
"createdAt": "2018-01-31T20:15:53.219+0000",
"ingredients": [
...
๋ฌธ์ ๋ ์ด๊ฒ์ด ํด๋ผ์ด์ธํธ ์ธก์ ํ๋์ฝ๋ฉ๋์ด์ผ ํ๋ฉฐ, ๋ณ๊ฒฝ์ผ๋ก ์ธํด ํด๋ผ์ด์ธํธ์ ์ค๋ฅ๊ฐ ์๊ธธ ์๋ ์๋ค.
์ด๋ฅผ ๋ฐฉ์งํ๊ธฐ ์ํด ๋ค์๊ณผ ๊ฐ์ด @Relation
์ด๋
ธํ
์ด์
์ ํตํด ์์ฑ๋ช
์ ๋ฐ๊ฟ์ค ์ ์๋ค.
@Relation(value="taco", collectionRelation="tacos")
public class TacoResource extends ResourceSupport {
//...
}
์ด์ tacoResourceList
์์ฑ๋ช
์ tacos
๋ก ๋ฐ๋๋ค.
_articles/web/backend/Spring/Spring5-HATEOAS.md