[Spring MVC] 로그인1 - 쿠키, (서버)세션, 서블릿 HttpSession

2022. 6. 28. 12:09Spring

package 구조

  • hello.login
    • domain
      • item
      • member
      • login
    • web
      • item
      • member
      • login

 

도메인이 가장 중요하다.

도메인 : 화면, UI, 기술 인프라 등등의 영역은 제외한 시스템이 구현해야 하는 핵심 비즈니스 업무 영역을 말합니다.

 

향후 web을 다른 기술로 바꾸어도 도메인은 그대로 유지할 수 있어야 합니다.

= web은 domain을 알고 있지만 domain은 web을 모르도록 설계

= web은 domain을 의존하지만, domain은 web을 의존하지 않는다.

예를 들어 web 패키지를 모두 삭제해도 domain에는 전혀 영향이 없도록 의존관계를 설계하는 것이 중요하고 반대로  domain은 web을 참조하면 안 됩니다.

 

 

홈 화면 개발

HomeController - home() 수정

@GetMapping("/")
public String home() {
 return "home";
}

 

templates/home.html

<!DOCTYPE HTML>
<html xmlns:th="http://www.thymeleaf.org">
<head>
    <meta charset="utf-8">
    <link th:href="@{/css/bootstrap.min.css}"
          href="../css/bootstrap.min.css" rel="stylesheet">
</head>
<body>

<div class="container" style="max-width: 600px">
    <div class="py-5 text-center">
        <h2>홈 화면</h2>
    </div>

    <div class="row">
        <div class="col">
            <button class="w-100 btn btn-secondary btn-lg" type="button"
                    th:onclick="|location.href='@{/members/add}'|">
                회원 가입
            </button>
        </div>
        <div class="col">
            <button class="w-100 btn btn-dark btn-lg" onclick="location.href='items.html'"
                    th:onclick="|location.href='@{/login}'|" type="button">
                로그인
            </button>
        </div>
    </div>

    <hr class="my-4">

</div> <!-- /container -->

</body>
</html>

 

회원 가입

domain > member > Member 엔티티

package hello.login.domain.member;

import lombok.Data;

import javax.validation.constraints.NotEmpty;

@Data
public class Member {

    private Long id;

    @NotEmpty
    private String loginId; //로그인 ID
    @NotEmpty
    private String name; //사용자 이름
    @NotEmpty
    private String password;
}

 

Member엔티티 저장하고 관리하는 장소 = MemberRepository

package hello.login.domain.member;

import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Repository;

import java.util.*;

@Slf4j
@Repository
public class MemberRepository { // Member엔티티 저장하고 관리하는 장소

    private static Map<Long, Member> store = new HashMap<>(); // 저장소
    private static long sequence = 0L;

    // 저장
    public Member save(Member member) {
        member.setId(++sequence);
        log.info("save: member={}", member);
        store.put(member.getId(), member);
        return member;
    }

    public Member findById(Long id) {
        return store.get(id);
    }

    public Optional<Member> findByLoginId(String loginId) { //못 찾을 수 있으므로 Optional
//        List<Member> all = findAll();
//        for(Member m : all) {
//            if(m.getLoginId().equals(loginId)) {
//                return Optional.of(m);
//            }
//        }
//        return Optional.empty();

        // java8 람다, 스트림
        return findAll().stream() // 리스트의 루프 돌면서
                .filter(m -> m.getLoginId().equals(loginId)) // DB where절
                .findFirst();
    }

    public List<Member> findAll() {
        return new ArrayList<>(store.values()); // 값들 List로 변환 후 반환
    }

    public void clearStore() { // 테스트시 초기화용
        store.clear();
    }
}

 

web > member > MemberController

package hello.login.web.member;

@Controller
@RequiredArgsConstructor
@RequestMapping("/members")
public class MemberController {

    private final MemberRepository memberRepository;

    @GetMapping("/add") // 회원등록 폼으로 이동
    public String addForm(@ModelAttribute("member") Member member) {

        return "members/addMemberForm";
    }

    @PostMapping("/add")
    public String save(@Valid @ModelAttribute Member member, BindingResult bindingResult) { // @Valid 유효성 검증
        if (bindingResult.hasErrors()) {
            return "members/addMemberForm";
        }

        memberRepository.save(member);
        return "redirect:/";
    }
}

