Gidhub BE Developer

Spring MVC - @ModelAttribute 알아보기

2019-09-13
goodGid

이 글의 코드 및 정보들은 강의를 들으며 정리한 내용을 토대로 작성하였습니다.

@ModelAttribute

  • 단일 데이터를

    각각 받아와 객체를 생성하지 않고

public class Event {
    private String name;

    @Max(10)
    private Integer count;
}
@GetMapping("/events")
@ResponseBody
public Event hello(@RequestParam String name, @RequestParam Integer count){
    Event event = new Event();
    event.setName(name);
    event.setCount(count);
    return event;
}
  • 복합 타입 객체로 입력을 받을 수 있게 된다.
@GetMapping("/events")
@ResponseBody
public Event hello(Event event){
    return event;
}
  • 또한 @ModelAttribute는 생략 가능하다.
public Event hello(@ModelAttribute Event event)
= public Event hello(Event event)

Binding 실패

  • Binding 과정에서

    여러 이유로 Binding 실패가 일어날 수 있다.

    이럴 경우 BindingResult를 통해 Error를 확인할 수 있다.

Controller

@GetMapping("/events")
@ResponseBody
public Event hello(@ModelAttribute Event event, BindingResult bindingResult){
    // Error를 확인할 수 있다.
    if(bindingResult.hasErrors()){
        bindingResult.getAllErrors().forEach(c ->{
            System.out.println(c);
        });
    }
    return event;
}

TC

@Test
public void helloTest() throws Exception {
    mockMvc.perform(get("/events")
                            .param("name", "goodGid")
                            .param("count", "test"))
            .andDo(print())
            .andExpect(status().isOk());
}

Result

Field error in object 'event' on field 'count': rejected value [test]; codes [typeMismatch.event.count,typeMismatch.count,typeMismatch.java.lang.Integer,typeMismatch]; arguments [org.springframework.context.support.DefaultMessageSourceResolvable: codes [event.count,count]; arguments []; default message [count]]; default message [Failed to convert property value of type 'java.lang.String' to required type 'java.lang.Integer' for property 'count'; nested exception is java.lang.NumberFormatException: For input string: "test"]

...

MockHttpServletResponse:
           Status = 200
    Error message = null
          Headers = [Content-Type:"application/json;charset=UTF-8"]
     Content type = application/json;charset=UTF-8
             Body = {"id":null,"name":"goodGid","count":null}
    Forwarded URL = null
   Redirected URL = null
          Cookies = []
  • TC 자체는 성공을 하지만

    (= Status = 200 )

    Body를 보면 name 에는

    요청시 입력한 goodGid가 들어갔지만

    count 에는 null이 담겨져 있음을 확인할 수 있다.

    count의 type은 Integer이기 때문이다.


BindingResult 위치

  • BindingResult은

    Binding하고자 하는 Argument의 바로 오른쪽에 위치해야 한다.

    아래와 같은 코드는 실패하게 된다.

@GetMapping("/events")
@ResponseBody
public Event hello(@ModelAttribute Event event,
                    String test,
                    BindingResult bindingResult) { // 잘못된 위치
    if (bindingResult.hasErrors()) {
        bindingResult.getAllErrors().forEach(c -> {
            System.out.println(c);
        });
    }
    return event;
}

Binding 이후 검증

  • 단순히 Binding의 성공 유무가 아닌

    특정 조건을 만족하는 값으로

    Binding이 되었는지 확인하기 위해서

    @Valid or @Validated 애노테이션을 사용할 수 있다.


  • Binding 자체는 성공하였지만

    로직 관점에서는 실패할 수 있다.


  • 예를 들어

    count 의 값은

    MAX가 10인데

    요청을 10보다 큰 값으로 하게 될 경우

    Binding은 성공하겠지만 Error 로그가 노출된다.

Domain

public class Event {

    private Integer id;

    private String name;

    @Max(10)
    private Integer count;
}

TC Result

Field error in object 'event' on field 'count': rejected value [11]; codes [Max.event.count,Max.count,Max.java.lang.Integer,Max]; arguments [org.springframework.context.support.DefaultMessageSourceResolvable: codes [event.count,count]; arguments []; default message [count],10]; default message [must be less than or equal to 10]

...

MockHttpServletResponse:
           Status = 200
    Error message = null
          Headers = [Content-Type:"application/json;charset=UTF-8"]
     Content type = application/json;charset=UTF-8
             Body = {"id":null,"name":"goodGid","count":11}
    Forwarded URL = null
   Redirected URL = null
          Cookies = []

Summary

  • Model Attribute 개념에 대해 학습하였다.

    보다시피 굉장히 중요한 개념이다.



Reference


Recommend

Index