관리 메뉴

샐님은 개발중

2. 로그인 - 세션 직접 구현 본문

스프링 MVC 2편 - 인프런 김영한/섹션 6,7 - 로그인 처리

2. 로그인 - 세션 직접 구현

샐님 2023. 7. 15. 01:13
728x90
반응형

 

package hello.login.web.session;

import org.springframework.stereotype.Component;
import org.thymeleaf.spring5.util.SpringRequestUtils;

import javax.servlet.http.Cookie;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.util.Arrays;
import java.util.Map;
import java.util.UUID;
import java.util.concurrent.ConcurrentHashMap;

/*
세션 관리
 */
@Component
public class SessionManager {

    public static final String SESSION_COOKIE_NAME ="mySessionId";

    /*
    동시성 문제(여러명이 들어왔을 때)를 해결하기 위해 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;
        }

        return Arrays.stream(request.getCookies())
                .filter(cookie -> cookie.getName().equals(cookieName))
                .findAny() // 순서상관없이 찾음
                .orElse(null);
    }
}

 

 

직접 구현한 세션 관리를 테스트해보자.

package hello.login.web.session;

import hello.login.domain.member.Member;
import org.junit.jupiter.api.Test;
import org.springframework.mock.web.MockHttpServletRequest;
import org.springframework.mock.web.MockHttpServletResponse;
import static org.assertj.core.api.Assertions.assertThat;
public class SessionManagerTest {
    SessionManager sessionManager = new SessionManager();
    @Test
    void sessionTest() {
        //세션 생성
        MockHttpServletResponse response = new MockHttpServletResponse();
        Member member = new Member();
        sessionManager.createSession(member, response);
        //요청에 응답 쿠키 저장
        MockHttpServletRequest request = new MockHttpServletRequest();
        request.setCookies(response.getCookies());
        //세션 조회
        Object result = sessionManager.getSession(request);
        assertThat(result).isEqualTo(member);
        //세션 만료
        sessionManager.expire(request);
        Object expired = sessionManager.getSession(request);
        assertThat(expired).isNull();
    }
}

 

 

 

테스트이기 때문에 HttpServletRequest , HttpservletResponse 의 역할을 대신해줄 MockHttpServletRequest , MockHttpServletResponse를 사용하면 임의의 응답과 요청 객체를 만들어 준다.

728x90
반응형