@ModelAttribute("member") Member member은 

@ModelAttribute Member member와 같습니다. 가끔 IDE에서 인식을 못하기 때문에 위와 같이 적었습니다.

 

templates > members > addMemberForm.html

<!DOCTYPE HTML>
<html xmlns:th="http://www.thymeleaf.org">
<head>
    <meta charset="utf-8">
    <link th:href="@{/css/bootstrap.min.css}"
          href="../css/bootstrap.min.css" rel="stylesheet">
    <style>
        .container {
            max-width: 560px;
        }
        .field-error {
            border-color: #dc3545;
            color: #dc3545;
        }
    </style>
</head>
<body>

<div class="container">

    <div class="py-5 text-center">
        <h2>회원 가입</h2>
    </div>

    <h4 class="mb-3">회원 정보 입력</h4>

    <form action="" th:action th:object="${member}" method="post">

        <div th:if="${#fields.hasGlobalErrors()}">
            <p class="field-error" th:each="err : ${#fields.globalErrors()}" th:text="${err}">전체 오류 메시지</p>
        </div>

        <div>
            <label for="loginId">로그인 ID</label>
            <input type="text" id="loginId" th:field="*{loginId}" class="form-control"
                   th:errorclass="field-error">
            <div class="field-error" th:errors="*{loginId}" />
        </div>
        <div>
            <label for="password">비밀번호</label>
            <input type="password" id="password" th:field="*{password}" class="form-control"
                   th:errorclass="field-error">
            <div class="field-error" th:errors="*{password}" />
        </div>
        <div>
            <label for="name">이름</label>
            <input type="text" id="name" th:field="*{name}" class="form-control"
                   th:errorclass="field-error">
            <div class="field-error" th:errors="*{name}" />
        </div>


        <hr class="my-4">

        <div class="row">
            <div class="col">
                <button class="w-100 btn btn-primary btn-lg" type="submit">회원 가입</button>
            </div>
            <div class="col">
                <button class="w-100 btn btn-secondary btn-lg" onclick="location.href='items.html'"
                        th:onclick="|location.href='@{/}'|"
                        type="button">취소</button>
            </div>
        </div>

    </form>

</div> <!-- /container -->
</body>
</html>

 

회원용 테스트 데이터 추가

 매번 회원 가입하고 로그인해야 하는 번거로움을 줄이기 위해 한 회원을 자동 등록해보겠습니다. 

package hello.login;

import hello.login.domain.item.Item;
import hello.login.domain.item.ItemRepository;
import hello.login.domain.member.Member;
import hello.login.domain.member.MemberRepository;
import lombok.RequiredArgsConstructor;
import org.springframework.stereotype.Component;

import javax.annotation.PostConstruct;

@Component
@RequiredArgsConstructor
public class TestDataInit {

    private final ItemRepository itemRepository;
    private final MemberRepository memberRepository;

    /**
     * 테스트용 데이터 추가
     */
    @PostConstruct
    public void init() {
        itemRepository.save(new Item("itemA", 10000, 10));
        itemRepository.save(new Item("itemB", 20000, 20));
		
        // 이 부분!!
        Member member = new Member();
        member.setLoginId("test");
        member.setPassword("test!");
        member.setName("테스터");

        memberRepository.save(member);

    }

}

 

로그인 기능 개발

domain > login > LoginService

데이터가 맞게 들어갔는지 판단하는 로직, 핵심 비즈니스 로직

: 회원을 조회한 다음에 파라미터로 넘어온 password와 비교해서 같으면 회원을 반환하고,

  만약 password가 다르면 null을 반환한다.

package hello.login.domain.login;

@Service
@RequiredArgsConstructor
public class LoginService {

    private final MemberRepository memberRepository;

    /**
     * @return null 로그인 실패
     */
    public Member login(String loginId, String password) {
//        Optional<Member> findMemeber = memberRepository.findByLoginId(loginId);
//        Member member = findMemeber.get();
//        if(member.getPassword().equals(password)) {
//            return member;
//        }else {
//            return null;
//        }

        // 위와 같은 코드
        return memberRepository.findByLoginId(loginId)
                .filter(m -> m.getPassword().equals(password))
                .orElse(null); // 그렇지 않다면
    }
}

 

