Skip to main content

Command Palette

Search for a command to run...

[Spring] 로그인 - 쿠키?세션?

Updated
3 min read
[Spring] 로그인 - 쿠키?세션?
S

Nice to meet u :) Im Backend Developer

1️⃣ 도메인

시스템이 구현해야하는 핵심 비즈니스 업무 영역

web은 도메인을 알고있지만 도메인은 web을 모르도록 설계해야한다.

2️⃣ 쿠키

쿼리 파라미터를 계속 유지하면서 요청하면 매우 어렵고 번거롭다. 만약 서버에서 로그인에 성공하면 HTTP응답에 쿠키를 담아 브라우저에게 전달한다. 그러면 브라우저는 해당 쿠키를 지속해서 이용할 수 있다.

  • 영속 쿠키 : 만료 날짜를 입력하면 해당 날짜까지 유지

  • 세션 쿠키 : 만료 날짜를 생략하면 브라우저 종료까지만 유지

👨🏻‍🏫 세션 쿠키

컨트롤러에서 쿠키를 설정할 수 있다.

  1. 로그인
    @PostMapping("/login")//로그인 성공시 세션쿠키 생성
    public String login(@Valid @ModelAttribute("loginForm") LoginForm loginForm,
                        BindingResult bindingResult, HttpServletResponse response) {
        Member login = loginService.login(loginForm.getLoginId(), loginForm.getPassword());
        //쿠키생성로직 -> 로그인 성공시 쿠키생성하고 response에 담아. 쿠키이름은 memberId
        //웹브라우저 종료전까지 회원 id를 서버에 계속 보내줌
        Cookie idCookie = new Cookie("memberId", String.valueOf(login.getId()));
        response.addCookie(idCookie);

        return "redirect:/";
    }

      @GetMapping("/")//cookievalue로 쿠키 조회, 로그인하지않은 사용자도 홈에 접근하니 required=false
      public String homeLogin(@CookieValue(name = "memberId",required = false) Long memberId,
                          Model model) {
      model.addAttribute("member", loginMember);//홈화면에 회원관련정보도 출력해야해서 member데이터도 담기
      return "loginHome";
    }
  • 결과

  1. 로그아웃
@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);//서버 해당 쿠키 종료 날짜 0지정
        response.addCookie(cookie);
    }
  • 결과

✅ 쿠키와 보안문제

  • 클라이언트가 쿠키를 강제로 변경할 수 있다.

  • 쿠키에 보관된 정보는 훔쳐갈 수 있다.
    ( 정보들이 웹브라우저에 보관, 네트워크 요청마다 클라이언트에서 서버로 전달 → 내 PC 혹은 네트워크 전송구간에서 털림)

  • 해커카 쿠키 훔쳐서 악의적인 요청이 가능하다.

✅ 쿠키와 보안문제 대안

  • 예측 불가능한 토큰 값 노출하기
    (서버에서 토큰과 사용자 id 매핑하고 서버에서 토큰관리)

  • 시간이 지나면 사용할 수 없도록 토큰 만료시간을 짧게 유지

3️⃣ 세션

서버에 중요한 정보를 보관하고 연결을 유지하는 방법

세션ID에 값을 저장해두고 쿠키로 클라이언트와 연결된다.

  • 서버 → 클라이언트 : 세션ID만 쿠키에 담아 전달

  • 클라이언트 → 쿠키저장소 : 세션ID쿠키 보관

  • 클라이언트 → 서버 : 세션ID쿠키 정보로 세션 저장소에서 조회 후 보관한 세션 정보를 사용

회원과 관련된 정보는 클라이언트에게 전달되지 않는다.

👨🏻‍🏫 HttpSession

서블릿을 통해 HttpSession을 생성하고 JSESSIONID이름의 쿠키를 사용한다.

    @PostMapping("/login")
    public String loginV3(@Valid @ModelAttribute LoginForm form, BindingResult bindingResult, HttpServletRequest request) {
        HttpSession session = request.getSession(true); //없으면 신규 생성 false=없으면 null
        session.setAttribute(SessionConst.LOGIN_MEMBER, login);//세션에 데이터 보관, 하나의 세션에 여러 값 보관 가능
        return "redirect:/";
    }

    @PostMapping("/logout")
    public String logoutV3(HttpServletRequest request) {
        HttpSession session = request.getSession(false);
        if (session != null) {
            session.invalidate(); //세션 제거
        }
        return "redirect:/";
    }

    @GetMapping("/")
    public String homeLoginV3(HttpServletRequest request,Model model) {
        HttpSession session = request.getSession(false);
        if (session == null) {
            return "home";
        }
        //세션값찾기
        Member loginMember = (Member) session.getAttribute(SessionConst.LOGIN_MEMBER);
        if(loginMember == null) {
            return "home";
        }
        model.addAttribute("member", loginMember);
        return "loginHome";
    }
  • session.setAttribute: 세션에 데이터 저장

  • request.getSession : 세션 조회

  • session.getAttribute : 세션에 보관한 데이터 조회

  • 결과 (JSESSIONID 확인)

