Skip to content

Commit 5f0b079

Browse files
gunwoong1630HwuanPagejohnhuh619garusitellCopilot
authored
release (#147)
* git initialize * feature/swagger-03-gunwoong (#5) * feat: 공통 도메인 구현 * feat: 메인 어플리케이션에 추가 * feat: swagger 추가 * feat: swagger 추가 * feature/base domain 04 gunwoong (#6) * feat: 공통 도메인 구현 * feat: 메인 어플리케이션에 추가 * feature/OpenAPI Test/02-HwuanPage * feature/OpenAPI Test/02-HwuanPage * Update SurfingForecastApiClient.java * feature/APICallTest-02-HwuanPage * feature/EntityInit-13-HwuanPage * feature/EntityInit-13-HwuanPage * feature/JellyfishEntityInit-13-HwuanPage * Update FishingType.java * feature/EntityInitialize-13-HwuanPage * feat: entity, repositor 구현 * feat: 예상 dto 구현 * chore: 의존성 추가 * feat: 로그인 구현 & 이후 토큰 발급 로직 구현 * fix: AuthCotnroller 수정 * fix: 클라이언트에서 카카오에서 코드를 받아 서버로 post 하게 수정 * feat: 토큰 검증 * feat: refresh token 블랙리스트 처리 로직 구현 * feat: refresh 토큰 블랙리스트 처리 & 재발급 로직 구현 * feat: SecurityFilterChain 엔드 포인트 허용 * feat: refresh 토큰 블랙리스트 검증 로직 구현 * feat: redis에서 refreshToken 블랙리스트 검증 * refactor: controller에 강하게 결합 되어 있던 로직들 분리 * test: member 관련 테스트 * chore: 하드코딩한 중요 값 Intellij IDEA 환경변수로 설정 * refactor: state 관리를 위해 세션 추가 * feat: member 정보 조회하는 서비스 로직 구현 * feat: member 정보 조회하는 서비스 로직 구현 * format: naver formatter로 포매팅 * chore: application-dev * fix: customException 처리 * Feat/meeting interface (#19) * feat : MeetingService 인터페이스 구현 * feat : ParticipantResponse * feat : MeetingListResponse 구현 * feat : MeetingDetailResponse구현 * feat : MeetingDetailAndMemberResponse 구현 * feat : ListSpot 구현 * feat : DetailSpot 구현 * feat : CreateMeetingRequest 구현 * feat : Tag 구현 * feat : Long -> long 변경 서비스와 Entity내에서 null값이 절대 나오지 않는다고 판단하는 값을 long으로 변경하였습니다. * feat : MeetingService.java -> 무한페이지로딩형식으로 바꾸었습니다. * Update src/main/java/sevenstar/marineleisure/meeting/Dto/Response/MeetingDetailResponse.java Co-authored-by: Copilot <[email protected]> --------- Co-authored-by: Copilot <[email protected]> * Feature/FavoritesAndAlertInterface-16-HwuanPage * feature/FavoritesAndAlertInterface-16-HwuanPage * Update AlertMapper.java * Update JellyfishRegionDensityRepository.java * Update AlertController.java * Update FavoriteController.java * Update FavoriteRepository.java * Update AlertController.java * Update JellyfishSpieces.java * Update JellyfishRegion.java * Update JellyfishRegion.java * feature/CustomExceptionInit-22-HwuanPage * feature/CustomExceptionInit-22-HwuanPage * Errorcode interface Change * Refactor application.yml 환경변수 설정 (#25) * refactor: application.yml 환경변수 설정 * Rename: 오타 수정 * Feature/spot service interface 29 gunwoong (#30) * feat: api * feat: api 스케줄링 * feat: spot service inteface * Feature/api scheduler 15 gunwoong (#28) * feat: api * feat: api 스케줄링 * feat: spot service inteface * test: remove legacy test * feat: apply open meteo * test: apply api test * feat: spot service (#34) * feat: spot service * feat: spot service 고도화 및 조회도 관련 서비스 추가 * feat: 조회도 관련 서비스 추가 * feat: 조회도 관련 서비스 추가 * feat: 조회도 관련 서비스 추가 * hotfix: duplicated controller method * feature/FavoriteCRUD-33-HwuanPage * DELETE COMPLETE * UPDATE COMPLETE * search COMPLETE * Before gunwoong * FavoriteCRUD create * feat/domain test * FavoriteSpotServiceTest * FavoriteSpotServiceTest * feature/FavoriteCURD-33-HwuanPage * add some description on service * Update FavoriteServiceImpl.java * Feature/spot preview 40 gunwoong (#41) * feat: spot preview & 리팩토링 * feat: spot preview & 리팩토링 * hotfix: jpa metamodel fix * fix: error fix * fix: 소셜 로그인 재시도 시 닉네임 UNIQUE 제약 위반 오류 발생 (#42) * fix: 로그아웃 후 재로그인 시 동일 정보로 db에 insert 하던 버그 수정 * refactor: 로그아웃 후 재로그인 시 동일 정보로 db에 insert 수정 사항 리팩토링 * test: 변경사항에 따른 테스트 코드 수정 * hofix: bug fix * Feature/Alert-22-HwuanPage * Create Pdf Parser * Web crawler run perpectly,but pdfparser do not work well * PDF parse to stack DB complete with OPENAI * CallAlert Complete * JellyFish PDF parsing work well * feature/ControllerTest Complete * feature/JellyfishAlert-26-HwuanPage * feat: 즐겨찾기 추가 및 리팩토링 (#49) * feat: 즐겨찾기 추가 및 리팩토링 * refactor: 리팩토링 * feat: 카카오 로그인을 stateless 하게 변경한다 (#51) * refactor: 기존 state 사용 방식 -> stateless 방식으로 변경 * refactor: 기존 state 사용 방식 -> stateless 방식으로 변경으로 인해 필요 없는 엔드 포인트 삭제 * test: 변경사항 test 수정 * feat: 카카오 측에서 인증 실패시에 반환 하는 에러 처리하는 코드 구현 * test: 카카오 측에서 인증 실패시에 반환 하는 에러 처리하는 테스트 추가 * fix: 주석 제거 * fix: exception 변경 * Feat/meeting service (#46) * WIP: Rebase를 위한 임시 저장 * feat : Meeting.java -> Meeting 엔터티 @builder 를 상위 어노테이션에 일단 추가시켰습니다. * feat : Meeting.java -> Meeting 엔터티 @builder 를 상위 어노테이션에 일단 추가시켰습니다. * feat : Meeting.java -> Meeting 엔터티 @builder 를 상위 어노테이션에 일단 추가시켰습니다. * Delete MeetingServiceImplReview.md * Delete MeetingServiceUserFlow.md * feat : 패키지명 변경 이슈 -> 패키지 명을 컨벤션에 따른 이름으로 변경했습니다. * feat : MeetingController.java long participantCount = participantRepository.countMeetingIdMember -> long participantCount = participantRepository.countMeetingId로 수정하였습니다. * feat : MeetingError.java MeetingError.java 를 추가하였습니다. * feat : MeetingMapper MeetingServiceImpl에서 사용중이었던 Mapper를 분리하였습니다. * feat : MeetingService.java 패키지 명 수정으로 인해서 수정사항이 있었습니다. * feat : MeetingServiceImpl.java 트랜잭션 관리 명확화 하였습니다. validate 패키지를 개선하였습니다. joinMeeting 중복 참여 제한 로직을 강화하였습니다. * `getAllMeetings(Long cursorId, int size)`: * 목적: 모든 모임 목록을 페이징 처리하여 조회합니다. cursorId를 사용하여 무한 스크롤과 같은 커서 기반 페이징을 지원합니다. * 특징: @transactional(readOnly = true)를 통해 읽기 전용 트랜잭션으로 최적화되었습니다. * `getMeetingDetails(Long meetingId)`: * 목적: 특정 모임의 상세 정보를 조회합니다. 호스트, 장소, 태그 등 연관된 정보를 함께 가져옵니다. * 개선 예정: 현재 N+1 문제가 발생할 수 있어, 향후 Fetch Join을 통한 성능 최적화가 필요합니다. * `getStatusMyMeetings(Long memberId, Long cursorId, int size, MeetingStatus meetingStatus)`: * 목적: 특정 회원의 상태별(예: 모집 중, 완료) 모임 목록을 페이징 처리하여 조회합니다. * `getMeetingDetailAndMember(Long memberId, Long meetingId)`: * 목적: 호스트가 자신의 모임 상세 정보와 참여자 목록을 조회합니다. 참여자들의 닉네임을 함께 제공합니다. * `countMeetings(Long memberId)`: * 목적: 특정 회원이 참여한 모임의 총 개수를 반환합니다. * `joinMeeting(Long meetingId, Long memberId)`: * 목적: 회원이 특정 모임에 참여합니다. * 주요 개선: 모임 상태 검증(verifyRecruiting), 중복 참여 검증(`verifyNotAlreadyParticipant`), 모임 정원 초과 검증(verifyMeetingCount) 로직이 강화되었습니다. * 개선 예정: 동시성 문제(Race Condition) 해결을 위한 비관적 락(Pessimistic Lock) 적용이 필요합니다. * `leaveMeeting(Long meetingId, Long memberId)`: * 목적: 회원이 모임에서 탈퇴합니다. * 주요 개선: 호스트 탈퇴 방지(verifyNotHost), 모임 상태에 따른 탈퇴 가능 여부 검증(verifyLeave) 로직이 추가되었습니다. * 개선 예정: MEETING_NOT_FOUND 대신 CANNOT_LEAVE_COMPLETED_MEETING과 같은 더 구체적인 에러 코드 적용이 필요합니다. * `createMeeting(Long memberId, CreateMeetingRequest request)`: * 목적: 새로운 모임을 생성합니다. 호스트를 참여자로 자동 등록하고 태그 정보를 저장합니다. * `updateMeeting(Long meetingId, Long memberId, UpdateMeetingRequest request)`: * 목적: 기존 모임의 정보를 수정합니다. 호스트만 수정할 수 있도록 검증합니다. * `deleteMeeting(Member member, Long meetingId)`: * 목적: 모임을 삭제합니다. * 개선 예정: 물리적 삭제 대신 논리적 삭제(Soft Delete) 방식 도입을 고려 중입니다. * feat : MeetingValidate.java,MemberValidate.java,ParticipantValidate,SpotValidate,TagValidate.java 검증로직을 추가하였습니다. * feat : MemberError.java , ParticipantRepository 기능을 추가하였습니다. --------- Co-authored-by: Hwang Seong Cheol a.k.a Hwuan Page <[email protected]> * Feature/integration init (#54) * feature/IntegrationSet(test&Build)-52-HwuanPage * data.sql unique update * image build needs * ignore dev.yml * remove dev.yml tracking and ignore it * prod * proded * Feature/activities 17 audwls239 (#56) * feature: 컨트롤러, 서비스 생성 * feature: 활동별 지수 조회(위치 기반) * feature: DTO 추가 * feature: 활동별 지수 조회(글로벌) 추가, 컨트롤러 수정 * feature: 활동별 지수 상세 조회(미완성) * feature: 해양 정보 조회 * feature: 활동 상세 조회 --------- Co-authored-by: Gunwoong cho <[email protected]> * feat : ParticipantError 입니다. * hotfix: error fix * fix : Directory 수정사항입니다. (#57) * hotfix: error fix * feat: member delete (#58) * fix: 멤버 삭제 구현 * feat: 멤버 삭제, 위/경도 수정 구현 * test: 테스트 수정 * Delete src/main/java/sevenstar/marineleisure/meeting/repository/MemberRepository.java * Delete src/main/java/sevenstar/marineleisure/meeting/repository/OutdoorSpotSpotRepository.java * Delete src/main/resources/test.http --------- Co-authored-by: LEESUNBIN <[email protected]> Co-authored-by: Gunwoong cho <[email protected]> * fix : ParticipantRepository (#59) existsByMeetingIdAndUserId 로 수정하였습니다. * fix : ParticipantRepository (#60) memberId -> userId로 수정하였습니다. * fix: token (#61) * feature/base domain 04 gunwoong (#6) * feat: 공통 도메인 구현 * feat: 메인 어플리케이션에 추가 * feature/CustomExceptionInit-22-HwuanPage * feature/CustomExceptionInit-22-HwuanPage * Errorcode interface Change * Refactor application.yml 환경변수 설정 (#25) * refactor: application.yml 환경변수 설정 * Rename: 오타 수정 * Feature/spot service interface 29 gunwoong (#30) * feat: api * feat: api 스케줄링 * feat: spot service inteface * feat: 카카오 로그인을 stateless 하게 변경한다 (#51) * refactor: 기존 state 사용 방식 -> stateless 방식으로 변경 * refactor: 기존 state 사용 방식 -> stateless 방식으로 변경으로 인해 필요 없는 엔드 포인트 삭제 * test: 변경사항 test 수정 * feat: 카카오 측에서 인증 실패시에 반환 하는 에러 처리하는 코드 구현 * test: 카카오 측에서 인증 실패시에 반환 하는 에러 처리하는 테스트 추가 * fix: 주석 제거 * fix: exception 변경 * Feat/meeting service (#46) * WIP: Rebase를 위한 임시 저장 # Conflicts: # src/main/java/sevenstar/marineleisure/global/exception/enums/CommonErrorCode.java # src/main/java/sevenstar/marineleisure/global/swagger/SwaggerController.java * feat : Meeting.java -> Meeting 엔터티 @builder 를 상위 어노테이션에 일단 추가시켰습니다. * feat : Meeting.java -> Meeting 엔터티 @builder 를 상위 어노테이션에 일단 추가시켰습니다. * feat : Meeting.java -> Meeting 엔터티 @builder 를 상위 어노테이션에 일단 추가시켰습니다. * Delete MeetingServiceImplReview.md * Delete MeetingServiceUserFlow.md * feat : 패키지명 변경 이슈 -> 패키지 명을 컨벤션에 따른 이름으로 변경했습니다. * feat : MeetingController.java long participantCount = participantRepository.countMeetingIdMember -> long participantCount = participantRepository.countMeetingId로 수정하였습니다. * feat : MeetingError.java MeetingError.java 를 추가하였습니다. * feat : MeetingMapper MeetingServiceImpl에서 사용중이었던 Mapper를 분리하였습니다. * feat : MeetingService.java 패키지 명 수정으로 인해서 수정사항이 있었습니다. * feat : MeetingServiceImpl.java 트랜잭션 관리 명확화 하였습니다. validate 패키지를 개선하였습니다. joinMeeting 중복 참여 제한 로직을 강화하였습니다. * `getAllMeetings(Long cursorId, int size)`: * 목적: 모든 모임 목록을 페이징 처리하여 조회합니다. cursorId를 사용하여 무한 스크롤과 같은 커서 기반 페이징을 지원합니다. * 특징: @transactional(readOnly = true)를 통해 읽기 전용 트랜잭션으로 최적화되었습니다. * `getMeetingDetails(Long meetingId)`: * 목적: 특정 모임의 상세 정보를 조회합니다. 호스트, 장소, 태그 등 연관된 정보를 함께 가져옵니다. * 개선 예정: 현재 N+1 문제가 발생할 수 있어, 향후 Fetch Join을 통한 성능 최적화가 필요합니다. * `getStatusMyMeetings(Long memberId, Long cursorId, int size, MeetingStatus meetingStatus)`: * 목적: 특정 회원의 상태별(예: 모집 중, 완료) 모임 목록을 페이징 처리하여 조회합니다. * `getMeetingDetailAndMember(Long memberId, Long meetingId)`: * 목적: 호스트가 자신의 모임 상세 정보와 참여자 목록을 조회합니다. 참여자들의 닉네임을 함께 제공합니다. * `countMeetings(Long memberId)`: * 목적: 특정 회원이 참여한 모임의 총 개수를 반환합니다. * `joinMeeting(Long meetingId, Long memberId)`: * 목적: 회원이 특정 모임에 참여합니다. * 주요 개선: 모임 상태 검증(verifyRecruiting), 중복 참여 검증(`verifyNotAlreadyParticipant`), 모임 정원 초과 검증(verifyMeetingCount) 로직이 강화되었습니다. * 개선 예정: 동시성 문제(Race Condition) 해결을 위한 비관적 락(Pessimistic Lock) 적용이 필요합니다. * `leaveMeeting(Long meetingId, Long memberId)`: * 목적: 회원이 모임에서 탈퇴합니다. * 주요 개선: 호스트 탈퇴 방지(verifyNotHost), 모임 상태에 따른 탈퇴 가능 여부 검증(verifyLeave) 로직이 추가되었습니다. * 개선 예정: MEETING_NOT_FOUND 대신 CANNOT_LEAVE_COMPLETED_MEETING과 같은 더 구체적인 에러 코드 적용이 필요합니다. * `createMeeting(Long memberId, CreateMeetingRequest request)`: * 목적: 새로운 모임을 생성합니다. 호스트를 참여자로 자동 등록하고 태그 정보를 저장합니다. * `updateMeeting(Long meetingId, Long memberId, UpdateMeetingRequest request)`: * 목적: 기존 모임의 정보를 수정합니다. 호스트만 수정할 수 있도록 검증합니다. * `deleteMeeting(Member member, Long meetingId)`: * 목적: 모임을 삭제합니다. * 개선 예정: 물리적 삭제 대신 논리적 삭제(Soft Delete) 방식 도입을 고려 중입니다. * feat : MeetingValidate.java,MemberValidate.java,ParticipantValidate,SpotValidate,TagValidate.java 검증로직을 추가하였습니다. * feat : MemberError.java , ParticipantRepository 기능을 추가하였습니다. --------- Co-authored-by: Hwang Seong Cheol a.k.a Hwuan Page <[email protected]> * feat: 테스트용 액세스 토큰 생성 * feature/base domain 04 gunwoong (#6) * feat: 공통 도메인 구현 * feat: 메인 어플리케이션에 추가 * Refactor application.yml 환경변수 설정 (#25) * refactor: application.yml 환경변수 설정 * Rename: 오타 수정 * Feature/spot service interface 29 gunwoong (#30) * feat: api * feat: api 스케줄링 * feat: spot service inteface * feature/base domain 04 gunwoong (#6) * feat: 공통 도메인 구현 * feat: 메인 어플리케이션에 추가 * feature/CustomExceptionInit-22-HwuanPage * feature/CustomExceptionInit-22-HwuanPage * Errorcode interface Change * Feature/spot service interface 29 gunwoong (#30) * feat: api * feat: api 스케줄링 * feat: spot service inteface * Feature/api scheduler 15 gunwoong (#28) * feat: api * feat: api 스케줄링 * feat: spot service inteface * test: remove legacy test * feat: apply open meteo * test: apply api test * feature/FavoriteCRUD-33-HwuanPage * DELETE COMPLETE * UPDATE COMPLETE * search COMPLETE * Before gunwoong * FavoriteCRUD create * feat/domain test * FavoriteSpotServiceTest * FavoriteSpotServiceTest * feature/FavoriteCURD-33-HwuanPage * add some description on service * Update FavoriteServiceImpl.java * fix: error fix * fix: 소셜 로그인 재시도 시 닉네임 UNIQUE 제약 위반 오류 발생 (#42) * fix: 로그아웃 후 재로그인 시 동일 정보로 db에 insert 하던 버그 수정 * refactor: 로그아웃 후 재로그인 시 동일 정보로 db에 insert 수정 사항 리팩토링 * test: 변경사항에 따른 테스트 코드 수정 * hofix: bug fix * feat: 카카오 로그인을 stateless 하게 변경한다 (#51) * refactor: 기존 state 사용 방식 -> stateless 방식으로 변경 * refactor: 기존 state 사용 방식 -> stateless 방식으로 변경으로 인해 필요 없는 엔드 포인트 삭제 * test: 변경사항 test 수정 * feat: 카카오 측에서 인증 실패시에 반환 하는 에러 처리하는 코드 구현 * test: 카카오 측에서 인증 실패시에 반환 하는 에러 처리하는 테스트 추가 * fix: 주석 제거 * fix: exception 변경 * Feat/meeting service (#46) * WIP: Rebase를 위한 임시 저장 # Conflicts: # src/main/java/sevenstar/marineleisure/global/exception/enums/CommonErrorCode.java # src/main/java/sevenstar/marineleisure/global/swagger/SwaggerController.java * feat : Meeting.java -> Meeting 엔터티 @builder 를 상위 어노테이션에 일단 추가시켰습니다. * feat : Meeting.java -> Meeting 엔터티 @builder 를 상위 어노테이션에 일단 추가시켰습니다. * feat : Meeting.java -> Meeting 엔터티 @builder 를 상위 어노테이션에 일단 추가시켰습니다. * Delete MeetingServiceImplReview.md * Delete MeetingServiceUserFlow.md * feat : 패키지명 변경 이슈 -> 패키지 명을 컨벤션에 따른 이름으로 변경했습니다. * feat : MeetingController.java long participantCount = participantRepository.countMeetingIdMember -> long participantCount = participantRepository.countMeetingId로 수정하였습니다. * feat : MeetingError.java MeetingError.java 를 추가하였습니다. * feat : MeetingMapper MeetingServiceImpl에서 사용중이었던 Mapper를 분리하였습니다. * feat : MeetingService.java 패키지 명 수정으로 인해서 수정사항이 있었습니다. * feat : MeetingServiceImpl.java 트랜잭션 관리 명확화 하였습니다. validate 패키지를 개선하였습니다. joinMeeting 중복 참여 제한 로직을 강화하였습니다. * `getAllMeetings(Long cursorId, int size)`: * 목적: 모든 모임 목록을 페이징 처리하여 조회합니다. cursorId를 사용하여 무한 스크롤과 같은 커서 기반 페이징을 지원합니다. * 특징: @transactional(readOnly = true)를 통해 읽기 전용 트랜잭션으로 최적화되었습니다. * `getMeetingDetails(Long meetingId)`: * 목적: 특정 모임의 상세 정보를 조회합니다. 호스트, 장소, 태그 등 연관된 정보를 함께 가져옵니다. * 개선 예정: 현재 N+1 문제가 발생할 수 있어, 향후 Fetch Join을 통한 성능 최적화가 필요합니다. * `getStatusMyMeetings(Long memberId, Long cursorId, int size, MeetingStatus meetingStatus)`: * 목적: 특정 회원의 상태별(예: 모집 중, 완료) 모임 목록을 페이징 처리하여 조회합니다. * `getMeetingDetailAndMember(Long memberId, Long meetingId)`: * 목적: 호스트가 자신의 모임 상세 정보와 참여자 목록을 조회합니다. 참여자들의 닉네임을 함께 제공합니다. * `countMeetings(Long memberId)`: * 목적: 특정 회원이 참여한 모임의 총 개수를 반환합니다. * `joinMeeting(Long meetingId, Long memberId)`: * 목적: 회원이 특정 모임에 참여합니다. * 주요 개선: 모임 상태 검증(verifyRecruiting), 중복 참여 검증(`verifyNotAlreadyParticipant`), 모임 정원 초과 검증(verifyMeetingCount) 로직이 강화되었습니다. * 개선 예정: 동시성 문제(Race Condition) 해결을 위한 비관적 락(Pessimistic Lock) 적용이 필요합니다. * `leaveMeeting(Long meetingId, Long memberId)`: * 목적: 회원이 모임에서 탈퇴합니다. * 주요 개선: 호스트 탈퇴 방지(verifyNotHost), 모임 상태에 따른 탈퇴 가능 여부 검증(verifyLeave) 로직이 추가되었습니다. * 개선 예정: MEETING_NOT_FOUND 대신 CANNOT_LEAVE_COMPLETED_MEETING과 같은 더 구체적인 에러 코드 적용이 필요합니다. * `createMeeting(Long memberId, CreateMeetingRequest request)`: * 목적: 새로운 모임을 생성합니다. 호스트를 참여자로 자동 등록하고 태그 정보를 저장합니다. * `updateMeeting(Long meetingId, Long memberId, UpdateMeetingRequest request)`: * 목적: 기존 모임의 정보를 수정합니다. 호스트만 수정할 수 있도록 검증합니다. * `deleteMeeting(Member member, Long meetingId)`: * 목적: 모임을 삭제합니다. * 개선 예정: 물리적 삭제 대신 논리적 삭제(Soft Delete) 방식 도입을 고려 중입니다. * feat : MeetingValidate.java,MemberValidate.java,ParticipantValidate,SpotValidate,TagValidate.java 검증로직을 추가하였습니다. * feat : MemberError.java , ParticipantRepository 기능을 추가하였습니다. --------- Co-authored-by: Hwang Seong Cheol a.k.a Hwuan Page <[email protected]> * fix : jellyfish 부분 * fix: activity 부분 * fix: member 부분 * fix: member 부분 * fix: spot 부분 * fix: forecast 부분 * fix: favorite 부분 * fix: alert 부분 * fix: meeting 부분 --------- Co-authored-by: Gunwoong cho <[email protected]> Co-authored-by: Hwang Seong Cheol a.k.a Hwuan Page <[email protected]> Co-authored-by: MyungJin <[email protected]> Co-authored-by: LEESUNBIN <[email protected]> Co-authored-by: gunwoong <[email protected]> * hotfix/fix-alert&favorites-62-HwuanPage * fix(hotfix/Meeting) : rebase로 인한 코드 누락 수정 (#65) * hotfix: 코드 누락 해결 (#67) * Fix/fix 70 gunwoong (#71) * hotfix: fix * hotfix: fix * hotfix: fix * fix: application-prod.yml에서 쿠키를 쓸지 말지 결정할 수 있게 수정 (#69) * fix: application-prod.yml에서 쿠키를 쓸지 말지 결정할 수 있게 수정 * test: 테스트 코드 작성 * fix: activities 시큐리티 엔드포인트 허용. redirecturi 수정 * Chore/docker set andvariable-68-hwuanPage * chore/ReadytoDeployv1.0.0-68-HuwanPage * chore/ReadytoDeploymentv1.0.0-68-HuwanPage * remove etc * prod * refactor: blacklist 엔티티의 jti에 인덱스를 건다. (#74) * Feat/meeting test 75 (#77) * feat : Meetingtest 를 위한 Util 파일입니니다. * feat : Meetingtest 를 위한 Util 파일입니니다. * feat : MeetingServiceImplTest 단위테스트입니다. * feat : MeetingControllerTest 통합테스트입니다. * feat : Build Lombok을 테스트를 위한 수정입니다. * feat : Tag 엔티티 Tag List<String> content 를 변환하기 위한 파일입니다. * feat : MeetingServiceImpl * feat : MeetingServiceImpl에서 수정하는 응답을 수정 , 매퍼를 수정하였습니다. * feat : Meeting에서 필요한 url을 열어뒀습니다. * space prob solve * stack-trace-DEBUG * hotfix/data.sql deprecate-HwuanPage (#79) * hotfix/data.sql deprecate-HwuanPage * portnum fix * Xtest * test X * workflow fix * add id * fix docker-compose-image-root * release/v1-marineleisure * fix: blacklist 엔티티의 jti에 인덱스를 건다. (#83) * fix: cors 프론트엔드 배포 도메인 추가 (#84) * fix: blacklist 엔티티의 jti에 인덱스를 건다. * fix: cors 프로트엔드 도메인 추가 * hotfix/method_allowed_patch-HwuanPage (#86) * Refactor/exception hwuan page (#87) * refacotr/favorite-Exception-update * fix kakao_redirect_uri * Feature/map service refactoring 76 gunwoong (#85) * feat: mapServiceRefactoring * refactoring: spot detail refactoring * refactoring: GeoUtils refactoring * test: repository test disable for prod * fix: apply flyway to yml * fix: disable test * refactor: khoa refactoring * fix: bug * fix: sql * fix: yml 환경변수 추가 * fix: detail field name 수정 * feature: 스케줄링 비동기 구현 (#91) * refactor: cacheable (#103) * Fix/meeting urland role (#100) * fix : MeetingServiceImpl getStatusMeetings -> getStatusMeeting_role : Guest 인지 host 인지 판단하는 로직을 추가 * fix : MeetingRepository getStatusMeetings -> getStatusMeeting_role : Guest 인지 host 인지 판단하는 로직을 추가 * fix : MeetingError MEETING_MEMBER_NOT_FOUND 에러를 추가하였습니다. 미팅에서 맴버를 확인할 수 없는 에러입니다. * fix : MeetingController MeetingController 에서 role 을 확인하여 추가 확인할 수 있도록 하였습니다. * fix : MeetingServiceImplTest, MeetingControllerTest.java URL 개선에 의한 새로운 테스트 입니다. * fix : MeetingServiceImplTest, MeetingControllerTest.java @disabled 추가 하였습니다. * feat: 회원 탈퇴 시 카카오 연결 끊기도 수행하게 구현한다. (#98) * feat: member 삭제 시 kakao 연결 끊기 로직도 수행하게 구현 * test: 변경 사항 test * feat : Meeting의 커서방식에서 매핑을 하였습니다. (#94) * feat: 카카오 로그인 과정에서 pkce를 통해 보안 관점에서 개선 (#106) * feat: 보안 인증 과정에서 PKCE 추가하여 구현 * test: 변경 사항 test 추가 * feat: PKCE 기반 보안 기능 코드 구조 변경 * test: PKCE 기반 보안 기능 test * refactor: PKCE 생성을 클라이언트 에게 넘긴다. * test: pkce test 플로우 변경에 따라 변경 * fix: member entity의 nickname 중복을 허용한다 * fix: 테스트를 위해 SchedulerService.java 의 @RequiredArgsConstrucor 지운 부분 복구 * fix: 테스트를 위해 SchedulerService.java 의 @RequiredArgsConstrucor 지운 부분 복구 * refactor: open-meteo 서비스 관련 리팩토링 (#95) * refactor: RichDomain으로 변경 내역입니다. (#105) * Fix: login redirect (#107) * fix: 로그인 요청때의 리다이렉트 uri를 토큰 교환시에도 사용 * test: test * fix: fallback 상황에서 리다이렉트 uri 찾는 로직 추가 * Refactor/meeting rich domain (#110) * refactor: RichDomain으로 변경 내역입니다. * refactor: 누락 프로젝트 파일이 있어 첨부합니다. * build: caffenine 적용 * relase (#111) (#112) * git initialize * feature/swagger-03-gunwoong (#5) * feat: 공통 도메인 구현 * feat: 메인 어플리케이션에 추가 * feat: swagger 추가 * feat: swagger 추가 * feature/base domain 04 gunwoong (#6) * feat: 공통 도메인 구현 * feat: 메인 어플리케이션에 추가 * feature/OpenAPI Test/02-HwuanPage * feature/OpenAPI Test/02-HwuanPage * Update SurfingForecastApiClient.java * feature/APICallTest-02-HwuanPage * feature/EntityInit-13-HwuanPage * feature/EntityInit-13-HwuanPage * feature/JellyfishEntityInit-13-HwuanPage * Update FishingType.java * feature/EntityInitialize-13-HwuanPage * feat: entity, repositor 구현 * feat: 예상 dto 구현 * chore: 의존성 추가 * feat: 로그인 구현 & 이후 토큰 발급 로직 구현 * fix: AuthCotnroller 수정 * fix: 클라이언트에서 카카오에서 코드를 받아 서버로 post 하게 수정 * feat: 토큰 검증 * feat: refresh token 블랙리스트 처리 로직 구현 * feat: refresh 토큰 블랙리스트 처리 & 재발급 로직 구현 * feat: SecurityFilterChain 엔드 포인트 허용 * feat: refresh 토큰 블랙리스트 검증 로직 구현 * feat: redis에서 refreshToken 블랙리스트 검증 * refactor: controller에 강하게 결합 되어 있던 로직들 분리 * test: member 관련 테스트 * chore: 하드코딩한 중요 값 Intellij IDEA 환경변수로 설정 * refactor: state 관리를 위해 세션 추가 * feat: member 정보 조회하는 서비스 로직 구현 * feat: member 정보 조회하는 서비스 로직 구현 * format: naver formatter로 포매팅 * chore: application-dev * fix: customException 처리 * Feat/meeting interface (#19) * feat : MeetingService 인터페이스 구현 * feat : ParticipantResponse * feat : MeetingListResponse 구현 * feat : MeetingDetailResponse구현 * feat : MeetingDetailAndMemberResponse 구현 * feat : ListSpot 구현 * feat : DetailSpot 구현 * feat : CreateMeetingRequest 구현 * feat : Tag 구현 * feat : Long -> long 변경 서비스와 Entity내에서 null값이 절대 나오지 않는다고 판단하는 값을 long으로 변경하였습니다. * feat : MeetingService.java -> 무한페이지로딩형식으로 바꾸었습니다. * Update src/main/java/sevenstar/marineleisure/meeting/Dto/Response/MeetingDetailResponse.java --------- * Feature/FavoritesAndAlertInterface-16-HwuanPage * feature/FavoritesAndAlertInterface-16-HwuanPage * Update AlertMapper.java * Update JellyfishRegionDensityRepository.java * Update AlertController.java * Update FavoriteController.java * Update FavoriteRepository.java * Update AlertController.java * Update JellyfishSpieces.java * Update JellyfishRegion.java * Update JellyfishRegion.java * feature/CustomExceptionInit-22-HwuanPage * feature/CustomExceptionInit-22-HwuanPage * Errorcode interface Change * Refactor application.yml 환경변수 설정 (#25) * refactor: application.yml 환경변수 설정 * Rename: 오타 수정 * Feature/spot service interface 29 gunwoong (#30) * feat: api * feat: api 스케줄링 * feat: spot service inteface * Feature/api scheduler 15 gunwoong (#28) * feat: api * feat: api 스케줄링 * feat: spot service inteface * test: remove legacy test * feat: apply open meteo * test: apply api test * feat: spot service (#34) * feat: spot service * feat: spot service 고도화 및 조회도 관련 서비스 추가 * feat: 조회도 관련 서비스 추가 * feat: 조회도 관련 서비스 추가 * feat: 조회도 관련 서비스 추가 * hotfix: duplicated controller method * feature/FavoriteCRUD-33-HwuanPage * DELETE COMPLETE * UPDATE COMPLETE * search COMPLETE * Before gunwoong * FavoriteCRUD create * feat/domain test * FavoriteSpotServiceTest * FavoriteSpotServiceTest * feature/FavoriteCURD-33-HwuanPage * add some description on service * Update FavoriteServiceImpl.java * Feature/spot preview 40 gunwoong (#41) * feat: spot preview & 리팩토링 * feat: spot preview & 리팩토링 * hotfix: jpa metamodel fix * fix: error fix * fix: 소셜 로그인 재시도 시 닉네임 UNIQUE 제약 위반 오류 발생 (#42) * fix: 로그아웃 후 재로그인 시 동일 정보로 db에 insert 하던 버그 수정 * refactor: 로그아웃 후 재로그인 시 동일 정보로 db에 insert 수정 사항 리팩토링 * test: 변경사항에 따른 테스트 코드 수정 * hofix: bug fix * Feature/Alert-22-HwuanPage * Create Pdf Parser * Web crawler run perpectly,but pdfparser do not work well * PDF parse to stack DB complete with OPENAI * CallAlert Complete * JellyFish PDF parsing work well * feature/ControllerTest Complete * feature/JellyfishAlert-26-HwuanPage * feat: 즐겨찾기 추가 및 리팩토링 (#49) * feat: 즐겨찾기 추가 및 리팩토링 * refactor: 리팩토링 * feat: 카카오 로그인을 stateless 하게 변경한다 (#51) * refactor: 기존 state 사용 방식 -> stateless 방식으로 변경 * refactor: 기존 state 사용 방식 -> stateless 방식으로 변경으로 인해 필요 없는 엔드 포인트 삭제 * test: 변경사항 test 수정 * feat: 카카오 측에서 인증 실패시에 반환 하는 에러 처리하는 코드 구현 * test: 카카오 측에서 인증 실패시에 반환 하는 에러 처리하는 테스트 추가 * fix: 주석 제거 * fix: exception 변경 * Feat/meeting service (#46) * WIP: Rebase를 위한 임시 저장 * feat : Meeting.java -> Meeting 엔터티 @builder 를 상위 어노테이션에 일단 추가시켰습니다. * feat : Meeting.java -> Meeting 엔터티 @builder 를 상위 어노테이션에 일단 추가시켰습니다. * feat : Meeting.java -> Meeting 엔터티 @builder 를 상위 어노테이션에 일단 추가시켰습니다. * Delete MeetingServiceImplReview.md * Delete MeetingServiceUserFlow.md * feat : 패키지명 변경 이슈 -> 패키지 명을 컨벤션에 따른 이름으로 변경했습니다. * feat : MeetingController.java long participantCount = participantRepository.countMeetingIdMember -> long participantCount = participantRepository.countMeetingId로 수정하였습니다. * feat : MeetingError.java MeetingError.java 를 추가하였습니다. * feat : MeetingMapper MeetingServiceImpl에서 사용중이었던 Mapper를 분리하였습니다. * feat : MeetingService.java 패키지 명 수정으로 인해서 수정사항이 있었습니다. * feat : MeetingServiceImpl.java 트랜잭션 관리 명확화 하였습니다. validate 패키지를 개선하였습니다. joinMeeting 중복 참여 제한 로직을 강화하였습니다. * `getAllMeetings(Long cursorId, int size)`: * 목적: 모든 모임 목록을 페이징 처리하여 조회합니다. cursorId를 사용하여 무한 스크롤과 같은 커서 기반 페이징을 지원합니다. * 특징: @transactional(readOnly = true)를 통해 읽기 전용 트랜잭션으로 최적화되었습니다. * `getMeetingDetails(Long meetingId)`: * 목적: 특정 모임의 상세 정보를 조회합니다. 호스트, 장소, 태그 등 연관된 정보를 함께 가져옵니다. * 개선 예정: 현재 N+1 문제가 발생할 수 있어, 향후 Fetch Join을 통한 성능 최적화가 필요합니다. * `getStatusMyMeetings(Long memberId, Long cursorId, int size, MeetingStatus meetingStatus)`: * 목적: 특정 회원의 상태별(예: 모집 중, 완료) 모임 목록을 페이징 처리하여 조회합니다. * `getMeetingDetailAndMember(Long memberId, Long meetingId)`: * 목적: 호스트가 자신의 모임 상세 정보와 참여자 목록을 조회합니다. 참여자들의 닉네임을 함께 제공합니다. * `countMeetings(Long memberId)`: * 목적: 특정 회원이 참여한 모임의 총 개수를 반환합니다. * `joinMeeting(Long meetingId, Long memberId)`: * 목적: 회원이 특정 모임에 참여합니다. * 주요 개선: 모임 상태 검증(verifyRecruiting), 중복 참여 검증(`verifyNotAlreadyParticipant`), 모임 정원 초과 검증(verifyMeetingCount) 로직이 강화되었습니다. * 개선 예정: 동시성 문제(Race Condition) 해결을 위한 비관적 락(Pessimistic Lock) 적용이 필요합니다. * `leaveMeeting(Long meetingId, Long memberId)`: * 목적: 회원이 모임에서 탈퇴합니다. * 주요 개선: 호스트 탈퇴 방지(verifyNotHost), 모임 상태에 따른 탈퇴 가능 여부 검증(verifyLeave) 로직이 추가되었습니다. * 개선 예정: MEETING_NOT_FOUND 대신 CANNOT_LEAVE_COMPLETED_MEETING과 같은 더 구체적인 에러 코드 적용이 필요합니다. * `createMeeting(Long memberId, CreateMeetingRequest request)`: * 목적: 새로운 모임을 생성합니다. 호스트를 참여자로 자동 등록하고 태그 정보를 저장합니다. * `updateMeeting(Long meetingId, Long memberId, UpdateMeetingRequest request)`: * 목적: 기존 모임의 정보를 수정합니다. 호스트만 수정할 수 있도록 검증합니다. * `deleteMeeting(Member member, Long meetingId)`: * 목적: 모임을 삭제합니다. * 개선 예정: 물리적 삭제 대신 논리적 삭제(Soft Delete) 방식 도입을 고려 중입니다. * feat : MeetingValidate.java,MemberValidate.java,ParticipantValidate,SpotValidate,TagValidate.java 검증로직을 추가하였습니다. * feat : MemberError.java , ParticipantRepository 기능을 추가하였습니다. --------- * Feature/integration init (#54) * feature/IntegrationSet(test&Build)-52-HwuanPage * data.sql unique update * image build needs * ignore dev.yml * remove dev.yml tracking and ignore it * prod * proded * Feature/activities 17 audwls239 (#56) * feature: 컨트롤러, 서비스 생성 * feature: 활동별 지수 조회(위치 기반) * feature: DTO 추가 * feature: 활동별 지수 조회(글로벌) 추가, 컨트롤러 수정 * feature: 활동별 지수 상세 조회(미완성) * feature: 해양 정보 조회 * feature: 활동 상세 조회 --------- * feat : ParticipantError 입니다. * hotfix: error fix * fix : Directory 수정사항입니다. (#57) * hotfix: error fix * feat: member delete (#58) * fix: 멤버 삭제 구현 * feat: 멤버 삭제, 위/경도 수정 구현 * test: 테스트 수정 * Delete src/main/java/sevenstar/marineleisure/meeting/repository/MemberRepository.java * Delete src/main/java/sevenstar/marineleisure/meeting/repository/OutdoorSpotSpotRepository.java * Delete src/main/resources/test.http --------- * fix : ParticipantRepository (#59) existsByMeetingIdAndUserId 로 수정하였습니다. * fix : ParticipantRepository (#60) memberId -> userId로 수정하였습니다. * fix: token (#61) * feature/base domain 04 gunwoong (#6) * feat: 공통 도메인 구현 * feat: 메인 어플리케이션에 추가 * feature/CustomExceptionInit-22-HwuanPage * feature/CustomExceptionInit-22-HwuanPage * Errorcode interface Change * Refactor application.yml 환경변수 설정 (#25) * refactor: application.yml 환경변수 설정 * Rename: 오타 수정 * Feature/spot service interface 29 gunwoong (#30) * feat: api * feat: api 스케줄링 * feat: spot service inteface * feat: 카카오 로그인을 stateless 하게 변경한다 (#51) * refactor: 기존 state 사용 방식 -> stateless 방식으로 변경 * refactor: 기존 state 사용 방식 -> stateless 방식으로 변경으로 인해 필요 없는 엔드 포인트 삭제 * test: 변경사항 test 수정 * feat: 카카오 측에서 인증 실패시에 반환 하는 에러 처리하는 코드 구현 * test: 카카오 측에서 인증 실패시에 반환 하는 에러 처리하는 테스트 추가 * fix: 주석 제거 * fix: exception 변경 * Feat/meeting service (#46) * WIP: Rebase를 위한 임시 저장 # Conflicts: # src/main/java/sevenstar/marineleisure/global/exception/enums/CommonErrorCode.java # src/main/java/sevenstar/marineleisure/global/swagger/SwaggerController.java * feat : Meeting.java -> Meeting 엔터티 @builder 를 상위 어노테이션에 일단 추가시켰습니다. * feat : Meeting.java -> Meeting 엔터티 @builder 를 상위 어노테이션에 일단 추가시켰습니다. * feat : Meeting.java -> Meeting 엔터티 @builder 를 상위 어노테이션에 일단 추가시켰습니다. * Delete MeetingServiceImplReview.md * Delete MeetingServiceUserFlow.md * feat : 패키지명 변경 이슈 -> 패키지 명을 컨벤션에 따른 이름으로 변경했습니다. * feat : MeetingController.java long participantCount = participantRepository.countMeetingIdMember -> long participantCount = participantRepository.countMeetingId로 수정하였습니다. * feat : MeetingError.java MeetingError.java 를 추가하였습니다. * feat : MeetingMapper MeetingServiceImpl에서 사용중이었던 Mapper를 분리하였습니다. * feat : MeetingService.java 패키지 명 수정으로 인해서 수정사항이 있었습니다. * feat : MeetingServiceImpl.java 트랜잭션 관리 명확화 하였습니다. validate 패키지를 개선하였습니다. joinMeeting 중복 참여 제한 로직을 강화하였습니다. * `getAllMeetings(Long cursorId, int size)`: * 목적: 모든 모임 목록을 페이징 처리하여 조회합니다. cursorId를 사용하여 무한 스크롤과 같은 커서 기반 페이징을 지원합니다. * 특징: @transactional(readOnly = true)를 통해 읽기 전용 트랜잭션으로 최적화되었습니다. * `getMeetingDetails(Long meetingId)`: * 목적: 특정 모임의 상세 정보를 조회합니다. 호스트, 장소, 태그 등 연관된 정보를 함께 가져옵니다. * 개선 예정: 현재 N+1 문제가 발생할 수 있어, 향후 Fetch Join을 통한 성능 최적화가 필요합니다. * `getStatusMyMeetings(Long memberId, Long cursorId, int size, MeetingStatus meetingStatus)`: * 목적: 특정 회원의 상태별(예: 모집 중, 완료) 모임 목록을 페이징 처리하여 조회합니다. * `getMeetingDetailAndMember(Long memberId, Long meetingId)`: * 목적: 호스트가 자신의 모임 상세 정보와 참여자 목록을 조회합니다. 참여자들의 닉네임을 함께 제공합니다. * `countMeetings(Long memberId)`: * 목적: 특정 회원이 참여한 모임의 총 개수를 반환합니다. * `joinMeeting(Long meetingId, Long memberId)`: * 목적: 회원이 특정 모임에 참여합니다. * 주요 개선: 모임 상태 검증(verifyRecruiting), 중복 참여 검증(`verifyNotAlreadyParticipant`), 모임 정원 초과 검증(verifyMeetingCount) 로직이 강화되었습니다. * 개선 예정: 동시성 문제(Race Condition) 해결을 위한 비관적 락(Pessimistic Lock) 적용이 필요합니다. * `leaveMeeting(Long meetingId, Long memberId)`: * 목적: 회원이 모임에서 탈퇴합니다. * 주요 개선: 호스트 탈퇴 방지(verifyNotHost), 모임 상태에 따른 탈퇴 가능 여부 검증(verifyLeave) 로직이 추가되었습니다. * 개선 예정: MEETING_NOT_FOUND 대신 CANNOT_LEAVE_COMPLETED_MEETING과 같은 더 구체적인 에러 코드 적용이 필요합니다. * `createMeeting(Long memberId, CreateMeetingRequest request)`: * 목적: 새로운 모임을 생성합니다. 호스트를 참여자로 자동 등록하고 태그 정보를 저장합니다. * `updateMeeting(Long meetingId, Long memberId, UpdateMeetingRequest request)`: * 목적: 기존 모임의 정보를 수정합니다. 호스트만 수정할 수 있도록 검증합니다. * `deleteMeeting(Member member, Long meetingId)`: * 목적: 모임을 삭제합니다. * 개선 예정: 물리적 삭제 대신 논리적 삭제(Soft Delete) 방식 도입을 고려 중입니다. * feat : MeetingValidate.java,MemberValidate.java,ParticipantValidate,SpotValidate,TagValidate.java 검증로직을 추가하였습니다. * feat : MemberError.java , ParticipantRepository 기능을 추가하였습니다. --------- * feat: 테스트용 액세스 토큰 생성 * feature/base domain 04 gunwoong (#6) * feat: 공통 도메인 구현 * feat: 메인 어플리케이션에 추가 * Refactor application.yml 환경변수 설정 (#25) * refactor: application.yml 환경변수 설정 * Rename: 오타 수정 * Feature/spot service interface 29 gunwoong (#30) * feat: api * feat: api 스케줄링 * feat: spot service inteface * feature/base domain 04 gunwoong (#6) * feat: 공통 도메인 구현 * feat: 메인 어플리케이션에 추가 * feature/CustomExceptionInit-22-HwuanPage * feature/CustomExceptionInit-22-HwuanPage * Errorcode interface Change * Feature/spot service interface 29 gunwoong (#30) * feat: api * feat: api 스케줄링 * feat: spot service inteface * Feature/api scheduler 15 gunwoong (#28) * feat: api * feat: api 스케줄링 * feat: spot service inteface * test: remove legacy test * feat: apply open meteo * test: apply api test * feature/FavoriteCRUD-33-HwuanPage * DELETE COMPLETE * UPDATE COMPLETE * search COMPLETE * Before gunwoong * FavoriteCRUD create * feat/domain test * FavoriteSpotServiceTest * FavoriteSpotServiceTest * feature/FavoriteCURD-33-HwuanPage * add some description on service * Update FavoriteServiceImpl.java * fix: error fix * fix: 소셜 로그인 재시도 시 닉네임 UNIQUE 제약 위반 오류 발생 (#42) * fix: 로그아웃 후 재로그인 시 동일 정보로 db에 insert 하던 버그 수정 * refactor: 로그아웃 후 재로그인 시 동일 정보로 db에 insert 수정 사항 리팩토링 * test: 변경사항에 따른 테스트 코드 수정 * hofix: bug fix * feat: 카카오 로그인을 stateless 하게 변경한다 (#51) * refactor: 기존 state 사용 방식 -> stateless 방식으로 변경 * refactor: 기존 state 사용 방식 -> stateless 방식으로 변경으로 인해 필요 없는 엔드 포인트 삭제 * test: 변경사항 test 수정 * feat: 카카오 측에서 인증 실패시에 반환 하는 에러 처리하는 코드 구현 * test: 카카오 측에서 인증 실패시에 반환 하는 에러 처리하는 테스트 추가 * fix: 주석 제거 * fix: exception 변경 * Feat/meeting service (#46) * WIP: Rebase를 위한 임시 저장 # Conflicts: # src/main/java/sevenstar/marineleisure/global/exception/enums/CommonErrorCode.java # src/main/java/sevenstar/marineleisure/global/swagger/SwaggerController.java * feat : Meeting.java -> Meeting 엔터티 @builder 를 상위 어노테이션에 일단 추가시켰습니다. * feat : Meeting.java -> Meeting 엔터티 @builder 를 상위 어노테이션에 일단 추가시켰습니다. * feat : Meeting.java -> Meeting 엔터티 @builder 를 상위 어노테이션에 일단 추가시켰습니다. * Delete MeetingServiceImplReview.md * Delete MeetingServiceUserFlow.md * feat : 패키지명 변경 이슈 -> 패키지 명을 컨벤션에 따른 이름으로 변경했습니다. * feat : MeetingController.java long participantCount = participantRepository.countMeetingIdMember -> long participantCount = participantRepository.countMeetingId로 수정하였습니다. * feat : MeetingError.java MeetingError.java 를 추가하였습니다. * feat : MeetingMapper MeetingServiceImpl에서 사용중이었던 Mapper를 분리하였습니다. * feat : MeetingService.java 패키지 명 수정으로 인해서 수정사항이 있었습니다. * feat : MeetingServiceImpl.java 트랜잭션 관리 명확화 하였습니다. validate 패키지를 개선하였습니다. joinMeeting 중복 참여 제한 로직을 강화하였습니다. * `getAllMeetings(Long cursorId, int size)`: * 목적: 모든 모임 목록을 페이징 처리하여 조회합니다. cursorId를 사용하여 무한 스크롤과 같은 커서 기반 페이징을 지원합니다. * 특징: @transactional(readOnly = true)를 통해 읽기 전용 트랜잭션으로 최적화되었습니다. * `getMeetingDetails(Long meetingId)`: * 목적: 특정 모임의 상세 정보를 조회합니다. 호스트, 장소, 태그 등 연관된 정보를 함께 가져옵니다. * 개선 예정: 현재 N+1 문제가 발생할 수 있어, 향후 Fetch Join을 통한 성능 최적화가 필요합니다. * `getStatusMyMeetings(Long memberId, Long cursorId, int size, MeetingStatus meetingStatus)`: * 목적: 특정 회원의 상태별(예: 모집 중, 완료) 모임 목록을 페이징 처리하여 조회합니다. * `getMeetingDetailAndMember(Long memberId, Long meetingId)`: * 목적: 호스트가 자신의 모임 상세 정보와 참여자 목록을 조회합니다. 참여자들의 닉네임을 함께 제공합니다. * `countMeetings(Long memberId)`: * 목적: 특정 회원이 참여한 모임의 총 개수를 반환합니다. * `joinMeeting(Long meetingId, Long memberId)`: * 목적: 회원이 특정 모임에 참여합니다. * 주요 개선: 모임 상태 검증(verifyRecruiting), 중복 참여 검증(`verifyNotAlreadyParticipant`), 모임 정원 초과 검증(verifyMeetingCount) 로직이 강화되었습니다. * 개선 예정: 동시성 문제(Race Condition) 해결을 위한 비관적 락(Pessimistic Lock) 적용이 필요합니다. * `leaveMeeting(Long meetingId, Long memberId)`: * 목적: 회원이 모임에서 탈퇴합니다. * 주요 개선: 호스트 탈퇴 방지(verifyNotHost), 모임 상태에 따른 탈퇴 가능 여부 검증(verifyLeave) 로직이 추가되었습니다. * 개선 예정: MEETING_NOT_FOUND 대신 CANNOT_LEAVE_COMPLETED_MEETING과 같은 더 구체적인 에러 코드 적용이 필요합니다. * `createMeeting(Long memberId, CreateMeetingRequest request)`: * 목적: 새로운 모임을 생성합니다. 호스트를 참여자로 자동 등록하고 태그 정보를 저장합니다. * `updateMeeting(Long meetingId, Long memberId, UpdateMeetingRequest request)`: * 목적: 기존 모임의 정보를 수정합니다. 호스트만 수정할 수 있도록 검증합니다. * `deleteMeeting(Member member, Long meetingId)`: * 목적: 모임을 삭제합니다. * 개선 예정: 물리적 삭제 대신 논리적 삭제(Soft Delete) 방식 도입을 고려 중입니다. * feat : MeetingValidate.java,MemberValidate.java,ParticipantValidate,SpotValidate,TagValidate.java 검증로직을 추가하였습니다. * feat : MemberError.java , ParticipantRepository 기능을 추가하였습니다. --------- * fix : jellyfish 부분 * fix: activity 부분 * fix: member 부분 * fix: member 부분 * fix: spot 부분 * fix: forecast 부분 * fix: favorite 부분 * fix: alert 부분 * fix: meeting 부분 --------- * hotfix/fix-alert&favorites-62-HwuanPage * fix(hotfix/Meeting) : rebase로 인한 코드 누락 수정 (#65) * hotfix: 코드 누락 해결 (#67) * Fix/fix 70 gunwoong (#71) * hotfix: fix * hotfix: fix * hotfix: fix * fix: application-prod.yml에서 쿠키를 쓸지 말지 결정할 수 있게 수정 (#69) * fix: application-prod.yml에서 쿠키를 쓸지 말지 결정할 수 있게 수정 * test: 테스트 코드 작성 * fix: activities 시큐리티 엔드포인트 허용. redirecturi 수정 * Chore/docker set andvariable-68-hwuanPage * chore/ReadytoDeployv1.0.0-68-HuwanPage * chore/ReadytoDeploymentv1.0.0-68-HuwanPage * remove etc * prod * refactor: blacklist 엔티티의 jti에 인덱스를 건다. (#74) * Feat/meeting test 75 (#77) * feat : Meetingtest 를 위한 Util 파일입니니다. * feat : Meetingtest 를 위한 Util 파일입니니다. * feat : MeetingServiceImplTest 단위테스트입니다. * feat : MeetingControllerTest 통합테스트입니다. * feat : Build Lombok을 테스트를 위한 수정입니다. * feat : Tag 엔티티 Tag List<String> content 를 변환하기 위한 파일입니다. * feat : MeetingServiceImpl * feat : MeetingServiceImpl에서 수정하는 응답을 수정 , 매퍼를 수정하였습니다. * feat : Meeting에서 필요한 url을 열어뒀습니다. * space prob solve * stack-trace-DEBUG * hotfix/data.sql deprecate-HwuanPage (#79) * hotfix/data.sql deprecate-HwuanPage * portnum fix * Xtest * test X * workflow fix * add id * fix docker-compose-image-root * release/v1-marineleisure * fix: blacklist 엔티티의 jti에 인덱스를 건다. (#83) * fix: cors 프론트엔드 배포 도메인 추가 (#84) * fix: blacklist 엔티티의 jti에 인덱스를 건다. * fix: cors 프로트엔드 도메인 추가 * hotfix/method_allowed_patch-HwuanPage (#86) * Refactor/exception hwuan page (#87) * refacotr/favorite-Exception-update * fix kakao_redirect_uri * Feature/map service refactoring 76 gunwoong (#85) * feat: mapServiceRefactoring * refactoring: spot detail refactoring * refactoring: GeoUtils refactoring * test: repository test disable for prod * fix: apply flyway to yml * fix: disable test * refactor: khoa refactoring * fix: bug * fix: sql * fix: yml 환경변수 추가 * fix: detail field name 수정 * feature: 스케줄링 비동기 구현 (#91) * refactor: cacheable (#103) * Fix/meeting urland role (#100) * fix : MeetingServiceImpl getStatusMeetings -> getStatusMeeting_role : Guest 인지 host 인지 판단하는 로직을 추가 * fix : MeetingRepository getStatusMeetings -> getStatusMeeting_role : Guest 인지 host 인지 판단하는 로직을 추가 * fix : MeetingError MEETING_MEMBER_NOT_FOUND 에러를 추가하였습니다. 미팅에서 맴버를 확인할 수 없는 에러입니다. * fix : MeetingController MeetingController 에서 role 을 확인하여 추가 확인할 수 있도록 하였습니다. * fix : MeetingServiceImplTest, MeetingControllerTest.java URL 개선에 의한 새로운 테스트 입니다. * fix : MeetingServiceImplTest, MeetingControllerTest.java @disabled 추가 하였습니다. * feat: 회원 탈퇴 시 카카오 연결 끊기도 수행하게 구현한다. (#98) * feat: member 삭제 시 kakao 연결 끊기 로직도 수행하게 구현 * test: 변경 사항 test * feat : Meeting의 커서방식에서 매핑을 하였습니다. (#94) * feat: 카카오 로그인 과정에서 pkce를 통해 보안 관점에서 개선 (#106) * feat: 보안 인증 과정에서 PKCE 추가하여 구현 * test: 변경 사항 test 추가 * feat: PKCE 기반 보안 기능 코드 구조 변경 * test: PKCE 기반 보안 기능 test * refactor: PKCE 생성을 클라이언트 에게 넘긴다. * test: pkce test 플로우 변경에 따라 변경 * fix: member entity의 nickname 중복을 허용한다 * fix: 테스트를 위해 SchedulerService.java 의 @RequiredArgsConstrucor 지운 부분 복구 * fix: 테스트를 위해 SchedulerService.java 의 @RequiredArgsConstrucor 지운 부분 복구 * refactor: open-meteo 서비스 관련 리팩토링 (#95) * refactor: RichDomain으로 변경 내역입니다. (#105) * Fix: login redirect (#107) * fix: 로그인 요청때의 리다이렉트 uri를 토큰 교환시에도 사용 * test: test * fix: fallback 상황에서 리다이렉트 uri 찾는 로직 추가 * Refactor/meeting rich domain (#110) * refactor: RichDomain으로 변경 내역입니다. * refactor: 누락 프로젝트 파일이 있어 첨부합니다. * build: caffenine 적용 --------- Co-authored-by: HwuanPage <[email protected]> Co-authored-by: JaeoneHeo <[email protected]> Co-authored-by: LEESUNBIN <[email protected]> Co-authored-by: Copilot <[email protected]> Co-authored-by: MyungJin <[email protected]> Co-authored-by: iseonbin <[email protected]> * fix: fallback 상황에서 리다이렉트 uri 찾는 로직 추가 (#113) * release (#114) (#115) * git initialize * feature/swagger-03-gunwoong (#5) * feat: 공통 도메인 구현 * feat: 메인 어플리케이션에 추가 * feat: swagger 추가 * feat: swagger 추가 * feature/base domain 04 gunwoong (#6) * feat: 공통 도메인 구현 * feat: 메인 어플리케이션에 추가 * feature/OpenAPI Test/02-HwuanPage * feature/OpenAPI Test/02-HwuanPage * Update SurfingForecastApiClient.java * feature/APICallTest-02-HwuanPage * feature/EntityInit-13-HwuanPage * feature/EntityInit-13-HwuanPage * feature/JellyfishEntityInit-13-HwuanPage * Update FishingType.java * feature/EntityInitialize-13-HwuanPage * feat: entity, repositor 구현 * feat: 예상 dto 구현 * chore: 의존성 추가 * feat: 로그인 구현 & 이후 토큰 발급 로직 구현 * fix: AuthCotnroller 수정 * fix: 클라이언트에서 카카오에서 코드를 받아 서버로 post 하게 수정 * feat: 토큰 검증 * feat: refresh token 블랙리스트 처리 로직 구현 * feat: refresh 토큰 블랙리스트 처리 & 재발급 로직 구현 * feat: SecurityFilterChain 엔드 포인트 허용 * feat: refresh 토큰 블랙리스트 검증 로직 구현 * feat: redis에서 refreshToken 블랙리스트 검증 * refactor: controller에 강하게 결합 되어 있던 로직들 분리 * test: member 관련 테스트 * chore: 하드코딩한 중요 값 Intellij IDEA 환경변수로 설정 * refactor: state 관리를 위해 세션 추가 * feat: member 정보 조회하는 서비스 로직 구현 * feat: member 정보 조회하는 서비스 로직 구현 * format: naver formatter로 포매팅 * chore: application-dev * fix: customException 처리 * Feat/meeting interface (#19) * feat : MeetingService 인터페이스 구현 * feat : ParticipantResponse * feat : MeetingListResponse 구현 * feat : MeetingDetailResponse구현 * feat : MeetingDetailAndMemberResponse 구현 * feat : ListSpot 구현 * feat : DetailSpot 구현 * feat : CreateMeetingRequest 구현 * feat : Tag 구현 * feat : Long -> long 변경 서비스와 Entity내에서 null값이 절대 나오지 않는다고 판단하는 값을 long으로 변경하였습니다. * feat : MeetingService.java -> 무한페이지로딩형식으로 바꾸었습니다. * Update src/main/java/sevenstar/marineleisure/meeting/Dto/Response/MeetingDetailResponse.java --------- * Feature/FavoritesAndAlertInterface-16-HwuanPage * feature/FavoritesAndAlertInterface-16-HwuanPage * Update AlertMapper.java * Update JellyfishRegionDensityRepository.java * Update AlertController.java * Update FavoriteController.java * Update FavoriteRepository.java * Update AlertController.java * Update JellyfishSpieces.java * Update JellyfishRegion.java * Update JellyfishRegion.java * feature/CustomExceptionInit-22-HwuanPage * feature/CustomExceptionInit-22-HwuanPage * Errorcode interface Change * Refactor application.yml 환경변수 설정 (#25) * refactor: application.yml 환경변수 설정 * Rename: 오타 수정 * Feature/spot service interface 29 gunwoong (#30) * feat: api * feat: api 스케줄링 * feat: spot service inteface * Feature/api scheduler 15 gunwoong (#28) * feat: api * feat: api 스케줄링 * feat: spot service inteface * test: remove legacy test * feat: apply open meteo * test: apply api test * feat: spot service (#34) * feat: spot service * feat: spot service 고도화 및 조회도 관련 서비스 추가 * feat: 조회도 관련 서비스 추가 * feat: 조회도 관련 서비스 추가 * feat: 조회도 관련 서비스 추가 * hotfix: duplicated controller method * feature/FavoriteCRUD-33-HwuanPage * DELETE COMPLETE * UPDATE COMPLETE * search COMPLETE * Before gunwoong * FavoriteCRUD create * feat/domain test * FavoriteSpotServiceTest * FavoriteSpotServiceTest * feature/FavoriteCURD-33-HwuanPage * add some description on service * Update FavoriteServiceImpl.java * Feature/spot preview 40 gunwoong (#41) * feat: spot preview & 리팩토링 * feat: spot preview & 리팩토링 * hotfix: jpa metamodel fix * fix: error fix * fix: 소셜 로그인 재시도 시 닉네임 UNIQUE 제약 위반 오류 발생 (#42) * fix: 로그아웃 후 재로그인 시 동일 정보로 db에 insert 하던 버그 수정 * refactor: 로그아웃 후 재로그인 시 동일 정보로 db에 insert 수정 사항 리팩토링 * test: 변경사항에 따른 테스트 코드 수정 * hofix: bug fix * Feature/Alert-22-HwuanPage * Create Pdf Parser * Web crawler run perpectly,but pdfparser do not work well * PDF parse to stack DB complete with OPENAI * CallAlert Complete * JellyFish PDF parsing work well * feature/ControllerTest Complete * feature/JellyfishAlert-26-HwuanPage * feat: 즐겨찾기 추가 및 리팩토링 (#49) * feat: 즐겨찾기 추가 및 리팩토링 * refactor: 리팩토링 * feat: 카카오 로그인을 stateless 하게 변경한다 (#51) * refactor: 기존 state 사용 방식 -> stateless 방식으로 변경 * refactor: 기존 state 사용 방식 -> stateless 방식으로 변경으로 인해 필요 없는 엔드 포인트 삭제 * test: 변경사항 test 수정 * feat: 카카오 측에서 인증 실패시에 반환 하는 에러 처리하는 코드 구현 * test: 카카오 측에서 인증 실패시에 반환 하는 에러 처리하는 테스트 추가 * fix: 주석 제거 * fix: exception 변경 * Feat/meeting service (#46) * WIP: Rebase를 위한 임시 저장 * feat : Meeting.java -> Meeting 엔터티 @builder 를 상위 어노테이션에 일단 추가시켰습니다. * feat : Meeting.java -> Meeting 엔터티 @builder 를 상위 어노테이션에 일단 추가시켰습니다. * feat : Meeting.java -> Meeting 엔터티 @builder 를 상위 어노테이션에 일단 추가시켰습니다. * Delete MeetingServiceImplReview.md * Delete MeetingServiceUserFlow.md * feat : 패키지명 변경 이슈 -> 패키지 명을 컨벤션에 따른 이름으로 변경했습니다. * feat : MeetingController.java long participantCount = participantRepository.countMeetingIdMember -> long participantCount = participantRepository.countMeetingId로 수정하였습니다. * feat : MeetingError.java MeetingError.java 를 추가하였습니다. * feat : MeetingMapper MeetingServiceImpl에서 사용중이었던 Mapper를 분리하였습니다. * feat : MeetingService.java 패키지 명 수정으로 인해서 수정사항이 있었습니다. * feat : MeetingServiceImpl.java 트랜잭션 관리 명확화 하였습니다. validate 패키지를 개선하였습니다. joinMeeting 중복 참여 제한 로직을 강화하였습니다. * `getAllMeetings(Long cursorId, int size)`: * 목적: 모든 모임 목록을 페이징 처리하여 조회합니다. cursorId를 사용하여 무한 스크롤과 같은 커서 기반 페이징을 지원합니다. * 특징: @transactional(readOnly = true)를 통해 읽기 전용 트랜잭션으로 최적화되었습니다. * `getMeetingDetails(Long meetingId)`: * 목적: 특정 모임의 상세 정보를 조회합니다. 호스트, 장소, 태그 등 연관된 정보를 함께 가져옵니다. * 개선 예정: 현재 N+1 문제가 발생할 수 있어, 향후 Fetch Join을 통한 성능 최적화가 필요합니다. * `getStatusMyMeetings(Long memberId, Long cursorId, int size, MeetingStatus meetingStatus)`: * 목적: 특정 회원의 상태별(예: 모집 중, 완료) 모임 목록을 페이징 처리하여 조회합니다. * `getMeetingDetailAndMember(Long memberId, Long meetingId)`: * 목적: 호스트가 자신의 모임 상세 정보와 참여자 목록을 조회합니다. 참여자들의 닉네임을 함께 제공합니다. * `countMeetings(Long memberId)`: * 목적: 특정 회원이 참여한 모임의 총 개수를 반환합니다. * `joinMeeting(Long meetingId, Long memberId)`: * 목적: 회원이 특정 모임에 참여합니다. * 주요 개선: 모임 상태 검증(verifyRecruiting), 중복 참여 검증(`verifyNotAlreadyParticipant`), 모임 정원 초과 검증(verifyMeetingCount) 로직이 강화되었습니다. * 개선 예정: 동시성 문제(Race Condition) 해결을 위한 비관적 락(Pessimistic Lock) 적용이 필요합니다. * `leaveMeeting(Long meetingId, Long memberId)`: * 목적: 회원이 모임에서 탈퇴합니다. * 주요 개선: 호스트 탈퇴 방지(verifyNotHost), 모임 상태에 따른 탈퇴 가능 여부 검증(verifyLeave) 로직이 추가되었습니다. * 개선 예정: MEETING_NOT_FOUND 대신 CANNOT_LEAVE_COMPLETED_MEETING과 같은 더 구체적인 에러 …
1 parent f151ed1 commit 5f0b079

File tree

13 files changed

+233
-6
lines changed

13 files changed

+233
-6
lines changed

build.gradle

Lines changed: 4 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -83,6 +83,10 @@ dependencies {
8383

8484
// circuit breaker dependencies
8585
implementation 'io.github.resilience4j:resilience4j-spring-boot3:2.2.0'
86+
87+
// mail
88+
implementation 'org.springframework.boot:spring-boot-starter-mail'
89+
8690
}
8791

8892
dependencyManagement {

src/main/java/sevenstar/marineleisure/favorite/repository/FavoriteRepository.java

Lines changed: 8 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -36,4 +36,12 @@ List<FavoriteItemVO> findFavoritesByMemberIdAndCursorId(
3636
Pageable pageable
3737
);
3838
boolean existsByMemberIdAndSpotId(Long memberId, Long spotId);
39+
40+
@Query(value = """
41+
SELECT m.email
42+
FROM FavoriteSpot fs
43+
JOIN Member m ON fs.memberId = m.id
44+
WHERE fs.spotId = :spotId
45+
""")
46+
List<String> findEmailByFavoriteBestSpot(Long spotId);
3947
}

src/main/java/sevenstar/marineleisure/global/api/scheduler/SchedulerService.java

Lines changed: 8 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,7 @@
1212
import sevenstar.marineleisure.global.api.kakao.service.PresetSchedulerService;
1313
import sevenstar.marineleisure.global.api.khoa.service.KhoaApiService;
1414
import sevenstar.marineleisure.global.api.openmeteo.dto.service.OpenMeteoService;
15+
import sevenstar.marineleisure.global.mail.MailService;
1516
import sevenstar.marineleisure.spot.repository.SpotViewQuartileRepository;
1617

1718
@Service
@@ -24,7 +25,7 @@ public class SchedulerService {
2425
private final PresetSchedulerService presetSchedulerService;
2526
private final SpotViewQuartileRepository spotViewQuartileRepository;
2627

27-
28+
private final MailService mailService;
2829
private final Executor taskExecutor;
2930

3031
/**
@@ -55,6 +56,12 @@ public void scheduler() {
5556
// 모든 병렬 작업이 완료될 때까지 기다림
5657
CompletableFuture.allOf(openMeteoFuture, presetSchedulerFuture, spotViewQuartileFuture).join();
5758

59+
try {
60+
mailService.sendMailToHaveFavoriteBestSpot(today);
61+
} catch (Exception e) {
62+
log.error("Error sending mail to users with favorite best spots", e);
63+
}
64+
5865
log.info("=== update data ===");
5966
}
6067
}

src/main/java/sevenstar/marineleisure/global/enums/ActivityCategory.java

Lines changed: 12 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,21 @@
11
package sevenstar.marineleisure.global.enums;
22

3+
import lombok.Getter;
34
import sevenstar.marineleisure.global.exception.CustomException;
45
import sevenstar.marineleisure.global.exception.enums.CommonErrorCode;
56

7+
@Getter
68
public enum ActivityCategory {
7-
FISHING,
8-
SURFING,
9-
SCUBA,
10-
MUDFLAT;
9+
FISHING("낚시"),
10+
SURFING("서핑"),
11+
SCUBA("스쿠버다이빙"),
12+
MUDFLAT("갯벌체험");
13+
14+
private String koreanName;
15+
16+
ActivityCategory(String koreanName) {
17+
this.koreanName = koreanName;
18+
}
1119

1220
public static ActivityCategory parse(String category) {
1321
try {
Lines changed: 142 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,142 @@
1+
package sevenstar.marineleisure.global.mail;
2+
3+
import java.time.LocalDate;
4+
import java.util.ArrayList;
5+
import java.util.EnumMap;
6+
import java.util.HashMap;
7+
import java.util.HashSet;
8+
import java.util.List;
9+
import java.util.Map;
10+
import java.util.Set;
11+
12+
import org.springframework.mail.javamail.JavaMailSender;
13+
import org.springframework.mail.javamail.MimeMessageHelper;
14+
import org.springframework.stereotype.Service;
15+
16+
import jakarta.mail.internet.MimeMessage;
17+
import lombok.RequiredArgsConstructor;
18+
import lombok.extern.slf4j.Slf4j;
19+
import sevenstar.marineleisure.favorite.repository.FavoriteRepository;
20+
import sevenstar.marineleisure.global.enums.ActivityCategory;
21+
import sevenstar.marineleisure.global.enums.TotalIndex;
22+
import sevenstar.marineleisure.spot.dto.EmailContent;
23+
import sevenstar.marineleisure.spot.dto.detail.provider.ActivityProvider;
24+
25+
@Service
26+
@RequiredArgsConstructor
27+
@Slf4j
28+
public class MailService {
29+
private static final String MESSAGE_SUBJECT = "[MarineLeisure] 즐겨찾기한 스팟이 최상의 컨디션이에요!";
30+
31+
private final JavaMailSender javaMailSender;
32+
private final FavoriteRepository favoriteRepository;
33+
private final List<ActivityProvider> providers;
34+
35+
public void sendMail(String to, String subject, String htmlContent) {
36+
try {
37+
MimeMessage mimeMessage = javaMailSender.createMimeMessage();
38+
MimeMessageHelper helper = new MimeMessageHelper(mimeMessage, "UTF-8");
39+
helper.setFrom("[email protected]");
40+
helper.setTo(to);
41+
helper.setSubject(subject);
42+
helper.setText(htmlContent, true); // true = HTML
43+
44+
javaMailSender.send(mimeMessage);
45+
} catch (Exception e) {
46+
log.error("메일 전송 실패", e);
47+
}
48+
}
49+
50+
51+
public void sendMailToHaveFavoriteBestSpot(LocalDate date) {
52+
TotalIndex totalIndex = TotalIndex.VERY_GOOD;
53+
List<EmailContent> emailContents = new ArrayList<>();
54+
for (ActivityProvider provider : providers) {
55+
emailContents.addAll(provider.findEmailContent(totalIndex, date));
56+
}
57+
Map<String, Map<ActivityCategory, Set<String>>> result = new HashMap<>();
58+
for (EmailContent emailContent : emailContents) {
59+
List<String> emails = favoriteRepository.findEmailByFavoriteBestSpot(emailContent.spotId());
60+
for (String email : emails) {
61+
if (result.containsKey(email)) {
62+
result.get(email).get(emailContent.category()).add(emailContent.spotName());
63+
} else {
64+
Map<ActivityCategory, Set<String>> map = new EnumMap<>(ActivityCategory.class);
65+
for (ActivityCategory value : ActivityCategory.values()) {
66+
map.put(value, new HashSet<>());
67+
}
68+
map.get(emailContent.category()).add(emailContent.spotName());
69+
result.put(email, map);
70+
}
71+
}
72+
}
73+
for (Map.Entry<String, Map<ActivityCategory, Set<String>>> entry : result.entrySet()) {
74+
sendMail(entry.getKey(), MESSAGE_SUBJECT, transformEmailContent(entry.getValue()));
75+
}
76+
}
77+
78+
// private String transformEmailContent(Map<ActivityCategory, Set<String>> map) {
79+
// StringBuilder sb = new StringBuilder();
80+
// sb.append("<div style='font-family: Arial, sans-serif; font-size: 14px;'>");
81+
// sb.append("<p>안녕하세요, <strong>MarineLeisure</strong>입니다 🌊</p>");
82+
// sb.append("<p>고객님이 즐겨찾기한 장소 중, 오늘 같은 날 <strong>최상의 컨디션</strong>을 보이는 스팟들을 추천드립니다.</p>");
83+
//
84+
// sb.append("<ul>");
85+
// for (ActivityCategory category : ActivityCategory.values()) {
86+
// Set<String> spots = map.getOrDefault(category, Set.of());
87+
// String spotList = spots.isEmpty() ? "없어요 😢" : String.join(", ", spots);
88+
// sb.append("<li><strong>")
89+
// .append(category.getKoreanName())
90+
// .append("</strong>에 좋은 스팟: ")
91+
// .append(spotList)
92+
// .append("</li>");
93+
// }
94+
// sb.append("</ul>");
95+
//
96+
// sb.append("<p>👉 <a href=\"https://marineleisure.com\" target=\"_blank\">MarineLeisure 앱에서 자세히 보기</a></p>");
97+
// sb.append("<p>안전하고 즐거운 하루 보내세요 😊<br>MarineLeisure 드림</p>");
98+
// sb.append("</div>");
99+
//
100+
// return sb.toString();
101+
// }
102+
103+
private String transformEmailContent(Map<ActivityCategory, Set<String>> map) {
104+
StringBuilder sb = new StringBuilder();
105+
106+
sb.append("<div style='font-family: \"Apple SD Gothic Neo\", sans-serif; background-color: #f4f4f4; padding: 20px;'>")
107+
.append("<div style='max-width: 600px; margin: auto; background-color: #ffffff; border-radius: 10px; padding: 30px; box-shadow: 0 0 10px rgba(0,0,0,0.05);'>")
108+
109+
.append("<h2 style='color: #0077b6; text-align: center;'>🌊 MarineLeisure 추천 스팟 알림</h2>")
110+
.append("<p style='font-size: 15px; color: #333;'>")
111+
.append("고객님이 즐겨찾기한 해양 활동 스팟 중, 오늘 같은 날 <strong style='color: #0077b6;'>최고의 컨디션</strong>을 보이는 장소를 추천드릴게요!")
112+
.append("</p>");
113+
114+
for (ActivityCategory category : ActivityCategory.values()) {
115+
Set<String> spots = map.getOrDefault(category, Set.of());
116+
if (!spots.isEmpty()) {
117+
sb.append("<div style='margin-top: 20px;'>")
118+
.append("<h3 style='color: #023e8a; font-size: 16px;'>")
119+
.append("✔️ ").append(category.getKoreanName()).append(" 추천 스팟")
120+
.append("</h3>")
121+
.append("<ul style='padding-left: 20px;'>");
122+
for (String spot : spots) {
123+
sb.append("<li>").append(spot).append("</li>");
124+
}
125+
sb.append("</ul></div>");
126+
}
127+
}
128+
129+
sb.append("<div style='text-align: center; margin-top: 30px;'>")
130+
.append("<a href='https://marineleisure.vercel.app' target='_blank' style='background-color: #00b4d8; color: white; padding: 12px 24px; text-decoration: none; border-radius: 5px; font-weight: bold;'>MarineLeisure 앱에서 확인하기</a>")
131+
.append("</div>")
132+
133+
.append("<p style='margin-top: 30px; font-size: 14px; color: #555;'>")
134+
.append("안전하고 즐거운 하루 보내세요!<br><strong>MarineLeisure 드림</strong>")
135+
.append("</p>")
136+
137+
.append("</div></div>");
138+
139+
return sb.toString();
140+
}
141+
142+
}
Lines changed: 10 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
package sevenstar.marineleisure.spot.dto;
2+
3+
import sevenstar.marineleisure.global.enums.ActivityCategory;
4+
5+
public record EmailContent(
6+
Long spotId,
7+
String spotName,
8+
ActivityCategory category
9+
) {
10+
}

src/main/java/sevenstar/marineleisure/spot/dto/detail/provider/ActivityProvider.java

Lines changed: 4 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -18,8 +18,10 @@
1818
import sevenstar.marineleisure.global.api.openmeteo.dto.item.UvIndexItem;
1919
import sevenstar.marineleisure.global.enums.ActivityCategory;
2020
import sevenstar.marineleisure.global.enums.FishingType;
21+
import sevenstar.marineleisure.global.enums.TotalIndex;
2122
import sevenstar.marineleisure.global.utils.GeoUtils;
2223
import sevenstar.marineleisure.spot.domain.OutdoorSpot;
24+
import sevenstar.marineleisure.spot.dto.EmailContent;
2325
import sevenstar.marineleisure.spot.repository.ActivityRepository;
2426
import sevenstar.marineleisure.spot.repository.OutdoorSpotRepository;
2527

@@ -43,6 +45,8 @@ public abstract class ActivityProvider {
4345

4446
public abstract void update(LocalDate startDate, LocalDate endDate);
4547

48+
public abstract List<EmailContent> findEmailContent(TotalIndex totalIndex, LocalDate forecastDate);
49+
4650
@Transactional
4751
protected OutdoorSpot createOutdoorSpot(KhoaItem item, FishingType fishingType) {
4852
return outdoorSpotRepository.findByLatitudeAndLongitudeAndCategory(item.getLatitude(), item.getLongitude(),

src/main/java/sevenstar/marineleisure/spot/dto/detail/provider/FishingProvider.java

Lines changed: 6 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,7 @@
2323
import sevenstar.marineleisure.global.enums.TotalIndex;
2424
import sevenstar.marineleisure.global.utils.DateUtils;
2525
import sevenstar.marineleisure.spot.domain.OutdoorSpot;
26+
import sevenstar.marineleisure.spot.dto.EmailContent;
2627
import sevenstar.marineleisure.spot.dto.projection.FishingReadProjection;
2728
import sevenstar.marineleisure.spot.mapper.SpotDetailMapper;
2829
import sevenstar.marineleisure.spot.repository.ActivityRepository;
@@ -94,6 +95,11 @@ public void update(LocalDate startDate, LocalDate endDate) {
9495
}
9596
}
9697

98+
@Override
99+
public List<EmailContent> findEmailContent(TotalIndex totalIndex, LocalDate forecastDate) {
100+
return fishingRepository.findEmailContentByTotalIndexAndForecastDate(totalIndex, forecastDate);
101+
}
102+
97103
private List<ActivitySpotDetail> transform(List<FishingReadProjection> fishingForecasts) {
98104
List<ActivitySpotDetail> details = new ArrayList<>();
99105
for (FishingReadProjection fishingForecast : fishingForecasts) {

src/main/java/sevenstar/marineleisure/spot/dto/detail/provider/MudflatProvider.java

Lines changed: 6 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,7 @@
1919
import sevenstar.marineleisure.global.enums.TotalIndex;
2020
import sevenstar.marineleisure.global.utils.DateUtils;
2121
import sevenstar.marineleisure.spot.domain.OutdoorSpot;
22+
import sevenstar.marineleisure.spot.dto.EmailContent;
2223
import sevenstar.marineleisure.spot.mapper.SpotDetailMapper;
2324
import sevenstar.marineleisure.spot.repository.ActivityRepository;
2425

@@ -73,6 +74,11 @@ public void update(LocalDate startDate, LocalDate endDate) {
7374
}
7475
}
7576

77+
@Override
78+
public List<EmailContent> findEmailContent(TotalIndex totalIndex, LocalDate forecastDate) {
79+
return mudflatRepository.findEmailContentByTotalIndexAndForecastDate(totalIndex, forecastDate);
80+
}
81+
7682
private List<ActivitySpotDetail> transform(List<Mudflat> mudflatForecasts) {
7783
List<ActivitySpotDetail> details = new ArrayList<>();
7884
for (Mudflat mudflatForecast : mudflatForecasts) {

src/main/java/sevenstar/marineleisure/spot/dto/detail/provider/ScubaProvider.java

Lines changed: 6 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,7 @@
2121
import sevenstar.marineleisure.global.enums.TotalIndex;
2222
import sevenstar.marineleisure.global.utils.DateUtils;
2323
import sevenstar.marineleisure.spot.domain.OutdoorSpot;
24+
import sevenstar.marineleisure.spot.dto.EmailContent;
2425
import sevenstar.marineleisure.spot.mapper.SpotDetailMapper;
2526
import sevenstar.marineleisure.spot.repository.ActivityRepository;
2627

@@ -76,6 +77,11 @@ public void update(LocalDate startDate, LocalDate endDate) {
7677
}
7778
}
7879

80+
@Override
81+
public List<EmailContent> findEmailContent(TotalIndex totalIndex, LocalDate forecastDate) {
82+
return scubaRepository.findEmailContentByTotalIndexAndForecastDate(totalIndex, forecastDate);
83+
}
84+
7985
private List<ActivitySpotDetail> transform(List<Scuba> scubaForecasts) {
8086
List<ActivitySpotDetail> details = new ArrayList<>();
8187
for (Scuba scubaForecast : scubaForecasts) {

0 commit comments

Comments
 (0)