web > login > LoginForm

package hello.login.web.login;

import lombok.Data;

import javax.validation.constraints.NotEmpty;

@Data
public class LoginForm {

    @NotEmpty
    private String loginId;

    @NotEmpty
    private String password;
}

 

web > login > LoginController

package hello.login.web.login;

@Slf4j
@Controller
@RequiredArgsConstructor
public class LoginController {

    private final LoginService loginService;
   
    @GetMapping("/login") // 폼 보여줌
    public String loginForm(@ModelAttribute("loginForm") LoginForm form) {
        return "login/loginForm";
    }

    @PostMapping("/login") // 검증 로직
    public String login(@Valid @ModelAttribute LoginForm form, BindingResult bindingResult) {
        if (bindingResult.hasErrors()) {
            return "login/loginForm";
        }

       Member loginMember = loginService.login(form.getLoginId(), form.getPassword());

        if (loginMember == null) {
            // 오류를 보내는게 아닌, bindingResult 처리
            // bindingResult.reject는 글로벌 오류
            bindingResult.reject("loginFail", "아이디 또는 비밀번호가 맞지 않습니다.");
            return "login/loginForm";
        }


        //로그인 성공 처리(추후 넣을 예정)

       
        return "redirect:/";

    }


}

 

templates > login > loginForm.html

<!DOCTYPE HTML>
<html xmlns:th="http://www.thymeleaf.org">
<head>
    <meta charset="utf-8">
    <link th:href="@{/css/bootstrap.min.css}"
          href="../css/bootstrap.min.css" rel="stylesheet">
    <style>
        .container {
            max-width: 560px;
        }
        .field-error {
            border-color: #dc3545;
            color: #dc3545;
        }
    </style>
</head>
<body>

<div class="container">

    <div class="py-5 text-center">
        <h2>로그인</h2>
    </div>

    <form action="item.html" th:action th:object="${loginForm}" method="post">

        <div th:if="${#fields.hasGlobalErrors()}">
            <p class="field-error" th:each="err : ${#fields.globalErrors()}" th:text="${err}">전체 오류 메시지</p>
        </div>

        <div>
            <label for="loginId">로그인 ID</label>
            <input type="text" id="loginId" th:field="*{loginId}" class="form-control"
                   th:errorclass="field-error">
            <div class="field-error" th:errors="*{loginId}" />
        </div>
        <div>
            <label for="password">비밀번호</label>
            <input type="password" id="password" th:field="*{password}" class="form-control"
                   th:errorclass="field-error">
            <div class="field-error" th:errors="*{password}" />
        </div>

        <hr class="my-4">

        <div class="row">
            <div class="col">
                <button class="w-100 btn btn-primary btn-lg" type="submit">로그인</button>
            </div>
            <div class="col">
                <button class="w-100 btn btn-secondary btn-lg" onclick="location.href='items.html'"
                        th:onclick="|location.href='@{/}'|"
                        type="button">취소</button>
            </div>
        </div>

    </form>

</div> <!-- /container -->
</body>
</html>

 

실행

로그인이 성공하면 홈으로 이동하고, 로그인에 실패하면 "아이디 또는 비밀번호가 맞지 않습니다."라는 경고와 함께 로그인 폼이 나타납니다.

 

+ 추가 요구사항

"로그인이 되면 홈 화면에 고객 이름이 보이게 해 주세요."

로그인의 상태를 유지하면서, 로그인에 성공한 사용자가 홈 화면에 접근 시 이름을 보여주기 위해서는 어떻게 해야 할지 

알아보겠습니다.

 

 

로그인 - 쿠키 사용

web > login > LoginController

package hello.login.web.login;

@Slf4j
@Controller
@RequiredArgsConstructor
public class LoginController {

    private final LoginService loginService;
   
    @GetMapping("/login") // 폼 보여줌
    public String loginForm(@ModelAttribute("loginForm") LoginForm form) {
        return "login/loginForm";
    }

