diff --git a/.idea/codeStyles/Project.xml b/.idea/codeStyles/Project.xml
deleted file mode 100644
index 4d5fd611..00000000
--- a/.idea/codeStyles/Project.xml
+++ /dev/null
@@ -1,58 +0,0 @@
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
diff --git a/.idea/codeStyles/codeStyleConfig.xml b/.idea/codeStyles/codeStyleConfig.xml
deleted file mode 100644
index 79ee123c..00000000
--- a/.idea/codeStyles/codeStyleConfig.xml
+++ /dev/null
@@ -1,5 +0,0 @@
-
-
-
-
-
\ No newline at end of file
diff --git a/README.md b/README.md
index 5fc2c69d..8517dc14 100644
--- a/README.md
+++ b/README.md
@@ -1,2 +1,194 @@
# outfit-of-the-weather
[#OOTW] 그날의 날씨를 보고 오늘의 코디를 공유하는 커뮤니티 By 빽엔두리
+
+### 클론 타겟
+
+- [꾸온꾸](https://korean.visitkorea.or.kr/detail/rem_detail.do?cotid=14a13909-134a-4a0e-a581-fc1ace95d925)
+- 현재 날씨를 확인하고 날씨에 맞춰 아바타를 코디한 후 전국의 여행지를 배경으로 사진을 찍어 업로드하는 게시판
+- 레퍼런스 이미지
+
+![스크린샷 2024-01-16 오후 3.17.30.png](https://prod-files-secure.s3.us-west-2.amazonaws.com/a557051f-c7ef-49b4-83dc-fbd1373d5693/6cef6311-6a06-48cd-9643-c1a131869bec/%E1%84%89%E1%85%B3%E1%84%8F%E1%85%B3%E1%84%85%E1%85%B5%E1%86%AB%E1%84%89%E1%85%A3%E1%86%BA_2024-01-16_%E1%84%8B%E1%85%A9%E1%84%92%E1%85%AE_3.17.30.png)
+
+### 개발 목표
+
+- 현재의 날씨를 보고 그에 맞는 아바타 코디를 선택해 이미지를 저장할 수 있고 저장한 이미지를 공유할 수 있는 게시판을 만드는 것을 1차 프로젝트 mvp로 삼고 진행
+- 이후 프로젝트 고도화에 따라 여러 기능을 추가적으로 구현 예정
+- 최종적으로 해당 콘텐츠를 메인으로 아바타 아이템 상거래가 가능한 커머스 시스템 도입을 목표로 진행 예정
+
+
+moscow
+
+- Must Have: 날씨 API 기반 게시글, 사용자 인증, 이미지 서비스
+- Should Have: 댓글, 무한 스크롤, 사용자 인증/인가, 소셜 로그인, 좋아요, 해시태그, 히스토리
+- Could Have: 검색, 서비스 구독, 사용자 권한, 사용자 인증/인가
+- Won't Have: 게시글 북마크, 알림, 이메일, 아바타 아이템 커머스, 통합회원, 휴대폰 인증
+
+
+
+### 기술 스택
+
+- java 17
+- springboot 3.2.0
+- spring security
+- jsonwebtoken jjwt 0.12.3
+- openfeign 4.1.0
+- spring data jpa
+- spring data redis
+- spring mail
+- green mail
+- minio 8.5.7
+
+### 팀 역할(SM, PO, Dev)
+
+
+
+1. 그라운드 룰
+ - **트러블 슈팅** 경험은 **2주 내**에 기록하고 공유
+ - [Git Wiki](https://github.com/spring-comes-to-us/outfit-of-the-weather/wiki)에 작성
+ - **가독성**을 중요하게 생각하여 코드 작성.
+ - 서로의 지식을 **공유**하고 코드와 의견을 **존중**하자
+2. 스크럼 규칙
+
+ | 진행 시간 | 내용 |
+ | --- | --- |
+ | 매일 오전 9시, 15~30분 (QR 찍고 진행) | - 어제 한 일(전날 커밋 기록) 공유 - 오늘 할 일(체크 리스트) 공유 |
+ | 금요일 오전 9시 (QR 찍자마자 진행) | - 스프린트(1주) 회고, 피드백 |
+3. 트러블 슈팅
+4. 프로젝트 관리
+
+## 주요 기능
+
+### 유저 기능
+
+- 회원가입
+ - Email 검증
+ - [RFC 5322](https://www.rfc-editor.org/rfc/rfc5322#section-3.4.1)기준의 정규 표현식 검증
+ - 최대 길이 255
+ - Password 검증
+ - 최소 한 개 이상의 영문자, 숫자, 특수문자가 각각 필요
+ - 최소 길이 8, 최대 길이 30
+ - DTO에서`@Password`커스텀 어노테이션 기반으로 진행
+ - Entity 생성자를 통한 2차 검증
+- 인증 코드 발송
+ - 회원가입 시 회원가입한 Email로 인증 코드 날리는 시스템 도입 (재전송 가능)
+ - Spring `MailSender`로 회원 가입자 이메일로 회원 가입 인증 코드 전송
+ - mail을 보내는 메서드는 `@Async`를 이용하여 비동기 처리
+ - 회원 정보를 저장하되 회원 인증 여부(`certified`)를 `false`로 저장하고 `true`가 될 때까지 로그인 서비스를 이용할 수 없게 설정
+ - `GreenMail` 라이브러리를 통한 인증 코드 전송 테스트 진행
+- 회원가입 코드 인증
+ - 가입한 이메일과 코드를 body에 담아 회원 가입 인증 진행
+ - 회원 테이블의 `certified` column을 `true`로 변경하고 로그인 하여 사용할 수 있게 만듬
+ - 인증 코드를 `Redis`에서 관리
+ - Redis TTL 설정으로 10분이 지난 인증 코드는 만료 처리
+- 로그인
+ - jwtToken 기반의 인증 시스템 도입
+ - 로그인 정보를 받은 후 유저 검증 후 `TokenProvider`를 통해 토큰 발급 및 검증
+ - 토큰 프로세스는 `jsonwebtoken jjwt` 라이브러리를 활용
+- Token 인증 필터
+ - SecurityFilter 내부에 `JwtAuthenticationFilter`를 추가
+ - `JwtAuthenticationFilter`에서 로그인 시 사용한 토큰을 파싱 후 `userId`를 기반으로 `UsernameAuthenticationToken`을 생성
+ 후 `SecurityContext`에 저장
+ - 로그인이 필요한 API에서 `Authentication` 객체를 받거나 직접 `principal`을 호출하는 메서드로 user 정보 호출
+
+### 현재 날씨 조회 기능
+
+- 기상청 OpenAPI 호출
+ - `OpenFeign`을 이용하여 기상청 API를 호출
+ - 기상청 API 응답 중 현재 날씨 정보(현재 기온, 강수 형태, 하늘 상태) 정보 조회
+ - 복잡한 OpenAPI 응답을 처리하기 위해 `@JsonDeserialize` 를 이용하여 API 응답 비직렬화 구현
+ - 요청이 제한된 기상청 API를 반복적으로 호출하는 문제 상황
+ - 추후 날씨 요청 위치를 Enum(시.도.동 등)으로 관리하고 Redis 기능을 이용하여 캐싱 기능 추가하여 외부 API 중복 호출을 줄일 예정
+- `Coordinate` 클래스 유효성 검사
+ - `@Grid` 커스텀 어노테이션 기반으로 Controller에서 유효성 검사 진행
+
+### 게시판 기능
+
+- 게시판 CRUD 기능 작성
+ - 로그인 여부에 따라 보여주는 정보 분기문 작성
+ - jmeter 부하 테스트 중 게시글 목록 조회 성능이 가장 안 좋은 것을 확인
+ - 추후 조회 성능을 더 분석해보고 개선 예정
+ - 목록 조회는 필터 조회 기능을 추가할 예정
+ - 게시글 생성 시 `OpenFeign`을 이용하여 기상청 OpenAPI 호출
+ - 게시글 작성 시 일교차(최저/최고 기온) 정보 반영하여 저장
+ - 레이어별 요청 값 유효성 검사
+ - 요청 dto에서 annotation으로 1차 유효성 검사
+ - Entity 생성시 2차 유효성 검사
+- 게시판 좋아요 기능
+ - 좋아요 여부(`isLike`) 바꿔주는 Patch API로 좋아요 기능 처리
+ - 게시글에 좋아요가 눌러진 숫자를 게시판 테이블에 추가
+ - 좋아요 동시성 문제 해결
+ - 한 사람이 한 게시물에 대해 여러 번 좋아요를 눌러서 생기는 문제
+ - 여러 사람이 한 게시물에 대하여 동시에 좋아요를 누르는 문제
+ - 비관적 락(`Pessimistic Lock`)을 사용하여 동시성 문제 해결
+ - jmeter 부하 테스트 중 두 번째로 성능이 안 좋은 것을 확인
+ (평균 시간이 전체 조회를 제외하고 가장 오래 걸림)
+ - 추후 성능 테스트 결과 확인 후 낙관적 락(`Optimistic Lock`) or `Redis`를 이용한 성능 개선 예정
+
+### 아바타 이미지 기능
+
+- 아바타 이미지 등록 및 조회 기능
+- Enum Validation annotation 생성 및 적용
+ - 모든 Enum 타입을 String으로 받고 그 Enum에 해당하는 값이 존재하지 않을 시 예외를 반환하는 annotation 생성
+- AvatarItem 전용 validation 클래스로 Entity validation 진행
+- 파일 저장 오픈 소스 `MinIO` 도입
+ - 추후 배포 시 S3와의 호환성을 위해 로컬 작업에서 MinIO를 도입
+
+ → S3 설정과 MinIO 설정 및 메서드가 동일하여 interface로 작성, 배포 버전에서는 S3 도입 예정
+
+ - Docker를 이용해 컨테이너로 이미지 저장소 생성
+ - `MinIO` 설정 후 이미지 저장 및 이미지 URL 반환 서비스 생성
+ - 게시판, 아바타 이미지에 로직 진행 중 이미지 저장 후 실패하면 이미지 삭제하는 로직 추가
+ - 대용량 이미지, 이미지 이외의 타입을 필터링 해주는 annotation 기반 validation 적용
+ - Post, AvatarItem 등 이미지를 사용하는 테이블이 많아 이미지를 관리할 수 있는 별도의 테이블 생성
+ - 추후 아바타 이미지/게시글의 수정 삭제 로직 후 발생 가능한 이미지 정합성 문제 확인 및 scheduling을 도입, 일정 시간 마다 `@Async` 를 이용하여 비동기적으로 처리 작업중
+ - filename의 unique 조건을 위해 UUID를 이용한 filename 생성 도입
+
+### CI 도입(Github Action)
+
+- pull request 시 테스트 코드를 모두 돌려보는 CI 환경 생성
+ - 로컬환경과 동일하게 Docker-compose 기반의 컨테이너 구동 후 test, build를 진행하는 방식 적용
+- github secret 기반으로 환경 변수 지정
+- Jacoco 테스트 리포트 자동화
+
+### Jacoco 도입
+
+- test coverage를 설정하여 기능 개발 시 test code를 작성하지 못한 케이스 확인
+- 메서드 개행 제한 및 분기문에 따른 test 기준 추가
+ - Class Line Coverage → 70% 이상
+ - Branch Coverage → 70% 이상
+ - Method Line Count → 20줄
+- 전체 프로젝트 테스트 커버리지 **✨98.48%✨** (2024.01.18 15:00 기준)
+
+### Check style 도입
+
+- google checkstyle 파일을 기반으로 팀 프로젝트에 적합한 코드 스타일 컨벤션 정의
+ - import 문, 여백 등 코드 스타일 컨벤션을 build 시 확인할 수 있도록 gradle 설정
+- 정의한 checkstyle 파일을 통해 Intelij 코드 스타일 자동 정렬 설정
+
+### JMeter 부하 테스트 실행
+
+- 수치로 나타난 객관적인 성능을 파악하기 위해 일부 API 부하 테스트 진행
+- 동시 접속이 많아지면 일부 요청이 예외를 반환하는 것을 확인
+ - 최대 DB Connection Pool을 조절함으로써 해결
diff --git a/build.gradle b/build.gradle
index d0bbb07d..5d9d90df 100644
--- a/build.gradle
+++ b/build.gradle
@@ -3,7 +3,6 @@ plugins {
id 'io.spring.dependency-management' version '1.1.4'
id 'java'
id 'jacoco'
- id 'checkstyle'
}
group = 'com.backendoori'
@@ -83,14 +82,6 @@ jacocoTestCoverageVerification {
}
}
}
- dependsOn jacocoTestReport
-}
-checkstyle {
- toolVersion = "10.12.5"
- configFile = file("${rootDir}/config/checkstyle/checkstyle.xml")
- configProperties = [
- "org.checkstyle.google.suppressionfilter.config" : "suppressions.xml",
- "org.checkstyle.google.suppressionxpathfilter.config": "suppressions.xml"
- ]
+ dependsOn jacocoTestReport
}
diff --git a/config/checkstyle/checkstyle.xml b/config/checkstyle/checkstyle.xml
deleted file mode 100644
index 2dcd740e..00000000
--- a/config/checkstyle/checkstyle.xml
+++ /dev/null
@@ -1,365 +0,0 @@
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
diff --git a/config/checkstyle/configuration_1_3.dtd b/config/checkstyle/configuration_1_3.dtd
deleted file mode 100644
index 40771d17..00000000
--- a/config/checkstyle/configuration_1_3.dtd
+++ /dev/null
@@ -1,54 +0,0 @@
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
diff --git a/config/checkstyle/suppressions.xml b/config/checkstyle/suppressions.xml
deleted file mode 100644
index 27e6779f..00000000
--- a/config/checkstyle/suppressions.xml
+++ /dev/null
@@ -1,8 +0,0 @@
-
-
-
-
-
-
diff --git a/config/checkstyle/suppressions_1_2.dtd b/config/checkstyle/suppressions_1_2.dtd
deleted file mode 100644
index 6fe68c90..00000000
--- a/config/checkstyle/suppressions_1_2.dtd
+++ /dev/null
@@ -1,16 +0,0 @@
-
-
-
-
-
-
diff --git a/src/main/java/com/backendoori/ootw/OotwApplication.java b/src/main/java/com/backendoori/ootw/OotwApplication.java
index d781021c..e29ef933 100644
--- a/src/main/java/com/backendoori/ootw/OotwApplication.java
+++ b/src/main/java/com/backendoori/ootw/OotwApplication.java
@@ -6,8 +6,8 @@
@SpringBootApplication
public class OotwApplication {
- public static void main(String[] args) {
- SpringApplication.run(OotwApplication.class, args);
- }
+ public static void main(String[] args) {
+ SpringApplication.run(OotwApplication.class, args);
+ }
}
diff --git a/src/main/java/com/backendoori/ootw/domain/BaseEntity.java b/src/main/java/com/backendoori/ootw/domain/BaseEntity.java
index 08e56e7a..442577b7 100644
--- a/src/main/java/com/backendoori/ootw/domain/BaseEntity.java
+++ b/src/main/java/com/backendoori/ootw/domain/BaseEntity.java
@@ -1,8 +1,8 @@
package com.backendoori.ootw.domain;
-import java.time.LocalDateTime;
import jakarta.persistence.EntityListeners;
import jakarta.persistence.MappedSuperclass;
+import java.time.LocalDateTime;
import lombok.Getter;
import org.springframework.data.annotation.CreatedDate;
import org.springframework.data.annotation.LastModifiedDate;