(`RPb
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
GZ~q4|mN%#X
literal 0
HcmV?d00001
diff --git a/vote/gradle/wrapper/gradle-wrapper.properties b/vote/gradle/wrapper/gradle-wrapper.properties
new file mode 100644
index 0000000..e2847c8
--- /dev/null
+++ b/vote/gradle/wrapper/gradle-wrapper.properties
@@ -0,0 +1,7 @@
+distributionBase=GRADLE_USER_HOME
+distributionPath=wrapper/dists
+distributionUrl=https\://services.gradle.org/distributions/gradle-8.11.1-bin.zip
+networkTimeout=10000
+validateDistributionUrl=true
+zipStoreBase=GRADLE_USER_HOME
+zipStorePath=wrapper/dists
diff --git a/vote/gradlew b/vote/gradlew
new file mode 100755
index 0000000..f5feea6
--- /dev/null
+++ b/vote/gradlew
@@ -0,0 +1,252 @@
+#!/bin/sh
+
+#
+# Copyright © 2015-2021 the original authors.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# https://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+#
+# SPDX-License-Identifier: Apache-2.0
+#
+
+##############################################################################
+#
+# Gradle start up script for POSIX generated by Gradle.
+#
+# Important for running:
+#
+# (1) You need a POSIX-compliant shell to run this script. If your /bin/sh is
+# noncompliant, but you have some other compliant shell such as ksh or
+# bash, then to run this script, type that shell name before the whole
+# command line, like:
+#
+# ksh Gradle
+#
+# Busybox and similar reduced shells will NOT work, because this script
+# requires all of these POSIX shell features:
+# * functions;
+# * expansions «$var», «${var}», «${var:-default}», «${var+SET}»,
+# «${var#prefix}», «${var%suffix}», and «$( cmd )»;
+# * compound commands having a testable exit status, especially «case»;
+# * various built-in commands including «command», «set», and «ulimit».
+#
+# Important for patching:
+#
+# (2) This script targets any POSIX shell, so it avoids extensions provided
+# by Bash, Ksh, etc; in particular arrays are avoided.
+#
+# The "traditional" practice of packing multiple parameters into a
+# space-separated string is a well documented source of bugs and security
+# problems, so this is (mostly) avoided, by progressively accumulating
+# options in "$@", and eventually passing that to Java.
+#
+# Where the inherited environment variables (DEFAULT_JVM_OPTS, JAVA_OPTS,
+# and GRADLE_OPTS) rely on word-splitting, this is performed explicitly;
+# see the in-line comments for details.
+#
+# There are tweaks for specific operating systems such as AIX, CygWin,
+# Darwin, MinGW, and NonStop.
+#
+# (3) This script is generated from the Groovy template
+# https://github.com/gradle/gradle/blob/HEAD/platforms/jvm/plugins-application/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt
+# within the Gradle project.
+#
+# You can find Gradle at https://github.com/gradle/gradle/.
+#
+##############################################################################
+
+# Attempt to set APP_HOME
+
+# Resolve links: $0 may be a link
+app_path=$0
+
+# Need this for daisy-chained symlinks.
+while
+ APP_HOME=${app_path%"${app_path##*/}"} # leaves a trailing /; empty if no leading path
+ [ -h "$app_path" ]
+do
+ ls=$( ls -ld "$app_path" )
+ link=${ls#*' -> '}
+ case $link in #(
+ /*) app_path=$link ;; #(
+ *) app_path=$APP_HOME$link ;;
+ esac
+done
+
+# This is normally unused
+# shellcheck disable=SC2034
+APP_BASE_NAME=${0##*/}
+# Discard cd standard output in case $CDPATH is set (https://github.com/gradle/gradle/issues/25036)
+APP_HOME=$( cd -P "${APP_HOME:-./}" > /dev/null && printf '%s
+' "$PWD" ) || exit
+
+# Use the maximum available, or set MAX_FD != -1 to use that value.
+MAX_FD=maximum
+
+warn () {
+ echo "$*"
+} >&2
+
+die () {
+ echo
+ echo "$*"
+ echo
+ exit 1
+} >&2
+
+# OS specific support (must be 'true' or 'false').
+cygwin=false
+msys=false
+darwin=false
+nonstop=false
+case "$( uname )" in #(
+ CYGWIN* ) cygwin=true ;; #(
+ Darwin* ) darwin=true ;; #(
+ MSYS* | MINGW* ) msys=true ;; #(
+ NONSTOP* ) nonstop=true ;;
+esac
+
+CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
+
+
+# Determine the Java command to use to start the JVM.
+if [ -n "$JAVA_HOME" ] ; then
+ if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
+ # IBM's JDK on AIX uses strange locations for the executables
+ JAVACMD=$JAVA_HOME/jre/sh/java
+ else
+ JAVACMD=$JAVA_HOME/bin/java
+ fi
+ if [ ! -x "$JAVACMD" ] ; then
+ die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
+
+Please set the JAVA_HOME variable in your environment to match the
+location of your Java installation."
+ fi
+else
+ JAVACMD=java
+ if ! command -v java >/dev/null 2>&1
+ then
+ die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
+
+Please set the JAVA_HOME variable in your environment to match the
+location of your Java installation."
+ fi
+fi
+
+# Increase the maximum file descriptors if we can.
+if ! "$cygwin" && ! "$darwin" && ! "$nonstop" ; then
+ case $MAX_FD in #(
+ max*)
+ # In POSIX sh, ulimit -H is undefined. That's why the result is checked to see if it worked.
+ # shellcheck disable=SC2039,SC3045
+ MAX_FD=$( ulimit -H -n ) ||
+ warn "Could not query maximum file descriptor limit"
+ esac
+ case $MAX_FD in #(
+ '' | soft) :;; #(
+ *)
+ # In POSIX sh, ulimit -n is undefined. That's why the result is checked to see if it worked.
+ # shellcheck disable=SC2039,SC3045
+ ulimit -n "$MAX_FD" ||
+ warn "Could not set maximum file descriptor limit to $MAX_FD"
+ esac
+fi
+
+# Collect all arguments for the java command, stacking in reverse order:
+# * args from the command line
+# * the main class name
+# * -classpath
+# * -D...appname settings
+# * --module-path (only if needed)
+# * DEFAULT_JVM_OPTS, JAVA_OPTS, and GRADLE_OPTS environment variables.
+
+# For Cygwin or MSYS, switch paths to Windows format before running java
+if "$cygwin" || "$msys" ; then
+ APP_HOME=$( cygpath --path --mixed "$APP_HOME" )
+ CLASSPATH=$( cygpath --path --mixed "$CLASSPATH" )
+
+ JAVACMD=$( cygpath --unix "$JAVACMD" )
+
+ # Now convert the arguments - kludge to limit ourselves to /bin/sh
+ for arg do
+ if
+ case $arg in #(
+ -*) false ;; # don't mess with options #(
+ /?*) t=${arg#/} t=/${t%%/*} # looks like a POSIX filepath
+ [ -e "$t" ] ;; #(
+ *) false ;;
+ esac
+ then
+ arg=$( cygpath --path --ignore --mixed "$arg" )
+ fi
+ # Roll the args list around exactly as many times as the number of
+ # args, so each arg winds up back in the position where it started, but
+ # possibly modified.
+ #
+ # NB: a `for` loop captures its iteration list before it begins, so
+ # changing the positional parameters here affects neither the number of
+ # iterations, nor the values presented in `arg`.
+ shift # remove old arg
+ set -- "$@" "$arg" # push replacement arg
+ done
+fi
+
+
+# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
+DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"'
+
+# Collect all arguments for the java command:
+# * DEFAULT_JVM_OPTS, JAVA_OPTS, JAVA_OPTS, and optsEnvironmentVar are not allowed to contain shell fragments,
+# and any embedded shellness will be escaped.
+# * For example: A user cannot expect ${Hostname} to be expanded, as it is an environment variable and will be
+# treated as '${Hostname}' itself on the command line.
+
+set -- \
+ "-Dorg.gradle.appname=$APP_BASE_NAME" \
+ -classpath "$CLASSPATH" \
+ org.gradle.wrapper.GradleWrapperMain \
+ "$@"
+
+# Stop when "xargs" is not available.
+if ! command -v xargs >/dev/null 2>&1
+then
+ die "xargs is not available"
+fi
+
+# Use "xargs" to parse quoted args.
+#
+# With -n1 it outputs one arg per line, with the quotes and backslashes removed.
+#
+# In Bash we could simply go:
+#
+# readarray ARGS < <( xargs -n1 <<<"$var" ) &&
+# set -- "${ARGS[@]}" "$@"
+#
+# but POSIX shell has neither arrays nor command substitution, so instead we
+# post-process each arg (as a line of input to sed) to backslash-escape any
+# character that might be a shell metacharacter, then use eval to reverse
+# that process (while maintaining the separation between arguments), and wrap
+# the whole thing up as a single "set" statement.
+#
+# This will of course break if any of these variables contains a newline or
+# an unmatched quote.
+#
+
+eval "set -- $(
+ printf '%s\n' "$DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS" |
+ xargs -n1 |
+ sed ' s~[^-[:alnum:]+,./:=@_]~\\&~g; ' |
+ tr '\n' ' '
+ )" '"$@"'
+
+exec "$JAVACMD" "$@"
diff --git a/vote/gradlew.bat b/vote/gradlew.bat
new file mode 100644
index 0000000..9d21a21
--- /dev/null
+++ b/vote/gradlew.bat
@@ -0,0 +1,94 @@
+@rem
+@rem Copyright 2015 the original author or authors.
+@rem
+@rem Licensed under the Apache License, Version 2.0 (the "License");
+@rem you may not use this file except in compliance with the License.
+@rem You may obtain a copy of the License at
+@rem
+@rem https://www.apache.org/licenses/LICENSE-2.0
+@rem
+@rem Unless required by applicable law or agreed to in writing, software
+@rem distributed under the License is distributed on an "AS IS" BASIS,
+@rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+@rem See the License for the specific language governing permissions and
+@rem limitations under the License.
+@rem
+@rem SPDX-License-Identifier: Apache-2.0
+@rem
+
+@if "%DEBUG%"=="" @echo off
+@rem ##########################################################################
+@rem
+@rem Gradle startup script for Windows
+@rem
+@rem ##########################################################################
+
+@rem Set local scope for the variables with windows NT shell
+if "%OS%"=="Windows_NT" setlocal
+
+set DIRNAME=%~dp0
+if "%DIRNAME%"=="" set DIRNAME=.
+@rem This is normally unused
+set APP_BASE_NAME=%~n0
+set APP_HOME=%DIRNAME%
+
+@rem Resolve any "." and ".." in APP_HOME to make it shorter.
+for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi
+
+@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
+set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m"
+
+@rem Find java.exe
+if defined JAVA_HOME goto findJavaFromJavaHome
+
+set JAVA_EXE=java.exe
+%JAVA_EXE% -version >NUL 2>&1
+if %ERRORLEVEL% equ 0 goto execute
+
+echo. 1>&2
+echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. 1>&2
+echo. 1>&2
+echo Please set the JAVA_HOME variable in your environment to match the 1>&2
+echo location of your Java installation. 1>&2
+
+goto fail
+
+:findJavaFromJavaHome
+set JAVA_HOME=%JAVA_HOME:"=%
+set JAVA_EXE=%JAVA_HOME%/bin/java.exe
+
+if exist "%JAVA_EXE%" goto execute
+
+echo. 1>&2
+echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% 1>&2
+echo. 1>&2
+echo Please set the JAVA_HOME variable in your environment to match the 1>&2
+echo location of your Java installation. 1>&2
+
+goto fail
+
+:execute
+@rem Setup the command line
+
+set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
+
+
+@rem Execute Gradle
+"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %*
+
+:end
+@rem End local scope for the variables with windows NT shell
+if %ERRORLEVEL% equ 0 goto mainEnd
+
+:fail
+rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
+rem the _cmd.exe /c_ return code!
+set EXIT_CODE=%ERRORLEVEL%
+if %EXIT_CODE% equ 0 set EXIT_CODE=1
+if not ""=="%GRADLE_EXIT_CONSOLE%" exit %EXIT_CODE%
+exit /b %EXIT_CODE%
+
+:mainEnd
+if "%OS%"=="Windows_NT" endlocal
+
+:omega
diff --git a/vote/settings.gradle b/vote/settings.gradle
new file mode 100644
index 0000000..bfce6e0
--- /dev/null
+++ b/vote/settings.gradle
@@ -0,0 +1 @@
+rootProject.name = 'vote'
diff --git a/vote/src/main/java/photoground/ceos/vote/VoteApplication.java b/vote/src/main/java/photoground/ceos/vote/VoteApplication.java
new file mode 100644
index 0000000..2537e7e
--- /dev/null
+++ b/vote/src/main/java/photoground/ceos/vote/VoteApplication.java
@@ -0,0 +1,15 @@
+package photoground.ceos.vote;
+
+import org.springframework.boot.SpringApplication;
+import org.springframework.boot.autoconfigure.SpringBootApplication;
+import org.springframework.data.jpa.repository.config.EnableJpaAuditing;
+
+@SpringBootApplication
+@EnableJpaAuditing
+public class VoteApplication {
+
+ public static void main(String[] args) {
+ SpringApplication.run(VoteApplication.class, args);
+ }
+
+}
diff --git a/vote/src/main/java/photoground/ceos/vote/global/entity/BaseTimeEntity.java b/vote/src/main/java/photoground/ceos/vote/global/entity/BaseTimeEntity.java
new file mode 100644
index 0000000..c07e02d
--- /dev/null
+++ b/vote/src/main/java/photoground/ceos/vote/global/entity/BaseTimeEntity.java
@@ -0,0 +1,25 @@
+package photoground.ceos.vote.global.entity;
+
+import jakarta.persistence.Column;
+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;
+import org.springframework.data.jpa.domain.support.AuditingEntityListener;
+
+@Getter
+@MappedSuperclass
+@EntityListeners(AuditingEntityListener.class)
+public class BaseTimeEntity {
+
+ @CreatedDate
+ @Column(name = "created_at", updatable = false, columnDefinition = "timestamp")
+ private LocalDateTime createdAt;
+
+ @LastModifiedDate
+ @Column(name = "updated_at", columnDefinition = "timestamp")
+ private LocalDateTime updatedAt;
+
+}
diff --git a/vote/src/main/java/photoground/ceos/vote/global/exception/ApiExceptionHandler.java b/vote/src/main/java/photoground/ceos/vote/global/exception/ApiExceptionHandler.java
new file mode 100644
index 0000000..c389e42
--- /dev/null
+++ b/vote/src/main/java/photoground/ceos/vote/global/exception/ApiExceptionHandler.java
@@ -0,0 +1,37 @@
+package photoground.ceos.vote.global.exception;
+
+import lombok.extern.slf4j.Slf4j;
+import org.springframework.http.HttpStatus;
+import org.springframework.http.ResponseEntity;
+import org.springframework.web.bind.MethodArgumentNotValidException;
+import org.springframework.web.bind.annotation.ExceptionHandler;
+import org.springframework.web.bind.annotation.RestControllerAdvice;
+
+@Slf4j
+@RestControllerAdvice
+public class ApiExceptionHandler {
+
+ @ExceptionHandler(value = CustomException.class)
+ public ResponseEntity handleCustomException(CustomException e) {
+ log.error("[handleCustomException] {} : {}", e.getErrorCode().name(), e.getErrorCode().getMessage());
+ return ErrorResponse.fromException(e);
+ }
+
+ @ExceptionHandler(MethodArgumentNotValidException.class)
+ public ResponseEntity handleValidationException(MethodArgumentNotValidException ex) {
+ String errorMessage = ex.getBindingResult()
+ .getAllErrors()
+ .get(0)
+ .getDefaultMessage(); // 첫 번째 에러 메시지 가져오기
+
+ log.error("[Validation Error] {}", errorMessage);
+ return ResponseEntity
+ .status(HttpStatus.BAD_REQUEST)
+ .body(ErrorResponse.builder()
+ .status(HttpStatus.BAD_REQUEST)
+ .code("VALIDATION_ERROR")
+ .message(errorMessage)
+ .build());
+ }
+
+}
\ No newline at end of file
diff --git a/vote/src/main/java/photoground/ceos/vote/global/exception/CustomException.java b/vote/src/main/java/photoground/ceos/vote/global/exception/CustomException.java
new file mode 100644
index 0000000..499b934
--- /dev/null
+++ b/vote/src/main/java/photoground/ceos/vote/global/exception/CustomException.java
@@ -0,0 +1,19 @@
+package photoground.ceos.vote.global.exception;
+
+import lombok.Getter;
+
+@Getter
+public class CustomException extends RuntimeException {
+ private ErrorCode errorCode;
+
+ private String info;
+
+ public CustomException(ErrorCode errorCode){
+ this.errorCode = errorCode;
+ }
+
+ public CustomException(ErrorCode errorCode, String info){
+ this.errorCode = errorCode;
+ this.info = info;
+ }
+}
diff --git a/vote/src/main/java/photoground/ceos/vote/global/exception/ErrorCode.java b/vote/src/main/java/photoground/ceos/vote/global/exception/ErrorCode.java
new file mode 100644
index 0000000..78544d6
--- /dev/null
+++ b/vote/src/main/java/photoground/ceos/vote/global/exception/ErrorCode.java
@@ -0,0 +1,22 @@
+package photoground.ceos.vote.global.exception;
+
+import lombok.Getter;
+import lombok.RequiredArgsConstructor;
+import org.springframework.http.HttpStatus;
+
+@Getter
+@RequiredArgsConstructor
+public enum ErrorCode {
+ // Auth
+ BAD_CREDENTIALS(HttpStatus.UNAUTHORIZED, "아이디 또는 비밀번호가 잘못되었습니다."),
+ FAIL_AUTHENTICATION(HttpStatus.UNAUTHORIZED, "로그인에 실패했습니다. 다시 시도해주세요."),
+ INVALID_TOKEN(HttpStatus.UNAUTHORIZED, "유효하지 않은 토큰입니다."),
+ TOKEN_EXPIRED(HttpStatus.UNAUTHORIZED, "토큰이 만료되었습니다."),
+ FAIL_AUTHORIZATION(HttpStatus.FORBIDDEN, "권한이 없는 요청입니다."),
+ INTERNAL_SERVER_ERROR(HttpStatus.INTERNAL_SERVER_ERROR, "INTERNAL SERVER ERROR");
+
+
+ private final HttpStatus status;
+ private final String message;
+
+}
diff --git a/vote/src/main/java/photoground/ceos/vote/global/exception/ErrorResponse.java b/vote/src/main/java/photoground/ceos/vote/global/exception/ErrorResponse.java
new file mode 100644
index 0000000..c14aba2
--- /dev/null
+++ b/vote/src/main/java/photoground/ceos/vote/global/exception/ErrorResponse.java
@@ -0,0 +1,39 @@
+package photoground.ceos.vote.global.exception;
+
+import lombok.Builder;
+import lombok.Getter;
+import org.springframework.http.HttpStatus;
+import org.springframework.http.ResponseEntity;
+import org.springframework.web.bind.annotation.PathVariable;
+
+@Getter
+@Builder
+public class ErrorResponse {
+
+ private final HttpStatus status; // HTTP 상태 코드
+ private final String code; // 에러 코드
+ private final String message; // 에러 메시지
+
+ // ErrorResponse 생성 메서드
+ public static ErrorResponse of(HttpStatus status, String code, String message) {
+ return ErrorResponse.builder()
+ .status(status)
+ .message(message)
+ .code(code)
+ .build();
+ }
+
+ public static ResponseEntity fromException(CustomException e) {
+ String message = e.getErrorCode().getMessage();
+ if (e.getInfo() != null) {
+ message += " " + e.getInfo(); // 추가 정보가 있는 경우 결합
+ }
+ return ResponseEntity
+ .status(e.getErrorCode().getStatus())
+ .body(ErrorResponse.builder()
+ .status(e.getErrorCode().getStatus())
+ .code(e.getErrorCode().name())
+ .message(message)
+ .build());
+ }
+}
\ No newline at end of file
diff --git a/vote/src/test/java/photoground/ceos/vote/VoteApplicationTests.java b/vote/src/test/java/photoground/ceos/vote/VoteApplicationTests.java
new file mode 100644
index 0000000..c299f77
--- /dev/null
+++ b/vote/src/test/java/photoground/ceos/vote/VoteApplicationTests.java
@@ -0,0 +1,13 @@
+package photoground.ceos.vote;
+
+import org.junit.jupiter.api.Test;
+import org.springframework.boot.test.context.SpringBootTest;
+
+@SpringBootTest
+class VoteApplicationTests {
+
+ @Test
+ void contextLoads() {
+ }
+
+}
From 3e41670ce4a73d68f59686af5ec60bcfb7603964 Mon Sep 17 00:00:00 2001
From: sh0311 <0311alice@naver.com>
Date: Sat, 28 Dec 2024 14:28:43 +0900
Subject: [PATCH 02/27] =?UTF-8?q?=E2=9C=A8=20feat=20:=20entity=20=EC=83=9D?=
=?UTF-8?q?=EC=84=B1?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
.idea/uiDesigner.xml | 124 ++++++++++++++++++
.../vote/domain/member/entity/Member.java | 40 ++++++
.../ceos/vote/domain/member/entity/Part.java | 11 ++
.../ceos/vote/domain/member/entity/Team.java | 14 ++
.../vote/domain/vote/entity/LeaderVote.java | 23 ++++
.../vote/domain/vote/entity/TeamVote.java | 17 +++
.../ceos/vote/domain/vote/entity/Vote.java | 35 +++++
.../ceos/vote/test/TestController.java | 14 ++
8 files changed, 278 insertions(+)
create mode 100644 .idea/uiDesigner.xml
create mode 100644 vote/src/main/java/photoground/ceos/vote/domain/member/entity/Member.java
create mode 100644 vote/src/main/java/photoground/ceos/vote/domain/member/entity/Part.java
create mode 100644 vote/src/main/java/photoground/ceos/vote/domain/member/entity/Team.java
create mode 100644 vote/src/main/java/photoground/ceos/vote/domain/vote/entity/LeaderVote.java
create mode 100644 vote/src/main/java/photoground/ceos/vote/domain/vote/entity/TeamVote.java
create mode 100644 vote/src/main/java/photoground/ceos/vote/domain/vote/entity/Vote.java
create mode 100644 vote/src/main/java/photoground/ceos/vote/test/TestController.java
diff --git a/.idea/uiDesigner.xml b/.idea/uiDesigner.xml
new file mode 100644
index 0000000..2b63946
--- /dev/null
+++ b/.idea/uiDesigner.xml
@@ -0,0 +1,124 @@
+
+
+
+
+ -
+
+
+ -
+
+
+ -
+
+
+ -
+
+
+ -
+
+
+
+
+
+ -
+
+
+
+
+
+ -
+
+
+
+
+
+ -
+
+
+
+
+
+ -
+
+
+
+
+ -
+
+
+
+
+ -
+
+
+
+
+ -
+
+
+
+
+ -
+
+
+
+
+ -
+
+
+
+
+ -
+
+
+ -
+
+
+
+
+ -
+
+
+
+
+ -
+
+
+
+
+ -
+
+
+
+
+ -
+
+
+
+
+ -
+
+
+ -
+
+
+ -
+
+
+ -
+
+
+ -
+
+
+
+
+ -
+
+
+ -
+
+
+
+
+
\ No newline at end of file
diff --git a/vote/src/main/java/photoground/ceos/vote/domain/member/entity/Member.java b/vote/src/main/java/photoground/ceos/vote/domain/member/entity/Member.java
new file mode 100644
index 0000000..9073c60
--- /dev/null
+++ b/vote/src/main/java/photoground/ceos/vote/domain/member/entity/Member.java
@@ -0,0 +1,40 @@
+package photoground.ceos.vote.domain.member.entity;
+
+import jakarta.persistence.Column;
+import jakarta.persistence.Entity;
+import jakarta.persistence.EnumType;
+import jakarta.persistence.Enumerated;
+import jakarta.persistence.GeneratedValue;
+import jakarta.persistence.GenerationType;
+import jakarta.persistence.Id;
+import lombok.AccessLevel;
+import lombok.AllArgsConstructor;
+import lombok.Builder;
+import lombok.Getter;
+import lombok.NoArgsConstructor;
+import photoground.ceos.vote.global.entity.BaseTimeEntity;
+
+@Entity
+@NoArgsConstructor(access = AccessLevel.PROTECTED)
+@Getter
+@Builder
+@AllArgsConstructor
+public class Member extends BaseTimeEntity {
+
+ @Id
+ @GeneratedValue(strategy = GenerationType.IDENTITY)
+ @Column(name = "member_id")
+ private Long id;
+
+ private Long username;
+ private String password;
+ private String email;
+ private String name;
+
+ @Enumerated(EnumType.STRING)
+ private Part part;
+
+ @Enumerated(EnumType.STRING)
+ private Team team;
+
+}
diff --git a/vote/src/main/java/photoground/ceos/vote/domain/member/entity/Part.java b/vote/src/main/java/photoground/ceos/vote/domain/member/entity/Part.java
new file mode 100644
index 0000000..5b5a063
--- /dev/null
+++ b/vote/src/main/java/photoground/ceos/vote/domain/member/entity/Part.java
@@ -0,0 +1,11 @@
+package photoground.ceos.vote.domain.member.entity;
+
+import lombok.AllArgsConstructor;
+
+@AllArgsConstructor
+public enum Part {
+ FRONTEND("프론트엔드"),
+ BACKEND("백엔드");
+
+ private final String name;
+}
diff --git a/vote/src/main/java/photoground/ceos/vote/domain/member/entity/Team.java b/vote/src/main/java/photoground/ceos/vote/domain/member/entity/Team.java
new file mode 100644
index 0000000..767e0e1
--- /dev/null
+++ b/vote/src/main/java/photoground/ceos/vote/domain/member/entity/Team.java
@@ -0,0 +1,14 @@
+package photoground.ceos.vote.domain.member.entity;
+
+import lombok.AllArgsConstructor;
+
+@AllArgsConstructor
+public enum Team {
+ PHOTO_GROUND("포토그라운드"),
+ ANGEL_BRIDGE("엔젤브릿지"),
+ PEDAL_GENIE("페달지니"),
+ CAKE_WAY("케이크WAY"),
+ CUPFEE_DEAL("커피딜");
+
+ private final String name;
+}
diff --git a/vote/src/main/java/photoground/ceos/vote/domain/vote/entity/LeaderVote.java b/vote/src/main/java/photoground/ceos/vote/domain/vote/entity/LeaderVote.java
new file mode 100644
index 0000000..4233d79
--- /dev/null
+++ b/vote/src/main/java/photoground/ceos/vote/domain/vote/entity/LeaderVote.java
@@ -0,0 +1,23 @@
+package photoground.ceos.vote.domain.vote.entity;
+
+import jakarta.persistence.DiscriminatorValue;
+import jakarta.persistence.Entity;
+import jakarta.persistence.FetchType;
+import jakarta.persistence.JoinColumn;
+import jakarta.persistence.ManyToOne;
+import lombok.AccessLevel;
+import lombok.Getter;
+import lombok.NoArgsConstructor;
+import photoground.ceos.vote.domain.member.entity.Member;
+
+@Entity
+@NoArgsConstructor(access = AccessLevel.PROTECTED)
+@Getter
+@DiscriminatorValue("L")
+public class LeaderVote extends Vote {
+
+ @ManyToOne(fetch = FetchType.LAZY)
+ @JoinColumn(name = "candidate_id")
+ private Member candidate;
+
+}
diff --git a/vote/src/main/java/photoground/ceos/vote/domain/vote/entity/TeamVote.java b/vote/src/main/java/photoground/ceos/vote/domain/vote/entity/TeamVote.java
new file mode 100644
index 0000000..5c5ede4
--- /dev/null
+++ b/vote/src/main/java/photoground/ceos/vote/domain/vote/entity/TeamVote.java
@@ -0,0 +1,17 @@
+package photoground.ceos.vote.domain.vote.entity;
+
+import jakarta.persistence.DiscriminatorValue;
+import jakarta.persistence.Entity;
+import lombok.AccessLevel;
+import lombok.Getter;
+import lombok.NoArgsConstructor;
+import photoground.ceos.vote.domain.member.entity.Team;
+
+@Entity
+@NoArgsConstructor(access = AccessLevel.PROTECTED)
+@Getter
+@DiscriminatorValue("T")
+public class TeamVote extends Vote {
+
+ private Team team;
+}
diff --git a/vote/src/main/java/photoground/ceos/vote/domain/vote/entity/Vote.java b/vote/src/main/java/photoground/ceos/vote/domain/vote/entity/Vote.java
new file mode 100644
index 0000000..e20c216
--- /dev/null
+++ b/vote/src/main/java/photoground/ceos/vote/domain/vote/entity/Vote.java
@@ -0,0 +1,35 @@
+package photoground.ceos.vote.domain.vote.entity;
+
+import jakarta.persistence.Column;
+import jakarta.persistence.DiscriminatorColumn;
+import jakarta.persistence.Entity;
+import jakarta.persistence.FetchType;
+import jakarta.persistence.GeneratedValue;
+import jakarta.persistence.GenerationType;
+import jakarta.persistence.Id;
+import jakarta.persistence.Inheritance;
+import jakarta.persistence.InheritanceType;
+import jakarta.persistence.JoinColumn;
+import jakarta.persistence.ManyToOne;
+import lombok.AccessLevel;
+import lombok.Getter;
+import lombok.NoArgsConstructor;
+import photoground.ceos.vote.domain.member.entity.Member;
+import photoground.ceos.vote.global.entity.BaseTimeEntity;
+
+@Entity
+@NoArgsConstructor(access = AccessLevel.PROTECTED)
+@Getter
+@Inheritance(strategy = InheritanceType.SINGLE_TABLE)
+@DiscriminatorColumn(name = "vote_type")
+public class Vote extends BaseTimeEntity {
+
+ @Id
+ @GeneratedValue(strategy = GenerationType.IDENTITY)
+ @Column(name = "vote_id")
+ private Long id;
+
+ @ManyToOne(fetch = FetchType.LAZY)
+ @JoinColumn(name = "voter_id")
+ private Member voter;
+}
diff --git a/vote/src/main/java/photoground/ceos/vote/test/TestController.java b/vote/src/main/java/photoground/ceos/vote/test/TestController.java
new file mode 100644
index 0000000..e8fb7db
--- /dev/null
+++ b/vote/src/main/java/photoground/ceos/vote/test/TestController.java
@@ -0,0 +1,14 @@
+package photoground.ceos.vote.test;
+
+import org.springframework.http.ResponseEntity;
+import org.springframework.web.bind.annotation.GetMapping;
+import org.springframework.web.bind.annotation.RestController;
+
+@RestController
+public class TestController {
+
+ @GetMapping("/test")
+ public ResponseEntity test(){
+ return ResponseEntity.ok("Hello World");
+ }
+}
From 0a133eec891f52a090fd861444876d5037492a92 Mon Sep 17 00:00:00 2001
From: sh0311 <111671498+sh0311@users.noreply.github.com>
Date: Sun, 29 Dec 2024 12:43:53 +0900
Subject: [PATCH 03/27] Create cicd.yml
---
.github/workflows/cicd.yml | 57 ++++++++++++++++++++++++++++++++++++++
1 file changed, 57 insertions(+)
create mode 100644 .github/workflows/cicd.yml
diff --git a/.github/workflows/cicd.yml b/.github/workflows/cicd.yml
new file mode 100644
index 0000000..2f88ee9
--- /dev/null
+++ b/.github/workflows/cicd.yml
@@ -0,0 +1,57 @@
+name: CICD
+
+on:
+ push:
+ branches: [ "master" ]
+ pull_request:
+ branches: [ "master" ]
+
+permissions:
+ contents: read
+
+jobs:
+ build:
+ runs-on: ubuntu-latest
+ steps:
+ - uses: actions/checkout@v3 #github actions의 탬플릿(자동으로 pull받음)
+ - name: Install JDK 17
+ uses: actions/setup-java@v3
+ with:
+ java-version: '17'
+ distribution: 'temurin'
+
+ - name: Build with Gradle
+ run: |
+ mkdir -p ./src/main/resources
+ echo ${{ secrets.APPLICATION }} | base64 --decode > ./src/main/resources/application.yml
+ echo ${{ secrets.APPLICATION_PROD }} | base64 --decode > ./src/main/resources/application-prod.yml
+ echo ${{ secrets.APPLICATION_LOCAL }} | base64 --decode > ./src/main/resources/application-local.yml
+ chmod 777 ./gradlew
+ ./gradlew clean build -x test #jar파일 생성
+
+ - name: Login to DockerHub
+ uses: docker/login-action@v1
+ with:
+ username: ${{ secrets.DOCKERHUB_USERNAME }}
+ password: ${{ secrets.DOCKERHUB_TOKEN }}
+
+ - name: Build Docker (create Docker image)
+ run: docker build --platform linux/amd64 -t ${{ secrets.DOCKERHUB_USERNAME }}/ceos_vote .
+ - name: Push Docker
+ run: docker push ${{ secrets.DOCKERHUB_USERNAME }}/ceos_vote:latest
+
+ deploy:
+ needs: build
+ runs-on: ubuntu-latest
+ steps:
+ - name: Docker compose #ec2로 이동해서 ec2에 접속
+ uses: appleboy/ssh-action@master
+ with:
+ username: ubuntu
+ host: ${{ secrets.VOTE_SERVER_IP }}
+ key: ${{ secrets.EC2_SSH_KEY }}
+ script: |
+ sudo docker pull ${{ secrets.DOCKERHUB_USERNAME }}/ceos_vote:latest
+ sudo docker ps -q --filter "name=ceos_vote" | grep -q . && sudo docker stop ceos_vote && sudo docker rm ceos_vote || true
+ sudo docker run -d -p 8080:8080 ${{ secrets.DOCKER_USERNAME }}/ceos_vote:latest
+
From e7ed737a33a5ffe42a884454004dc9ed5a1f0138 Mon Sep 17 00:00:00 2001
From: sh0311 <111671498+sh0311@users.noreply.github.com>
Date: Sun, 29 Dec 2024 14:07:00 +0900
Subject: [PATCH 04/27] Update cicd.yml
---
.github/workflows/cicd.yml | 7 +++++++
1 file changed, 7 insertions(+)
diff --git a/.github/workflows/cicd.yml b/.github/workflows/cicd.yml
index 2f88ee9..7412b35 100644
--- a/.github/workflows/cicd.yml
+++ b/.github/workflows/cicd.yml
@@ -20,6 +20,13 @@ jobs:
java-version: '17'
distribution: 'temurin'
+ - name: Verify files and directories
+ run: |
+ echo "Current directory:"
+ pwd
+ echo "Files in current directory:"
+ ls -al
+
- name: Build with Gradle
run: |
mkdir -p ./src/main/resources
From b45f9aab12a731a1df09b6d8f9ae9124f4655161 Mon Sep 17 00:00:00 2001
From: sh0311 <0311alice@naver.com>
Date: Sun, 29 Dec 2024 14:10:07 +0900
Subject: [PATCH 05/27] =?UTF-8?q?feat=20:=20Dockerfile=20=EC=83=9D?=
=?UTF-8?q?=EC=84=B1?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
Dockerfile | 13 +++++++++++++
vote/.gitignore | 2 +-
2 files changed, 14 insertions(+), 1 deletion(-)
create mode 100644 Dockerfile
diff --git a/Dockerfile b/Dockerfile
new file mode 100644
index 0000000..c82b85d
--- /dev/null
+++ b/Dockerfile
@@ -0,0 +1,13 @@
+# 1. Java 이미지를 기반으로 설정
+FROM openjdk:17-jdk
+
+ARG JAR_FILE=/build/libs/*.jar
+
+# 3. JAR 파일 복사
+COPY ${JAR_FILE} app.jar
+
+# 4. 환경 변수로 Spring Profile 설정 (기본값: local)
+ENV SPRING_PROFILES_ACTIVE=dev
+
+# 5. 애플리케이션 실행
+ENTRYPOINT ["java", "-Dspring.profiles.active=${SPRING_PROFILES_ACTIVE}", "-jar", "app.jar"]
\ No newline at end of file
diff --git a/vote/.gitignore b/vote/.gitignore
index ffd6cd6..2c9b6a7 100644
--- a/vote/.gitignore
+++ b/vote/.gitignore
@@ -1,7 +1,7 @@
/resources
application.yml
application-local.yml
-application-dev.yml
+application-prod.yml
From a17f0ee8174b69250f3a1596367fbc66842790cb Mon Sep 17 00:00:00 2001
From: sh0311 <0311alice@naver.com>
Date: Sun, 29 Dec 2024 14:16:36 +0900
Subject: [PATCH 06/27] =?UTF-8?q?=F0=9F=9A=9A=20move=20:=20=ED=8F=B4?=
=?UTF-8?q?=EB=8D=94=EA=B5=AC=EC=A1=B0=20=EB=B3=80=EA=B2=BD?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
.DS_Store | Bin 0 -> 6148 bytes
vote/.gitattributes => .gitattributes | 0
vote/.gitignore => .gitignore | 4 +--
.idea/modules/vote.main.iml | 4 +--
HELP.md | 25 ++++++++++++++++++
vote/build.gradle => build.gradle | 0
.../wrapper/gradle-wrapper.jar | Bin
.../wrapper/gradle-wrapper.properties | 0
vote/gradlew => gradlew | 0
vote/gradlew.bat => gradlew.bat | 0
vote/settings.gradle => settings.gradle | 0
.../ceos/vote/VoteApplication.java | 0
.../vote/domain/member/entity/Member.java | 0
.../ceos/vote/domain/member/entity/Part.java | 0
.../ceos/vote/domain/member/entity/Team.java | 0
.../vote/domain/vote/entity/LeaderVote.java | 0
.../vote/domain/vote/entity/TeamVote.java | 0
.../ceos/vote/domain/vote/entity/Vote.java | 0
.../vote/global/entity/BaseTimeEntity.java | 0
.../global/exception/ApiExceptionHandler.java | 0
.../global/exception/CustomException.java | 0
.../ceos/vote/global/exception/ErrorCode.java | 0
.../vote/global/exception/ErrorResponse.java | 0
.../ceos/vote/test/TestController.java | 0
.../ceos/vote/VoteApplicationTests.java | 0
25 files changed, 29 insertions(+), 4 deletions(-)
create mode 100644 .DS_Store
rename vote/.gitattributes => .gitattributes (100%)
rename vote/.gitignore => .gitignore (90%)
create mode 100644 HELP.md
rename vote/build.gradle => build.gradle (100%)
rename {vote/gradle => gradle}/wrapper/gradle-wrapper.jar (100%)
rename {vote/gradle => gradle}/wrapper/gradle-wrapper.properties (100%)
rename vote/gradlew => gradlew (100%)
rename vote/gradlew.bat => gradlew.bat (100%)
rename vote/settings.gradle => settings.gradle (100%)
rename {vote/src => src}/main/java/photoground/ceos/vote/VoteApplication.java (100%)
rename {vote/src => src}/main/java/photoground/ceos/vote/domain/member/entity/Member.java (100%)
rename {vote/src => src}/main/java/photoground/ceos/vote/domain/member/entity/Part.java (100%)
rename {vote/src => src}/main/java/photoground/ceos/vote/domain/member/entity/Team.java (100%)
rename {vote/src => src}/main/java/photoground/ceos/vote/domain/vote/entity/LeaderVote.java (100%)
rename {vote/src => src}/main/java/photoground/ceos/vote/domain/vote/entity/TeamVote.java (100%)
rename {vote/src => src}/main/java/photoground/ceos/vote/domain/vote/entity/Vote.java (100%)
rename {vote/src => src}/main/java/photoground/ceos/vote/global/entity/BaseTimeEntity.java (100%)
rename {vote/src => src}/main/java/photoground/ceos/vote/global/exception/ApiExceptionHandler.java (100%)
rename {vote/src => src}/main/java/photoground/ceos/vote/global/exception/CustomException.java (100%)
rename {vote/src => src}/main/java/photoground/ceos/vote/global/exception/ErrorCode.java (100%)
rename {vote/src => src}/main/java/photoground/ceos/vote/global/exception/ErrorResponse.java (100%)
rename {vote/src => src}/main/java/photoground/ceos/vote/test/TestController.java (100%)
rename {vote/src => src}/test/java/photoground/ceos/vote/VoteApplicationTests.java (100%)
diff --git a/.DS_Store b/.DS_Store
new file mode 100644
index 0000000000000000000000000000000000000000..48fc58a87a9cd448df31d1f37ab5522644433667
GIT binary patch
literal 6148
zcmeHK%SyvQ6g|^c`x5NZjgT)W_y=ieK|w+E2Q+Oh#cH7s+-C1T_zQlYD?RtlP?I!u
zDI)ilJ7+REXC`x|WHJC;e=|M!Ag@wNyvT4ov4JC7SU-R
Dp(H14CoZs*yYNy2%o^Qy|UpsM2>LtNsT{HIc<-ZD-#aYJ^G
z^S&aMDa50!;utHsv*vGxXFlQTCx_Loth!465~KVa)Tbq>@sILwddITWW*sQ_2d;p3j3K$1WJ@O92c`4CLT{vPmFK2rc
zcyYkgqnE>l!-oqayKq8rIy(E0lnxhrw9*_f2l5V7{bfh?|Ka}ge_mu)=72fyuN-i-
z!R?^WDTTdtXL7RFM$9{=gz!?2TL~+=6_YDl@d;Cd?U7W7alq6gTNw5spf*@x4*aSE
EUm!q|$p8QV
literal 0
HcmV?d00001
diff --git a/vote/.gitattributes b/.gitattributes
similarity index 100%
rename from vote/.gitattributes
rename to .gitattributes
diff --git a/vote/.gitignore b/.gitignore
similarity index 90%
rename from vote/.gitignore
rename to .gitignore
index 2c9b6a7..e6328b0 100644
--- a/vote/.gitignore
+++ b/.gitignore
@@ -5,10 +5,10 @@ application-prod.yml
-HELP.md
+vote/HELP.md
.gradle
build/
-!gradle/wrapper/gradle-wrapper.jar
+!../gradle/wrapper/gradle-wrapper.jar
!**/src/main/**/build/
!**/src/test/**/build/
diff --git a/.idea/modules/vote.main.iml b/.idea/modules/vote.main.iml
index cf84e60..afc1871 100644
--- a/.idea/modules/vote.main.iml
+++ b/.idea/modules/vote.main.iml
@@ -1,8 +1,8 @@
-
-
+
+
\ No newline at end of file
diff --git a/HELP.md b/HELP.md
new file mode 100644
index 0000000..1052f41
--- /dev/null
+++ b/HELP.md
@@ -0,0 +1,25 @@
+# Getting Started
+
+### Reference Documentation
+For further reference, please consider the following sections:
+
+* [Official Gradle documentation](https://docs.gradle.org)
+* [Spring Boot Gradle Plugin Reference Guide](https://docs.spring.io/spring-boot/3.4.1/gradle-plugin)
+* [Create an OCI image](https://docs.spring.io/spring-boot/3.4.1/gradle-plugin/packaging-oci-image.html)
+* [Spring Web](https://docs.spring.io/spring-boot/3.4.1/reference/web/servlet.html)
+* [Spring Data JPA](https://docs.spring.io/spring-boot/3.4.1/reference/data/sql.html#data.sql.jpa-and-spring-data)
+
+### Guides
+The following guides illustrate how to use some features concretely:
+
+* [Building a RESTful Web Service](https://spring.io/guides/gs/rest-service/)
+* [Serving Web Content with Spring MVC](https://spring.io/guides/gs/serving-web-content/)
+* [Building REST services with Spring](https://spring.io/guides/tutorials/rest/)
+* [Accessing Data with JPA](https://spring.io/guides/gs/accessing-data-jpa/)
+* [Accessing data with MySQL](https://spring.io/guides/gs/accessing-data-mysql/)
+
+### Additional Links
+These additional references should also help you:
+
+* [Gradle Build Scans – insights for your project's build](https://scans.gradle.com#gradle)
+
diff --git a/vote/build.gradle b/build.gradle
similarity index 100%
rename from vote/build.gradle
rename to build.gradle
diff --git a/vote/gradle/wrapper/gradle-wrapper.jar b/gradle/wrapper/gradle-wrapper.jar
similarity index 100%
rename from vote/gradle/wrapper/gradle-wrapper.jar
rename to gradle/wrapper/gradle-wrapper.jar
diff --git a/vote/gradle/wrapper/gradle-wrapper.properties b/gradle/wrapper/gradle-wrapper.properties
similarity index 100%
rename from vote/gradle/wrapper/gradle-wrapper.properties
rename to gradle/wrapper/gradle-wrapper.properties
diff --git a/vote/gradlew b/gradlew
similarity index 100%
rename from vote/gradlew
rename to gradlew
diff --git a/vote/gradlew.bat b/gradlew.bat
similarity index 100%
rename from vote/gradlew.bat
rename to gradlew.bat
diff --git a/vote/settings.gradle b/settings.gradle
similarity index 100%
rename from vote/settings.gradle
rename to settings.gradle
diff --git a/vote/src/main/java/photoground/ceos/vote/VoteApplication.java b/src/main/java/photoground/ceos/vote/VoteApplication.java
similarity index 100%
rename from vote/src/main/java/photoground/ceos/vote/VoteApplication.java
rename to src/main/java/photoground/ceos/vote/VoteApplication.java
diff --git a/vote/src/main/java/photoground/ceos/vote/domain/member/entity/Member.java b/src/main/java/photoground/ceos/vote/domain/member/entity/Member.java
similarity index 100%
rename from vote/src/main/java/photoground/ceos/vote/domain/member/entity/Member.java
rename to src/main/java/photoground/ceos/vote/domain/member/entity/Member.java
diff --git a/vote/src/main/java/photoground/ceos/vote/domain/member/entity/Part.java b/src/main/java/photoground/ceos/vote/domain/member/entity/Part.java
similarity index 100%
rename from vote/src/main/java/photoground/ceos/vote/domain/member/entity/Part.java
rename to src/main/java/photoground/ceos/vote/domain/member/entity/Part.java
diff --git a/vote/src/main/java/photoground/ceos/vote/domain/member/entity/Team.java b/src/main/java/photoground/ceos/vote/domain/member/entity/Team.java
similarity index 100%
rename from vote/src/main/java/photoground/ceos/vote/domain/member/entity/Team.java
rename to src/main/java/photoground/ceos/vote/domain/member/entity/Team.java
diff --git a/vote/src/main/java/photoground/ceos/vote/domain/vote/entity/LeaderVote.java b/src/main/java/photoground/ceos/vote/domain/vote/entity/LeaderVote.java
similarity index 100%
rename from vote/src/main/java/photoground/ceos/vote/domain/vote/entity/LeaderVote.java
rename to src/main/java/photoground/ceos/vote/domain/vote/entity/LeaderVote.java
diff --git a/vote/src/main/java/photoground/ceos/vote/domain/vote/entity/TeamVote.java b/src/main/java/photoground/ceos/vote/domain/vote/entity/TeamVote.java
similarity index 100%
rename from vote/src/main/java/photoground/ceos/vote/domain/vote/entity/TeamVote.java
rename to src/main/java/photoground/ceos/vote/domain/vote/entity/TeamVote.java
diff --git a/vote/src/main/java/photoground/ceos/vote/domain/vote/entity/Vote.java b/src/main/java/photoground/ceos/vote/domain/vote/entity/Vote.java
similarity index 100%
rename from vote/src/main/java/photoground/ceos/vote/domain/vote/entity/Vote.java
rename to src/main/java/photoground/ceos/vote/domain/vote/entity/Vote.java
diff --git a/vote/src/main/java/photoground/ceos/vote/global/entity/BaseTimeEntity.java b/src/main/java/photoground/ceos/vote/global/entity/BaseTimeEntity.java
similarity index 100%
rename from vote/src/main/java/photoground/ceos/vote/global/entity/BaseTimeEntity.java
rename to src/main/java/photoground/ceos/vote/global/entity/BaseTimeEntity.java
diff --git a/vote/src/main/java/photoground/ceos/vote/global/exception/ApiExceptionHandler.java b/src/main/java/photoground/ceos/vote/global/exception/ApiExceptionHandler.java
similarity index 100%
rename from vote/src/main/java/photoground/ceos/vote/global/exception/ApiExceptionHandler.java
rename to src/main/java/photoground/ceos/vote/global/exception/ApiExceptionHandler.java
diff --git a/vote/src/main/java/photoground/ceos/vote/global/exception/CustomException.java b/src/main/java/photoground/ceos/vote/global/exception/CustomException.java
similarity index 100%
rename from vote/src/main/java/photoground/ceos/vote/global/exception/CustomException.java
rename to src/main/java/photoground/ceos/vote/global/exception/CustomException.java
diff --git a/vote/src/main/java/photoground/ceos/vote/global/exception/ErrorCode.java b/src/main/java/photoground/ceos/vote/global/exception/ErrorCode.java
similarity index 100%
rename from vote/src/main/java/photoground/ceos/vote/global/exception/ErrorCode.java
rename to src/main/java/photoground/ceos/vote/global/exception/ErrorCode.java
diff --git a/vote/src/main/java/photoground/ceos/vote/global/exception/ErrorResponse.java b/src/main/java/photoground/ceos/vote/global/exception/ErrorResponse.java
similarity index 100%
rename from vote/src/main/java/photoground/ceos/vote/global/exception/ErrorResponse.java
rename to src/main/java/photoground/ceos/vote/global/exception/ErrorResponse.java
diff --git a/vote/src/main/java/photoground/ceos/vote/test/TestController.java b/src/main/java/photoground/ceos/vote/test/TestController.java
similarity index 100%
rename from vote/src/main/java/photoground/ceos/vote/test/TestController.java
rename to src/main/java/photoground/ceos/vote/test/TestController.java
diff --git a/vote/src/test/java/photoground/ceos/vote/VoteApplicationTests.java b/src/test/java/photoground/ceos/vote/VoteApplicationTests.java
similarity index 100%
rename from vote/src/test/java/photoground/ceos/vote/VoteApplicationTests.java
rename to src/test/java/photoground/ceos/vote/VoteApplicationTests.java
From c9f512b50df8f3de3c221cdd9b8de27277bfc49c Mon Sep 17 00:00:00 2001
From: sh0311 <111671498+sh0311@users.noreply.github.com>
Date: Sun, 29 Dec 2024 14:30:47 +0900
Subject: [PATCH 07/27] Update cicd.yml
---
.github/workflows/cicd.yml | 4 ++--
1 file changed, 2 insertions(+), 2 deletions(-)
diff --git a/.github/workflows/cicd.yml b/.github/workflows/cicd.yml
index 7412b35..254ef02 100644
--- a/.github/workflows/cicd.yml
+++ b/.github/workflows/cicd.yml
@@ -58,7 +58,7 @@ jobs:
host: ${{ secrets.VOTE_SERVER_IP }}
key: ${{ secrets.EC2_SSH_KEY }}
script: |
- sudo docker pull ${{ secrets.DOCKERHUB_USERNAME }}/ceos_vote:latest
+ sudo docker pull ${{ secrets.DOCKERHUB_USERNAME }}/ceos_vote
sudo docker ps -q --filter "name=ceos_vote" | grep -q . && sudo docker stop ceos_vote && sudo docker rm ceos_vote || true
- sudo docker run -d -p 8080:8080 ${{ secrets.DOCKER_USERNAME }}/ceos_vote:latest
+ sudo docker run -d -p 8080:8080 ${{ secrets.DOCKER_USERNAME }}/ceos_vote
From 3c5e4199964b228aa9e5d552ed1dc30e2226ec56 Mon Sep 17 00:00:00 2001
From: sh0311 <111671498+sh0311@users.noreply.github.com>
Date: Sun, 29 Dec 2024 14:42:01 +0900
Subject: [PATCH 08/27] Update cicd.yml
---
.github/workflows/cicd.yml | 12 ++++++++----
1 file changed, 8 insertions(+), 4 deletions(-)
diff --git a/.github/workflows/cicd.yml b/.github/workflows/cicd.yml
index 254ef02..364065c 100644
--- a/.github/workflows/cicd.yml
+++ b/.github/workflows/cicd.yml
@@ -43,7 +43,7 @@ jobs:
password: ${{ secrets.DOCKERHUB_TOKEN }}
- name: Build Docker (create Docker image)
- run: docker build --platform linux/amd64 -t ${{ secrets.DOCKERHUB_USERNAME }}/ceos_vote .
+ run: docker build --platform linux/amd64 -t ${{ secrets.DOCKERHUB_USERNAME }}/ceos_vote:latest .
- name: Push Docker
run: docker push ${{ secrets.DOCKERHUB_USERNAME }}/ceos_vote:latest
@@ -58,7 +58,11 @@ jobs:
host: ${{ secrets.VOTE_SERVER_IP }}
key: ${{ secrets.EC2_SSH_KEY }}
script: |
- sudo docker pull ${{ secrets.DOCKERHUB_USERNAME }}/ceos_vote
- sudo docker ps -q --filter "name=ceos_vote" | grep -q . && sudo docker stop ceos_vote && sudo docker rm ceos_vote || true
- sudo docker run -d -p 8080:8080 ${{ secrets.DOCKER_USERNAME }}/ceos_vote
+ sudo docker pull ${{ secrets.DOCKERHUB_USERNAME }}/ceos_vote:latest
+ if [ $(sudo docker ps -q --filter "name=ceos_vote") ]; then
+ echo "Stopping and removing existing container..."
+ sudo docker stop ceos_vote
+ sudo docker rm ceos_vote
+ fi
+ sudo docker run -d -p 8080:8080 ${{ secrets.DOCKERHUB_USERNAME }}/ceos_vote:latest
From 154e7f0fcebee7508ba1361b433268da82cbe888 Mon Sep 17 00:00:00 2001
From: sh0311 <111671498+sh0311@users.noreply.github.com>
Date: Sun, 29 Dec 2024 15:41:31 +0900
Subject: [PATCH 09/27] Update Dockerfile
---
Dockerfile | 4 ++--
1 file changed, 2 insertions(+), 2 deletions(-)
diff --git a/Dockerfile b/Dockerfile
index c82b85d..40be9f8 100644
--- a/Dockerfile
+++ b/Dockerfile
@@ -7,7 +7,7 @@ ARG JAR_FILE=/build/libs/*.jar
COPY ${JAR_FILE} app.jar
# 4. 환경 변수로 Spring Profile 설정 (기본값: local)
-ENV SPRING_PROFILES_ACTIVE=dev
+ENV SPRING_PROFILES_ACTIVE=prod
# 5. 애플리케이션 실행
-ENTRYPOINT ["java", "-Dspring.profiles.active=${SPRING_PROFILES_ACTIVE}", "-jar", "app.jar"]
\ No newline at end of file
+ENTRYPOINT ["java", "-Dspring.profiles.active=${SPRING_PROFILES_ACTIVE}", "-jar", "app.jar"]
From 7af69c6eec1cf11cdddf0ed320c8f8baa503636e Mon Sep 17 00:00:00 2001
From: sh0311 <0311alice@naver.com>
Date: Mon, 30 Dec 2024 15:23:45 +0900
Subject: [PATCH 10/27] =?UTF-8?q?=E2=9C=A8=20feat=20:=20=ED=9B=84=EB=B3=B4?=
=?UTF-8?q?=20=EC=A1=B0=ED=9A=8C?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
.idea/compiler.xml | 2 +-
.idea/gradle.xml | 9 ++++
.idea/modules.xml | 2 +-
.../vote/domain/member/entity/Member.java | 2 +-
.../ceos/vote/domain/member/entity/Team.java | 2 +
.../member/repository/MemberRepository.java | 10 +++++
.../domain/member/service/MemberService.java | 21 ++++++++++
.../vote/controller/VoteController.java | 36 ++++++++++++++++
.../ceos/vote/domain/vote/dto/LeaderDTO.java | 22 ++++++++++
.../vote/domain/vote/dto/LeaderListDTO.java | 18 ++++++++
.../vote/domain/vote/dto/TeamListDTO.java | 19 +++++++++
.../vote/repository/VoteRepository.java | 7 ++++
.../vote/domain/vote/service/VoteService.java | 42 +++++++++++++++++++
13 files changed, 189 insertions(+), 3 deletions(-)
create mode 100644 src/main/java/photoground/ceos/vote/domain/member/repository/MemberRepository.java
create mode 100644 src/main/java/photoground/ceos/vote/domain/member/service/MemberService.java
create mode 100644 src/main/java/photoground/ceos/vote/domain/vote/controller/VoteController.java
create mode 100644 src/main/java/photoground/ceos/vote/domain/vote/dto/LeaderDTO.java
create mode 100644 src/main/java/photoground/ceos/vote/domain/vote/dto/LeaderListDTO.java
create mode 100644 src/main/java/photoground/ceos/vote/domain/vote/dto/TeamListDTO.java
create mode 100644 src/main/java/photoground/ceos/vote/domain/vote/repository/VoteRepository.java
create mode 100644 src/main/java/photoground/ceos/vote/domain/vote/service/VoteService.java
diff --git a/.idea/compiler.xml b/.idea/compiler.xml
index 1d3def4..8c129a5 100644
--- a/.idea/compiler.xml
+++ b/.idea/compiler.xml
@@ -8,7 +8,7 @@
-
+
diff --git a/.idea/gradle.xml b/.idea/gradle.xml
index 01d672b..4fcec57 100644
--- a/.idea/gradle.xml
+++ b/.idea/gradle.xml
@@ -3,6 +3,15 @@