    @PostMapping("/login") // 검증 로직
    public String login(@Valid @ModelAttribute LoginForm form, BindingResult bindingResult, HttpServletResponse response) {
        if (bindingResult.hasErrors()) {
            return "login/loginForm";
        }

       Member loginMember = loginService.login(form.getLoginId(), form.getPassword());

        if (loginMember == null) {
            // 오류를 보내는게 아닌, bindingResult 처리
            // bindingResult.reject는 글로벌 오류
            bindingResult.reject("loginFail", "아이디 또는 비밀번호가 맞지 않습니다.");
            return "login/loginForm";
        }


        /**
         * 로그인 성공 처리
         *   서버에서 세션쿠키를 만들어서 클라이언트에게 전달
         *   그 후 어느 페이지에 가든지 클라이언트가 서버에 항상 쿠키 보내줌
         */

        //쿠키에 시간 정보를 주지 않으면 세션 쿠기이다(브라우저 종료시 모두 종료)
        // String.valueOf() : string이 들어가야 하므로(숫자를 문자로 바꿈)
        Cookie idCookie = new Cookie("memberId", String.valueOf(loginMember.getId()));
        // 서버에서 HTTP응답 보낼때 response에 넣어서 보냄, login메서드 파라미터에 HttpServletResponse 객체 추가 
        response.addCookie(idCookie);
        return "redirect:/";

    }


}

 

로그인에 성공하면 쿠키를 생성하고 HttpServletResponse에 담습니다. (쿠키 이름은 memberId , 값은 회원의 id )

웹 브라우저는 종료 전까지 회원의 id를 서버에 계속 보내줄 것이다.

 

실행해보면, HTTP 응답 헤더(Response Headers)에 쿠키가 추가된 것을 확인할 수 있고.(Set-Cookie)

브라우저가 요청을 보낼 때 Request Headers의 Cookie가 들어있음을 확인할 수 있습니다.

 

 

로그인 한 사용자 전용 홈 화면

HomeController

package hello.login.web;

@Slf4j
@Controller
@RequiredArgsConstructor
public class HomeController {

    private final MemberRepository memberRepository;

    @GetMapping("/")
    // 로그인 시 사용자 이름 뜨는 홈화면
    // @CookieValue : 쿠키 조회, 쿠키 없는 사용자도 들어와야 하므로 required = false
    public String homeLogin(@CookieValue(name = "memberId", required = false) Long memberId, Model model) {

        if (memberId == null) {
            return "home";
        }

        //로그인 성공 한 사용자(쿠키 있음)
        Member loginMember = memberRepository.findById(memberId);
        if (loginMember == null) {
            return "home";
        }

        model.addAttribute("member", loginMember);
        return "loginHome";
    }


}

 

loginHome.html

  로그인: ${member.name}

<!DOCTYPE HTML>
<html xmlns:th="http://www.thymeleaf.org">
<head>
    <meta charset="utf-8">
    <link th:href="@{/css/bootstrap.min.css}"
          href="../css/bootstrap.min.css" rel="stylesheet">
</head>
<body>

<div class="container" style="max-width: 600px">
    <div class="py-5 text-center">
        <h2>홈 화면</h2>
    </div>

    <h4 class="mb-3" th:text="|로그인: ${member.name}|">로그인 사용자 이름</h4>

    <hr class="my-4">

    <div class="row">
        <div class="col">
            <button class="w-100 btn btn-secondary btn-lg" type="button"
                    th:onclick="|location.href='@{/items}'|">
                상품 관리
            </button>
        </div>
        <div class="col">
            <form th:action="@{/logout}" method="post">
                <button class="w-100 btn btn-dark btn-lg" onclick="location.href='items.html'" type="submit">
                    로그아웃
                </button>
            </form>
        </div>
    </div>

    <hr class="my-4">

</div> <!-- /container -->

</body>
</html>

로그인에 성공하면 사용자 이름이 출력되면서 상품 관리, 로그아웃 버튼을 확인할 수 있습니다.

로그인에 성공 시 세션 쿠키가 지속해서 유지됩니다.

 

 

로그아웃 기능

로그아웃 버튼 클릭 시 쿠키를 날리기.

 

LoginController - logout 기능 추가

@PostMapping("/logout")
public String logout(HttpServletResponse response) {
 expireCookie(response, "memberId");
 return "redirect:/";
}

