logo

𝝅번째 알파카의 개발 낙서장

[OAuth2.0] ScribeJAVA로 OAuth2.0 인증서버 구축하기 - 2. 인증서버 설계하기

게시글
⏰ 2021-10-14 13:12:25

D O W N

https://user-images.githubusercontent.com/50317129/137171016-99af1db1-a346-4def-9329-6072b927bdc0.png
OAuth2.0 인증서버 구축기
이 게시글은 OAuth2.0 인증서버 구축기 시리즈의 12개 중 2번 째 게시글입니다.
https://user-images.githubusercontent.com/50317129/260317030-e4b8575b-f09e-47f4-ab70-168a817268c6.png

Table of Contents

https://user-images.githubusercontent.com/50317129/260317030-e4b8575b-f09e-47f4-ab70-168a817268c6.png

개요

OAuth2.0 프로토콜에 대한 사전 지식도 얻었으니, 이를 통해 인증서버를 직접 구축해보자.

Frontend, Backend 부분으로 나누어 웹 페이지를 통해 소셜 로그인을 수행하고, 직접 구현한 인증서버에서 이를 처리하는 시스템 일체를 구축하는 것이 궁극적인 목표다.

인증 서버의 인증 방식은 인가 코드 승인 (Authorization Code Grant)을 취한다.

제원

웹 페이지인 Frontend와 API 통신의 주체가 되는 인증서버인 Backend로 나뉜다.

Backend

💬 Language

🧱 Framework

  • 🔗
    Gradle
  • jakarta.servlet-api
  • lombok
  • jjwt
  • scribejava-apis
  • jersey-server
  • jersey-container-servlet
  • jersey-hk2
  • jersey-media-json-jackson

📦 Publish

Jersey는 RESTful API 프레임워크, scribeJAVA는 OAuth 프레임워크다.

Jersey 3는 Servlet 5.0 최신 스펙인 jakarta.*을 사용한다. 때문에 이를 구현한 Tomcat 10을 사용한다. Tomcat 10 미만 혹은 jakarta.*을 구현하지 않는 등 최신 서블릿 스펙을 사용할 수 없는 WAS의 경우 이 프로젝트는 배포할 수 없다.

Frontend

💬 Language

🧱 Framework

📦 Publish

원래는 Raspberry Pi로 배포하고 있었으나, 굳이 그럴 필요가 없어서 GutHub Pages로 배포한다.

Frontend는 단순히 인증 서버와의 API 통신 및 정보 표출 정도만 수행하므로, React 같은 복잡한 기술은 최대한 지양하고 최대한 기본 기능으로만 구성했다.

구성

인증서버는 크게 세 부분으로 나눌 수 있다.

  • controller - 지정한 URL의 요청을 받아 process에게 위임한다.
  • process - API의 동작을 수행하고, 응답을 생성한다
  • module - API 동작 수행에 필요한 모듈

이는 패키지로 구분한다.

controller

Jersey 프레임워크는 controller 영역에서 동작한다.

기존의 Servlet은 RESTful API를 설계하기위해 직접 구현해야할 부분이 존재한다.

RESTful API를 효과적으로 설계하기 위해 Jersey 프레임워크를 사용한다.

controller는 정의된 process로 요청을 전달하고, process가 반환하는 응답을 요청자에게 다시 전달한다.

  • /login

    • GET /:platform - 플랫폼별 인증 URL 반환 API
    • PUT /put - 정보 제공 동의 갱신 URL 반환 API
    • POST /:platform - 플랫폼별 로그인 API
    • POST /auto - 자동 로그인 API
  • /logout

    • POST / - 로그아웃 API
  • /revoke

    • POST / - 연동 해제 API
  • /userinfo

    • POST / - 사용자 정보 API

해당 프로젝트에선 위 controller들이 구현될 것이다.

process

process에선 실제 로직이 구현되는 영역이다. 이 때 필요하다면 module을 호출하기도 한다.

process수행 결과를 정해진 형식의 JSON을 기반으로 응답 객체를 만들어 controller로 반환한다.

JSON

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
{
    flag: true,
    title: "success",
    message: "test api response success",
    body: {
        key1: "value1",
        key2: true,
        key3: 4932,
        key4: [
            "A",
            "B",
            "C"
        ],
        key5: 93.203
    }
}
파라미터필수 여부데이터 형식내용
flagYboolean응답 정상 여부
titleYString응답 제목
messageYString응답 메세지
bodyYString | Number | boolean | Array | Object응답 정상 여부

응답 명세는 위와 같다.

process는 GET, POST 등 HTTP Method에 따라 나누어 관리한다.

module

module은 여러 process에서 반복적으로 사용되는 로직의 모듈화를 구현한 객체다.