✅ @SessionAttribute

세션 찾고, 세션의 데이터 찾는 복잡한 과정을 스프링에서 처리

    @GetMapping("/")
    public String homeLoginV3Spring(
@SessionAttribute(name = SessionConst.LOGIN_MEMBER,required = false) Member loginMember,
  Model model) {
        if(loginMember == null) {
            return "home";
        }
        model.addAttribute("member", loginMember);
        return "loginHome";
    }

✅ 세션 유지 시간

사용자가 서버에 요청한 시간(LastAccessedTime) 기준 30분 설정

application.properties

server.servlet.session.timeout=60 # 글로벌 설정 기본 1800 (30분)
session.setMaxInactiveInterval(1800) #1800초

More from this blog

[Spring] N+1문제 발생과 분석

✍️ 작성하게 된 이유 옷을 관리하는 서비스를 개발하면서 Cloth 엔티티와 그에 연관된 ClothWithAttributes, Attribute 데이터를 함께 조회하는 기능이 필요했다.그런데 연관 데이터를 조회할 때마다 쿼리가 폭발적으로 증가(N+1 문제) 하며, 성능이 급격히 저하되는 상황을 마주하게 되었다. Spring JPA의 대표적인 문제로 N+1임을 알고있었지만, 해결하는 방법은 Fetch Join밖에 몰랐다. 지연로딩되는 필드를 엔티...

Sep 17, 20256 min read
[Spring] N+1문제 발생과 분석

데이터베이스 기본 개념 정리

1️⃣ 데이터베이스(DB) & DBMS DB (Database): 일정한 규칙(스키마)에 따라 구조화되어 저장된 데이터의 집합. DBMS (Database Management System): DB를 제어/관리하는 시스템 소프트웨어. 특징: 실시간 접근 가능, 동시 공유 가능. 구조: 데이터베이스 → DBMS → 응용 프로그램 → 사용자 2️⃣ 엔티티(Entity) & 릴레이션(Relation) 엔티티: 여러 속성을 가진 "개체"...

Aug 5, 20252 min read
데이터베이스 기본 개념 정리

[Project] 날씨에 맞는 옷 추천 서비스 : 지그재그 크롤링 여정 기록 (1) ChromeDriver를 EC2에 설치하기

✍️ 작성하게 된 이유 무신사, 29cm는 Jsoup으로 충분히 크롤링이 가능했기 때문에, ZigZag도 당연히 Jsoup으로 처리될 것이라 생각했다. 무신사, 29cm와 마찬가지로 필요한 데이터는 모두 <script> 태그 안에 들어있었다. 하지만… 예상은 보기 좋게 빗나갔다. 🧪 현상 ✅ 로컬 크롤링 → 정상 작동 Jsoup으로 script 태그 내에서 대표 이미지와 상품명을 잘 추출 로컬 환경에서는 아무 문제 없이 작동 ❌ A...

Jul 30, 20253 min read
[Project] 날씨에 맞는 옷 추천 서비스 : 지그재그 크롤링 여정 기록 (1) ChromeDriver를 EC2에 설치하기

[Project] 날씨에 맞는 옷 추천 프로젝트: Selenium은 정말 필요한 선택이었을까? - 크롤링 삽질 기록

✍️ 작성하게 된 이유 날씨에 따라 옷을 추천해주는 서비스를 만들면서, 사용자가 입력한 구매 링크에서 옷 정보( 대표이미지, 상품명 )를 불러오는 기능이 필요했다. 처음에 해당 페이지를 동적 페이지로 판단했고, 자연스럽게 Selenium을 도입했다. 하지만 이 결정이 과연 최선이었는지는 수많은 시행착오 끝에야 알 수 있었다. 🕸️ Selenium을 선택한 이유 동적 페이지는 Jsoup으로 크롤링이 어렵다는 인식으로 처음부터 Selenium을 ...

Jul 28, 20254 min read
[Project] 날씨에 맞는 옷 추천 프로젝트: Selenium은 정말 필요한 선택이었을까? - 크롤링 삽질 기록

Soyulia's Blog

49 posts