private void expireCookie(HttpServletResponse response, String cookieName) {
 Cookie cookie = new Cookie(cookieName, null);
 cookie.setMaxAge(0);
 response.addCookie(cookie);
}

개발자 도구를 통해 보자면 , Response Headers > Set-Cookie : Max-Age=0를 확인할 수 있습니다.

해당 쿠키는 즉시 종료됩니다.

 

이렇게 쿠키만으로 로그인과 로그아웃 구현이 가능하다.

이 방법은 보안상 큰 문제가 있습니다.

 

 

쿠키와 보안 문제

보안 문제

1. 쿠키 값은 임의로 변경할 수 있다.

클라이언트가 쿠키를 강제로 변경하면 다른 사용자가 된다.

 

2. 쿠키에 보관된 정보는 훔쳐갈 수 있다.

 

대안

  • 쿠키에 중요한 값을 노출하지 않고, 사용자 별로 예측 불가능한 임의의 토큰(랜덤 값)을 노출하고, 서버에서 토큰과 사용자 id를 매핑해서 인식한다. 그리고 서버에서 토큰을 관리한다.
  • 해커가 토큰을 털어가도 시간이 지나면 사용할 수 없도록 서버에서 해당 토큰의 만료시간을 짧게(예: 30분) 유지한다. 

 

로그인 - 세션 사용

로그인 처리하기 - 세션 동작 방식

앞서 쿠키에 중요한 정보를 보관하는 방법은 여러 가지 보안 이슈가 있었다. 이 문제를 해결하려면 결국 중요한 정보를 모두 서버에 저장해야 하고 클라이언트와 서버는 추정 불가능한 임의의 식별자 값으로 연결해야 한다.

이렇게 서버에 중요한 정보를 보관하고 연결을 유지하는 방법을 세션이라 합니다.

 

세션 생성

세션 ID를 생성하는데, 추정 불가능해야 합니다.(UUID)

생성된 세션 ID와 세션에 보관할 값(memberA )을 서버의 세션 저장소에 보관합니다.

 

 

세션 id를 응답 쿠키로 전달

클라이언트와 서버는 결국 쿠키로 연결이 되어야 합니다.

서버는 클라이언트에 mySessionId라는 이름으로 세션 ID 쿠키에 담아서 전달하고

클라이언트는 쿠키 저장소에 mySessionId 쿠키를 보관합니다.

 

여기서 중요한 포인트

회원과 관련된 정보는 전혀 클라이언트에 전달하지 않는다는 것도오직 추정 불가능한 세션 ID만 쿠키를 통해 클라이언트에 전달했다는 것입니다.

 

 

정리 : 쿠키 사용했을 시 보안 문제 해결

세션을 사용해서 서버에서 중요한 정보를 관리

예상 불가능한 복잡한 세션 Id를 사용하므로 세션 Id가 털려도 여기에는 중요한 정보가 없다.

 

 

로그인 처리하기 - 세션 직접 만들기

세션을 더 잘 이해하기 위해 세션을 직접 만들어서 적용한 후 HttpSession을 이용해보도록 하겠습니다.

 

세션 생성
sessionId 생성 (임의의 토큰)
세션 저장소에 sessionId와 보관할 값(value) 저장
sessionId로 응답 쿠키를 생성해서 클라이언트에 전달

세션 조회
클라이언트가 요청한 sessionId 쿠키의 값으로, 세션 저장소에 보관한 값 조회

세션 만료
클라이언트가 요청한 sessionId 쿠키의 값으로, 세션 저장소에 보관한 sessionId와 값 제거

 

web > session > SessionManager

package hello.login.web.session;


/**
 * 세션 관리
 */
@Component // 스프링 빈으로 자동 등록
public class SessionManager {

    public static final String SESSION_COOKIE_NAME = "mySessionId"; // 사용할 곳이 많아서 상수로 만듦
    // 동시성 이슈 있을 때(여러 쓰레드 접근 이슈) 
    // HashMap 은 동시 요청에 안전하지 않다. 동시 요청에 안전한 ConcurrentHashMap 를 사용
    private Map<String, Object> sessionStore = new ConcurrentHashMap<>();