이 중 핵심 기능인 OAuth의 인증을 처리하는 모듈은 scribeJAVA 기반으로 작성된다.

로그인, 사용자 정보 요청 등 인증과 관련된 로직을 수행하는 process들은 모두 위 모듈을 호출하여 사용함으로써 중복 코드를 줄이고 유지보수성을 높인다.

OAuth 모듈

이 프로젝트의 Backend에 가장 중요한 역할을 담당하는 모듈로, OAuth 인증을 구현한다.

요즘은 어느정도 규모있는 플랫폼이라면 OAuth 서비스를 제공한다. 플랫폼은 달라도 OAuth를 준수한다면 모두 공통된 요청 받아 공통된 응답을 제공할 것이다. 공통 프로토콜을 사용하므로, 인증 모듈 하나를 만들어 모든 인증을 거기서 처리하면 될 것이다.

하지만 문제가 하나 있다. 아무리 공통 프로토콜을 사용한다지만, 플랫폼마다 추가적인 파라미터 혹은 헤더를 요구하기도 한다. 조건문을 떡칠한다면 모를까, 고작 인증 모듈 하나에서 플랫폼별 요구사항을 하나하나 처리하기엔 문제가 많다. 조건문을 통한 분기 역시 유지보수적인 측면에서 그리 좋은 방식은 아니다.



OAuth는 객체라기보단 일종의 개념에 더 가깝다. 이렇게 추상적인 개념의 형태는 JAVA의 인터페이스 내지는 추상 객체가 적합하다.

인증 모듈의 경우, 몇몇 플랫폼의 특이 사항으로 인해 문제가 발생하니, 기본적으로는 인증 모듈의 공통 로직을 사용하되, 추가적인 요구를 하는 플랫폼에 한해 로직을 변경하는 것이 효율적일 것이다. 따라서 인증 모듈은 인터페이스보단 추상 객체가 더 어울릴 것이다.

이렇게 인증 추상 모듈을 하나 생성하고, 각 소셜 플랫폼마다 인증 모듈을 상속받아 구현하게끔 설계하면 될 것이다.

문제가 없는 플랫폼은 기 선언된 인증 모듈의 메서드를 그대로 사용하고, 추가 요구사항이 있는 플랫폼은 해당 플랫폼의 인증 모듈에서 메서드를 오버라이딩하여 요구사항에 맞게 재설계한다.

인증 모듈 기능

구현해야할 기능은 아래와 같다.

  • 플랫폼 로그인 URL 생성 로직
  • 정보 제공 동의 갱신 URL 생성 로직
  • 서비스 로그인 로직 (인가코드 -> Access Token)
  • 자동 로그인 로직
  • 로그아웃 로직
  • 사용자 정보 호출 로직
  • 연동 해제 로직

최소한 위 기능이 구현되어야한다.

사용자 정보 호출의 경우, 플랫폼마다 주는 응답이 천차만별로 다르다.

사용자 정보는 OAuth로 전달받은 Access Token으로 Service Provider에 정보를 요청하여 받을 수 있다. OAuth는 어디까지나 인증을 받기 위한 표준 프로토콜로, 인증 토큰을 통해 수행하는 부가적인 활동에 대해서는 별다른 제약이 없으므로, Access Token 발급 이후의 API 요청 및 응답은 플랫폼마다 천차만별로 다르다.

때문에 사용자 정보 호출 로직의 경우, OAuth 표준 프로토콜에 구애받지 않는다. 이에 대응하기 위해 추상 메서드로 선언하여 각 모듈에서 응답을 정제하여 반환하도록 강제하자.

구조도

OAuth 모듈을 사용하는 API의 구조를 도식화하면 위와 같다.

Jersey가 URL을 캐치하여 정해진 process로 전달한다. process는 인증을 수행하기 위해 해당 플랫폼의 OAuth 구현체를 호출하여 로직을 수행할 것이다. 이후 결과가 반환되어 사용자에게 전달되는 구조다.

정리

지금까지 인증 서버에 대한 설계를 진행했다. 다음 장엔 위 설계를 토대로 인증 서버를 본격적으로 구축해본다. 그 중에서도 scribeJAVA와 OAuth 인증 모듈에 대해 다룰 예정이다.

구축 순서는 module -> process -> controller 순이다.

가장 안쪽에 위치한 프로세스가 module이므로, module에서부터 바깥쪽으로 순차적으로 구현할 예정이다.

🏷️ Related Tag

# JAVA
# OAuth2.0

😍 읽어주셔서 감사합니다!
도움이 되셨다면, 💝공감이나 🗨️댓글을 달아주시는 건 어떤가요?
블로그 운영에 큰 힘이 됩니다!
https://blog.itcode.dev/posts/2021/10/14/oauth2-java-server-2