Token

ock·2023년 3월 8일
0

해싱 (Hashing)


  • 암호화만 가능.
  • 해시 함수(Hash Function)을 사용하여 암호화를 진행.
    • 항상 같은 길이의 문자열을 리턴한다.
    • 서로 다른 문자열에 동일한 해시 함수를 사용하면 반드시 다른 결과값이 나온다.
    • 동일한 문자열에 동일한 해시 함수를 사용하면 항상 같은 결과값이 나온다. SHA1에 특정 입력 값을 넣었을 때 어떤 결과가 리턴되는지 보여주는 예시 (https://www.convertstring.com/ko/Hash/SHA1)



<레인보우 테이블과 솔트>

항상 같은 결과값이 나온다는 특성을 이용해 해시 함수를 거치기 이전의 값을 알아낼 수 있도록 기록해놓은 표인 레인보우 테이블이 존재한다. 레인보우 테이블에 기록된 값의 경우에는 유출이 되었을 때 래싱을 했더라도 해싱 이전의 값을 알아낼 수 있으므로 보안상 위협이 될 수 있다.

이 때 활용할 수 있는 것이 솔트(Salt).

⇒ 해싱 이전 값에 임의의 값을 더해 데이터가 유출 되더라도 해싱 이전의 값을 알아내기 더욱 어렵게 만드는 방법.

솔트를 사용하게 되면 해싱 값이 유출 되더라도 솔트가 함께 유출된 것이 아니라면 암호화 이전의 값을 알아내는 것은 불가능에 가깝다.

그런데 왜 복호화가 불가능한 암호화 방식을 사용하는 것일까?

⇒ 해싱의 목적은 데이터 그 자체를 사용하는 것이 아니라 동일한 값의 데이터를 사용하고 있는지 여부만 확인하는 것.

💦 해싱민감한 데이터를 다루어야 하는 상황에서 데이터 유출의 위험성은 줄이면서 데이터의 유효성을 검증하기 위해서 사용되는 단방향 암호화 방식이다.




Token


토큰을 사용하면 사용자의 인증 정보를 서버가 아닌 클라언트 측에 저장할 수 있다.

=> 서버에서 유저의 상태를 관리하던 세션 기반 인증의 한계를 극복한 인증방식
토큰은 유저의 인증상태를 클라이언트에 저장할 수 있다.

  1. 사용자가 인증 정보를 담아 서버에 로그인 요청을 보낸다.
  2. 서버는 데이터베이스에 저장된 사용자의 인증 정보를 확인한다.
  3. 인증에 성공했다면 해당 사용자의 인증 및 권한 정보를 서버의 비밀 키와 함께 토큰으로 암호화한다.
  4. 생성된 토큰을 클라이언트로 전달한다.
  • HTTP 상에서 인증 토큰을 보내기 위해 사용하는 헤더인 Authorization 헤더를 사용하거나, 쿠키로 전달하는 등의 방법을 사용
  1. 클라이언트는 전달받은 토큰을 저장한다.
  • 저장하는 위치는 Local Storage, Session Storage, Cookie 등 다양
  1. 클라이언트가 서버로 리소스를 요청할 때 토큰을 함께 전달한다.
  • 토큰을 보낼 때에도 Authorization 헤더를 사용하거나 쿠키로 전달할 수 있음
  1. 서버는 전달받은 토큰을 서버의 비밀 키를 통해 검증한다. 이를 통해 토큰이 위조되었는지 혹은 토큰의 유효 기간이 지나지 않았는지 등을 확인할 수 있다.
  2. 토큰이 유효하다면 클라이언트의 요청에 대한 응답 데이터를 전송한다.



토큰 인증 방식의 장점


  • 무상태성

    서버가 유저의 인증상태를 관리하지 않는다. 서버는 비밀 키를 통해 클라이언트에서 보낸 토큰의 유효성만 검증하면 되기 때문에 무상태적인 아키텍처를 구축할 수 있다.

  • 확장성

    다수의 서버가 공통된 세션 데이터를 가질 필요가 없다는 것도 토큰 기반 인증의 장점. 이를 통해 서버를 확장하기 더 용이하다.

  • 어디서나 토큰 생성 가능

    토큰의 생성과 검증이 하나의 서버에서 이루어지지 않아도 되기 때문에 토큰 생성만을 담담하는 서버를 구축할 수 있다. 이를 잘 활용하면 여러 서비스 간의 공통된 인증 서버를 구현할 수 있다.

  • 권한 부여에 용이

    토큰은 인증상태, 접근권한 등 다양한 정보를 담을 수 있기 때문에 사용자 권한 부여에 용이하다. 이를 활용해 어드민 권한 부여 및 정보에 접근할 수 있는 밤위도 설정할 수 있다.




JWT (JSON Web Token)


토큰 기반 인증 구현 시 대표적으로 사용하는 기술.
JSON 객체에 정보를 담고 이를 토큰으로 암호화하여 전송할 수 있는 기술
클라이언트가 서버에 요청을 보낼 때, 인증정보를 암호화된 JWT 토큰으로 제공하고, 서버는 이 토큰을 검증하여 인증정보를 확인할 수 있다.

<JWT의 구성>

JWT는 다음 그림과 같이 . 으로 나누어진 세 부분이 존재하며 각각을 Header, Payload, Signature라고 부른다.

1. Header
Header에는 마치 HTTP의 헤더처럼 해당 토큰 자체를 설명하는 데이터가 담겨있다.
토큰의 종류, 그리고 시그니처를 만들 때 사용할 알고리즘을 JSON 형태로 작성한다.

{
  "alg": "HS256",
  "typ": "JWT"
}

이 JSON 객체를 base64 방식으로 인코딩하면 JWT의 첫 번째 부분인 Header가 완성된다.

2. Payload
Payload는 HTTP의 페이로드와 마찬가지로 전달하려는 내용물을 담고 있는 부분이다.
어떤 정보에 접근 가능한지에 대한 권한, 유저의 이름과 같은 개인정보, 토큰의 발급 시간 및 만료 시간 등의 정보들을 JSON 형태로 담는다.

{
  "sub": "someInformation",
  "name": "phillip",
  "iat": 151623391
}

이 JSON 객체를 base64로 인코딩하면 JWT의 두 번째 부분인 Payload가 완성된다.

3. Signature
Signature는 토큰의 무결성을 확인할 수 있는 부분이다.
Header와 Payload가 완성되었다면, Signature는 이를 서버의 비밀 키(암호화에 추가할 salt)와 Header에서 지정한 알고리즘을 사용하여 해싱한다.




<토큰 인증 방식의 한계>

Signature을 사용해서 위조된 토큰을 알아낼 수는 있지만, 토큰 자체가 탈취된다면 토큰 인증 방식의 한계가 드러난다.
무상태성
인증 상태를 관리하는 주체가 서버가 아니므로, 토큰이 탈취되어도 해당 토큰을 강제로 만료시킬 수 없다.
따라서 토큰이 만료될 때까지 사용자로 가장해 계속해서 요청을 보낼 수 있다.

유효기간
토큰이 탈취되는 상황을 대비해서 유효 기간을 짧게 설정하면, 사용자는 토큰이 만료될 때마다 다시 로그인을 진행해야 하기 때문에 좋지 않은 사용자 경험을 제공한다.
그렇다고 유효 기간을 길게 설정하면 토큰이 탈취될 경우 더 치명적으로 작용할 수 있다.

토큰의 크기
토큰에 여러 정보를 담을 수 있는 만큼, 많은 데이터를 담으면 그만큼 암호화하는 과정도 길어지고 토큰의 크기도 커지기 때문에 네트워크 비용 문제가 생길 수 있다.




액세스 토큰(Access Token)과 리프레시 토큰(Refresh Token)


토큰 인증의 한계를 극복하기 위해 다양한 방법들이 고안되었지만 이 중 대표적인 구현 방법은 액세스 토큰과 리프레시 토큰을 함께 사용하는 것.

Access Token
서버에 접근하기 위한 토큰

Refresh Token
액세스 토큰이 만료되었을 때 새로운 액세스 토큰을 새로 발급받기 위해 사용되는 토큰
Refresh Token의 목적은 Access Token의 유효 기간을 짧고, 자주 재발급 하도록 만들어 보안을 강화하면서도 사용자에게 잦은 로그아웃 경험을 주지 않도록 하는 것.










(코드스테이츠)

profile
어제의 ock보다 성장한 오늘의 ock_FE 개발자

0개의 댓글