    /**
     * 세션 생성
     */
    public void createSession(Object value, HttpServletResponse response) {

        //세션 id를 생성하고, 값을 세션에 저장
        String sessionId = UUID.randomUUID().toString(); // 임의의 토큰
        sessionStore.put(sessionId, value);

        //쿠키 생성
        Cookie mySessionCookie = new Cookie(SESSION_COOKIE_NAME, sessionId);
        response.addCookie(mySessionCookie);
    }

    /**
     * 세션 조회
     */
    public Object getSession(HttpServletRequest request) {
        Cookie sessionCookie = findCookie(request, SESSION_COOKIE_NAME);
        if (sessionCookie == null) {
            return null;
        }
        return sessionStore.get(sessionCookie.getValue());
    }

    /**
     * 세션 만료
     */
    public void expire(HttpServletRequest request) {
        Cookie sessionCookie = findCookie(request, SESSION_COOKIE_NAME);
        if (sessionCookie != null) {
            sessionStore.remove(sessionCookie.getValue());
        }
    }


    // 쿠키 찾는 부분 분리함
    public Cookie findCookie(HttpServletRequest request, String cookieName) {
        if (request.getCookies() == null) {
            return null;
        }
        // Arrays.stream() : 배열을 스트림으로 바꿔줌, 배열의 값을 하나씩 루프 돌림
        return Arrays.stream(request.getCookies())
                .filter(cookie -> cookie.getName().equals(cookieName))
                .findAny()// 순서 상관없이 하나만 찾아서 반환
                .orElse(null);
    }

}

 

로그인 처리하기 - 직접 만든 세션 적용

LoginController

package hello.login.web.login;

@Slf4j
@Controller
@RequiredArgsConstructor
public class LoginController {

    private final LoginService loginService;
    private final SessionManager sessionManager;

   
    @PostMapping("/login")
    public String loginV2(@Valid @ModelAttribute LoginForm form, BindingResult bindingResult, HttpServletResponse response) {
        if (bindingResult.hasErrors()) {
            return "login/loginForm";
        }

        Member loginMember = loginService.login(form.getLoginId(), form.getPassword());

        if (loginMember == null) {
            bindingResult.reject("loginFail", "아이디 또는 비밀번호가 맞지 않습니다.");
            return "login/loginForm";
        }

        //로그인 성공 처리

        //세션 관리자를 통해 세션을 생성하고, 회원 데이터 보관
        sessionManager.createSession(loginMember, response);

        return "redirect:/";

    }


    @PostMapping("/logout")
    public String logoutV2(HttpServletRequest request) { //HttpServletRequest 써야 쿠키의 값을 꺼내서 만료시킴
        sessionManager.expire(request);
        return "redirect:/";
    }

}

 

HomeController

package hello.login.web;

@Slf4j
@Controller
@RequiredArgsConstructor
public class HomeController {

    private final MemberRepository memberRepository;
    private final SessionManager sessionManager;


    @GetMapping("/")
    public String homeLoginV2(HttpServletRequest request, Model model) {

        //세션 관리자에 저장된 회원 정보 조회
        Member member = (Member)sessionManager.getSession(request); // Object이므로 Member로 캐스팅

        //로그인
        if (member == null) {
            return "home";
        }

        model.addAttribute("member", member);
        return "loginHome";
    }


}

 

세션이라는 것이 뭔가 특별한 것이 아니라 단지 쿠키를 사용하는데, 서버에서 데이터를 유지하는 방법일 뿐입니다.

 

서블릿도 세션 개념을 지원

서블릿이 공식 지원하는 세션은 우리가 직접 만든 세션과 동작 방식이 거의 같고 추가로 세션을 일정 시간 사용하지 않으면 해당 세션을 삭제하는 기능까지 제공합니다,

 

 

로그인 - 서블릿 HTTP 세션 1

서블릿은 세션을 위해 HttpSession이라는 기능을 제공합니다.

서블릿을 통해 HttpSession을 생성하면 쿠키 이름이 JSESSIONID이고, 값은 추정 불가능한 랜덤 값인 쿠키를 생성합니다. 

 

web > SessionConst

HttpSession에 데이터를 보관하고 조회할 때, 같은 이름이 중복되어 사용되므로, 상수를 하나 정의해서 다른 클래스에서 이름을 가져다 쓰기 위해 만들었습니다. (인터페이스 또는 추상 클래스 - public으로 만듦)

package hello.login.web;

public interface SessionConst {
    String LOGIN_MEMBER = "loginMember";
}

 

web > login > LoginController > V3

package hello.login.web.login;

@Slf4j
@Controller
@RequiredArgsConstructor
public class LoginController {

    private final LoginService loginService;
    private final SessionManager sessionManager;

 
    @PostMapping("/login")
    public String loginV3(@Valid @ModelAttribute LoginForm form, BindingResult bindingResult, HttpServletRequest request) {
        if (bindingResult.hasErrors()) {
            return "login/loginForm";
        }

        Member loginMember = loginService.login(form.getLoginId(), form.getPassword());

        if (loginMember == null) {
            bindingResult.reject("loginFail", "아이디 또는 비밀번호가 맞지 않습니다.");
            return "login/loginForm";
        }

        //로그인 성공 처리
        //세션이 있으면 있는 세션 반환, 없으면 신규 세션을 생성해서 반환
        HttpSession session = request.getSession();
        //세션에 로그인 회원 정보 보관(키,밸류)
        session.setAttribute(SessionConst.LOGIN_MEMBER, loginMember);

        return "redirect:/";

    }


    @PostMapping("/logout")
    public String logoutV3(HttpServletRequest request) {
    	// 있는 세션 가져오고 없다면 새로 만들지 않음(false)
        HttpSession session = request.getSession(false);
        if (session != null) {
            session.invalidate();
        }
        return "redirect:/";
    }

}

HttpSession은 HttpServletRequest가 필요합니다.

 

세션을 생성하려면 request.getSession(true)를 사용(기본값이 true)

세션의 create 옵션

request.getSession(true) : 세션이 있으면 기존 세션을 반환한다. 세션이 없으면 새로운 세션을 생성해서 반환한다. request.getSession(false) : 세션이 있으면 기존 세션을 반환한다. 세션이 없으면 새로운 세션을 생성하지 않는다. null을 반환한다.

 

세션에 로그인 회원 정보 보관(키, 값)

session.setAttribute(SessionConst.LOGIN_MEMBER, loginMember);

 

 

홈 화면 수정 HomeController

package hello.login.web;

@Slf4j
@Controller
@RequiredArgsConstructor
public class HomeController {

    private final MemberRepository memberRepository;
    private final SessionManager sessionManager;


    @GetMapping("/")
    public String homeLoginV3(HttpServletRequest request, Model model) {
        // false처리: 로그인 하지 않은 사용자도 의미없는 세션이 만들어지는 것 방지
        HttpSession session = request.getSession(false);
        if (session == null) {
            return "home";
        }

        // Member로 캐스팅함
        // 이유 : LoginController에서 loginMember를 값으로 담았는데 이 타입이 Memeber이므로
        // Member loginMember = loginService.login(form.getLoginId(), form.getPassword());
        Member loginMember = (Member)session.getAttribute(SessionConst.LOGIN_MEMBER);

        //세션에 회원 데이터가 없으면 home 이동
        if (loginMember == null) {
            return "home";
        }

        //세션이 유지되면 로그인으로 이동
        model.addAttribute("member", loginMember);
        return "loginHome";
    }


}

session.getAttribute(SessionConst.LOGIN_MEMBER) : 로그인 시점에 세션에 보관한 회원 객체를 찾는다.

 

 

로그인  - 서블릿 HTTP 세션 2

스프링이 제공하는 @SessionAttribute 기능이 있습니다.

Session, attribute체크하는 로직 한 번에 해결 가능

 

이미 로그인된 사용자를 찾는 코드 

참고로 이 기능은 세션을 생성하지 않습니다.

@SessionAttribute(name = "loginMember", required = false) Member loginMember

 

 

홈 화면 HomeController

package hello.login.web;

@Slf4j
@Controller
@RequiredArgsConstructor
public class HomeController {

    private final MemberRepository memberRepository;
    private final SessionManager sessionManager;

    @GetMapping("/")
    public String homeLoginV3Spring(
            @SessionAttribute(name = SessionConst.LOGIN_MEMBER, required = false) Member loginMember, Model model) {

        //세션에 회원 데이터가 없으면 home
        if (loginMember == null) {
            return "home";
        }

        //세션이 유지되면 로그인으로 이동
        model.addAttribute("member", loginMember);
        return "loginHome";
    }

   
}

세션을 찾고, 세션에 들어있는 데이터를 찾는 번거로운 과정을 해결해주었습니다.

 

 

세션 정보와 타임아웃 설정

세션이 제공하는 정보들

sessionId : 세션 Id, JSESSIONID의 값

maxInactiveInterval : 세션의 유효 시간, 예) 1800초, (30분)

creationTime : 세션 생성 일시

lastAccessedTime : 세션과 연결된 사용자가 최근에 서버에 접근한 시간 (마지막에 접근한 시간)

isNew : 새로 생성된 세션인지 아닌지

 

이를 조합해서 세션 타임아웃을 설정할 수 있습니다.

 

 

세션 타임아웃 설정

세션은 사용자가 로그아웃을 직접 호출해서 session.invalidate()가 호출되는 경우에 삭제됩니다.

그런데 대부분의 사용자는 로그아웃을 선택하지 않고, 그냥 웹 브라우저를 종료하는데요, 문제는 HTTP가 비 연결성(ConnectionLess)이므로 서버 입장에서는 해당 사용자가 웹 브라우저를 종료한 것인지 아닌지를 인식할 수 없습니다.

 

이 경우 남아있는 세션을 무한정 보관하면 다음과 같은 문제가 발생할 수 있다.

1. 세션과 관련된 쿠키( JSESSIONID )를 탈취당했을 경우 악의적인 요청을 할 수 있다.

2. 세션은 기본적으로 메모리에 생성된다. 메모리의 크기가 무한하지 않기 때문에 꼭 필요한 경우만 생성해서 사용해야 한다.

 

세션의 종료 시점

세션 생성 시점이 아니라 사용자가 서버에 최근에 요청한 시간을 기준으로 30분 정도를 유지해주는 것입니다.

만약 사용자가 서비스를 사용하고 있으면, 세션의 생존 시간이 30분으로 계속 늘어나게 됩니다.  HttpSession 은 이 방식으로 생명주기를 관리합니다.

 

 

세션 타임아웃 설정

스프링 부트로 글로벌 설정 application.properties

server.servlet.session.timeout=60 : 60초, 기본은 1800(30분)

// (글로벌 설정은 분 단위로 설정. 60(1분), 120(2분), ...)
// 위에서 본 maxInactiveInterval

 

만약 특정 세션만 따로 관리하고 싶을 시,

session.setMaxInactiveInterval(1800); //1800

 

세션 타임아웃 발생

LastAccessedTime 이후로 timeout 시간이 지나면, WAS가 내부에서 해당 세션을 제거합니다.

 

 

 

 

정리

서블릿의 HttpSession 이 제공하는 타임아웃 기능 덕분에 세션을 안전하고 편리하게 사용 가능합니다.

실무에서 주의할 점은 세션에는 최소한의 데이터만 보관해야 합니다.

보관한 데이터 용량 * 사용자 수로 세션의 메모리 사용량이 급격하게 늘어나서 장애로 이어질 수 있기 때문입니다.

 

 

 

이렇게 쿠키, 세션, HttpSession을 이용한 로그인 처리에 대해 학습했습니다. 

하지만 여기서 추가로 처리해 주어야 할 게 있습니다. 지금은 로그인 안 한 사용자도 url만 알고 있으면 들어갈 수 있는 보안상 문제가 있기 때문입니다.

 

다음 포스팅에서는 [로그인 처리 - 필터, 인터셉터]을 통해

로그인 안 한 사용자를 걸러내는 방법을 알아보도록 하겠습니다.

 

 

 

 

 

참고 :

https://www.inflearn.com/course/%EC%8A%A4%ED%94%84%EB%A7%81-mvc-2/dashboard

 

스프링 MVC 2편 - 백엔드 웹 개발 활용 기술 - 인프런 | 강의

웹 애플리케이션 개발에 필요한 모든 웹 기술을 기초부터 이해하고, 완성할 수 있습니다. MVC 2편에서는 MVC 1편의 핵심 원리와 구조 위에 실무 웹 개발에 필요한 모든 활용 기술들을 학습할 수 있

www.inflearn.com