From cc07b8fbeab7df6d7d26bdef4701755b61fb0f29 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Kisv=C3=A1ri=20Benedek?= <446-kisbe32@users.noreply.dev.itk.ppke.hu> Date: Tue, 8 Apr 2025 11:47:41 +0200 Subject: [PATCH] adding everything --- backend/.gitattributes | 3 + backend/.gitignore | 37 +++ backend/Dockerfile | 10 + backend/build.gradle | 32 +++ backend/gradle/wrapper/gradle-wrapper.jar | Bin 0 -> 43705 bytes .../gradle/wrapper/gradle-wrapper.properties | 7 + backend/gradlew | 251 ++++++++++++++++++ backend/gradlew.bat | 94 +++++++ backend/settings.gradle | 1 + .../java/kisbe32/backend/AuthController.java | 30 +++ .../kisbe32/backend/BackendApplication.java | 27 ++ .../java/kisbe32/backend/LoginRequest.java | 41 +++ .../kisbe32/backend/OrderItemRepository.java | 16 ++ .../java/kisbe32/backend/OrderRepository.java | 16 ++ .../kisbe32/backend/ProductRepository.java | 19 ++ .../src/main/java/kisbe32/backend/User.java | 33 ++- .../java/kisbe32/backend/UserRepository.java | 9 + .../main/java/kisbe32/backend/WebConfig.java | 15 ++ .../src/main/resources/application.properties | 9 + .../kisbe32/backend/ApiIntegrationTest.java | 126 +++++++++ docker-compose.yml | 43 +++ frontend/.idea/vcs.xml | 1 + frontend/Dockerfile | 27 ++ frontend/src/components/Cart.vue | 3 +- frontend/src/components/Login.vue | 3 +- frontend/src/components/MainContent.vue | 3 +- frontend/src/components/Orders.vue | 3 +- frontend/src/components/Register.vue | 3 +- frontend/src/config/api.js | 1 + frontend/vite.config.js | 5 +- 30 files changed, 843 insertions(+), 25 deletions(-) create mode 100644 backend/.gitattributes create mode 100644 backend/.gitignore create mode 100644 backend/Dockerfile create mode 100644 backend/build.gradle create mode 100644 backend/gradle/wrapper/gradle-wrapper.jar create mode 100644 backend/gradle/wrapper/gradle-wrapper.properties create mode 100755 backend/gradlew create mode 100644 backend/gradlew.bat create mode 100644 backend/settings.gradle create mode 100644 backend/src/main/java/kisbe32/backend/AuthController.java create mode 100644 backend/src/main/java/kisbe32/backend/BackendApplication.java create mode 100644 backend/src/main/java/kisbe32/backend/LoginRequest.java create mode 100644 backend/src/main/java/kisbe32/backend/OrderItemRepository.java create mode 100644 backend/src/main/java/kisbe32/backend/OrderRepository.java create mode 100644 backend/src/main/java/kisbe32/backend/ProductRepository.java create mode 100644 backend/src/main/java/kisbe32/backend/UserRepository.java create mode 100644 backend/src/main/java/kisbe32/backend/WebConfig.java create mode 100644 backend/src/main/resources/application.properties create mode 100644 backend/src/test/java/kisbe32/backend/ApiIntegrationTest.java create mode 100644 docker-compose.yml create mode 100644 frontend/Dockerfile create mode 100644 frontend/src/config/api.js diff --git a/backend/.gitattributes b/backend/.gitattributes new file mode 100644 index 0000000..8af972c --- /dev/null +++ b/backend/.gitattributes @@ -0,0 +1,3 @@ +/gradlew text eol=lf +*.bat text eol=crlf +*.jar binary diff --git a/backend/.gitignore b/backend/.gitignore new file mode 100644 index 0000000..c2065bc --- /dev/null +++ b/backend/.gitignore @@ -0,0 +1,37 @@ +HELP.md +.gradle +build/ +!gradle/wrapper/gradle-wrapper.jar +!**/src/main/**/build/ +!**/src/test/**/build/ + +### STS ### +.apt_generated +.classpath +.factorypath +.project +.settings +.springBeans +.sts4-cache +bin/ +!**/src/main/**/bin/ +!**/src/test/**/bin/ + +### IntelliJ IDEA ### +.idea +*.iws +*.iml +*.ipr +out/ +!**/src/main/**/out/ +!**/src/test/**/out/ + +### NetBeans ### +/nbproject/private/ +/nbbuild/ +/dist/ +/nbdist/ +/.nb-gradle/ + +### VS Code ### +.vscode/ diff --git a/backend/Dockerfile b/backend/Dockerfile new file mode 100644 index 0000000..e6907e4 --- /dev/null +++ b/backend/Dockerfile @@ -0,0 +1,10 @@ +FROM eclipse-temurin:17-jdk AS build +WORKDIR /app +COPY . . +RUN ./gradlew bootJar --no-daemon + +FROM eclipse-temurin:17-jre +WORKDIR /app +COPY --from=build /app/build/libs/*.jar app.jar +EXPOSE 8080 +ENTRYPOINT ["java", "-jar", "app.jar"] diff --git a/backend/build.gradle b/backend/build.gradle new file mode 100644 index 0000000..deebb25 --- /dev/null +++ b/backend/build.gradle @@ -0,0 +1,32 @@ +plugins { + id 'java' + id 'org.springframework.boot' version '3.4.4' + id 'io.spring.dependency-management' version '1.1.7' +} + +group = 'kisbe32' +version = '0.0.1-SNAPSHOT' + +java { + toolchain { + languageVersion = JavaLanguageVersion.of(17) + } +} + +repositories { + mavenCentral() +} + +dependencies { + implementation 'org.springframework.boot:spring-boot-starter-data-jpa' + implementation 'org.springframework.boot:spring-boot-starter-web' + implementation 'org.springframework.boot:spring-boot-starter-data-rest' + implementation 'org.springframework.boot:spring-boot-starter-actuator' + runtimeOnly 'org.postgresql:postgresql' + testImplementation 'org.springframework.boot:spring-boot-starter-test' + testRuntimeOnly 'org.junit.platform:junit-platform-launcher' +} + +tasks.named('test') { + useJUnitPlatform() +} diff --git a/backend/gradle/wrapper/gradle-wrapper.jar b/backend/gradle/wrapper/gradle-wrapper.jar new file mode 100644 index 0000000000000000000000000000000000000000..9bbc975c742b298b441bfb90dbc124400a3751b9 GIT binary patch literal 43705 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 a;23lZ@MdKLX;x%VWDwwGV3<4?!~*~~4acnj literal 0 HcmV?d00001 diff --git a/backend/gradle/wrapper/gradle-wrapper.properties b/backend/gradle/wrapper/gradle-wrapper.properties new file mode 100644 index 0000000..37f853b --- /dev/null +++ b/backend/gradle/wrapper/gradle-wrapper.properties @@ -0,0 +1,7 @@ +distributionBase=GRADLE_USER_HOME +distributionPath=wrapper/dists +distributionUrl=https\://services.gradle.org/distributions/gradle-8.13-bin.zip +networkTimeout=10000 +validateDistributionUrl=true +zipStoreBase=GRADLE_USER_HOME +zipStorePath=wrapper/dists diff --git a/backend/gradlew b/backend/gradlew new file mode 100755 index 0000000..faf9300 --- /dev/null +++ b/backend/gradlew @@ -0,0 +1,251 @@ +#!/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\n' "$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, 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/backend/gradlew.bat b/backend/gradlew.bat new file mode 100644 index 0000000..9d21a21 --- /dev/null +++ b/backend/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/backend/settings.gradle b/backend/settings.gradle new file mode 100644 index 0000000..0f5036d --- /dev/null +++ b/backend/settings.gradle @@ -0,0 +1 @@ +rootProject.name = 'backend' diff --git a/backend/src/main/java/kisbe32/backend/AuthController.java b/backend/src/main/java/kisbe32/backend/AuthController.java new file mode 100644 index 0000000..e396a83 --- /dev/null +++ b/backend/src/main/java/kisbe32/backend/AuthController.java @@ -0,0 +1,30 @@ +package kisbe32.backend; + +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.http.HttpStatus; +import org.springframework.http.ResponseEntity; +import org.springframework.web.bind.annotation.CrossOrigin; +import org.springframework.web.bind.annotation.PostMapping; +import org.springframework.web.bind.annotation.RequestBody; +import org.springframework.web.bind.annotation.RestController; + +@RestController +@CrossOrigin // Enable CORS for all origins +public class AuthController { + + @Autowired + private UserRepository userRepository; + + @PostMapping("/login") + public ResponseEntity<LoginResponse> login(@RequestBody LoginRequest request) { + User user = userRepository.findByUsername(request.getUsername()); + + // Simple string comparison for password + if (user != null && user.getPassword().equals(request.getPassword())) { + return ResponseEntity.ok(new LoginResponse(true, user.getId(), user.getUsername())); + } else { + return ResponseEntity.status(HttpStatus.UNAUTHORIZED) + .body(new LoginResponse(false, null, null)); + } + } +} \ No newline at end of file diff --git a/backend/src/main/java/kisbe32/backend/BackendApplication.java b/backend/src/main/java/kisbe32/backend/BackendApplication.java new file mode 100644 index 0000000..89448d8 --- /dev/null +++ b/backend/src/main/java/kisbe32/backend/BackendApplication.java @@ -0,0 +1,27 @@ +package kisbe32.backend; + +import org.springframework.boot.SpringApplication; +import org.springframework.boot.autoconfigure.SpringBootApplication; +import org.springframework.data.rest.core.config.RepositoryRestConfiguration; +import org.springframework.data.rest.webmvc.config.RepositoryRestConfigurer; +import org.springframework.stereotype.Component; +import org.springframework.web.servlet.config.annotation.CorsRegistry; +import org.springframework.context.annotation.Configuration; +import org.springframework.web.servlet.config.annotation.WebMvcConfigurer; + +@SpringBootApplication +public class BackendApplication { + public static void main(String[] args) { + SpringApplication.run(BackendApplication.class, args); + } +} + +// Optional: Configure the REST API to expose IDs +@Component +class RestConfig implements RepositoryRestConfigurer { + @Override + public void configureRepositoryRestConfiguration(RepositoryRestConfiguration config, CorsRegistry cors) { + // Expose entity IDs in REST responses + config.exposeIdsFor(User.class, Product.class, Order.class, OrderItem.class); + } +} \ No newline at end of file diff --git a/backend/src/main/java/kisbe32/backend/LoginRequest.java b/backend/src/main/java/kisbe32/backend/LoginRequest.java new file mode 100644 index 0000000..26aa511 --- /dev/null +++ b/backend/src/main/java/kisbe32/backend/LoginRequest.java @@ -0,0 +1,41 @@ +package kisbe32.backend; + +class LoginRequest { + private String username; + private String password; + + // Default constructor needed for JSON deserialization + public LoginRequest() {} + + public LoginRequest(String username, String password) { + this.username = username; + this.password = password; + } + + public String getUsername() { return username; } + public void setUsername(String username) { this.username = username; } + + public String getPassword() { return password; } + public void setPassword(String password) { this.password = password; } +} + +class LoginResponse { + private boolean success; + private Integer userId; + private String username; + + public LoginResponse(boolean success, Integer userId, String username) { + this.success = success; + this.userId = userId; + this.username = username; + } + + public boolean isSuccess() { return success; } + public void setSuccess(boolean success) { this.success = success; } + + public Integer getUserId() { return userId; } + public void setUserId(Integer userId) { this.userId = userId; } + + public String getUsername() { return username; } + public void setUsername(String username) { this.username = username; } +} \ No newline at end of file diff --git a/backend/src/main/java/kisbe32/backend/OrderItemRepository.java b/backend/src/main/java/kisbe32/backend/OrderItemRepository.java new file mode 100644 index 0000000..1447349 --- /dev/null +++ b/backend/src/main/java/kisbe32/backend/OrderItemRepository.java @@ -0,0 +1,16 @@ +package kisbe32.backend; + +import org.springframework.data.jpa.repository.JpaRepository; +import org.springframework.data.repository.query.Param; +import org.springframework.data.rest.core.annotation.RepositoryRestResource; + +import java.util.List; + +@RepositoryRestResource(collectionResourceRel = "orderitems", path = "orderitems") +interface OrderItemRepository extends JpaRepository<OrderItem, Integer> { + // Find order items by order ID + List<OrderItem> findByOrderId(@Param("orderId") Integer orderId); + + // Find order items by product ID + List<OrderItem> findByProductId(@Param("productId") Integer productId); +} diff --git a/backend/src/main/java/kisbe32/backend/OrderRepository.java b/backend/src/main/java/kisbe32/backend/OrderRepository.java new file mode 100644 index 0000000..3ed2633 --- /dev/null +++ b/backend/src/main/java/kisbe32/backend/OrderRepository.java @@ -0,0 +1,16 @@ +package kisbe32.backend; + +import org.springframework.data.jpa.repository.JpaRepository; +import org.springframework.data.repository.query.Param; +import org.springframework.data.rest.core.annotation.RepositoryRestResource; + +import java.util.List; + +@RepositoryRestResource(collectionResourceRel = "orders", path = "orders") +interface OrderRepository extends JpaRepository<Order, Integer> { + // Find orders by user ID + List<Order> findByUserId(@Param("userId") Integer userId); + + // Find orders by status + List<Order> findByStatus(@Param("status") String status); +} diff --git a/backend/src/main/java/kisbe32/backend/ProductRepository.java b/backend/src/main/java/kisbe32/backend/ProductRepository.java new file mode 100644 index 0000000..2494bc1 --- /dev/null +++ b/backend/src/main/java/kisbe32/backend/ProductRepository.java @@ -0,0 +1,19 @@ +package kisbe32.backend; + +import org.springframework.data.jpa.repository.JpaRepository; +import org.springframework.data.repository.query.Param; +import org.springframework.data.rest.core.annotation.RepositoryRestResource; + +import java.util.List; + +@RepositoryRestResource(collectionResourceRel = "products", path = "products") +interface ProductRepository extends JpaRepository<Product, Integer> { + // Find products by category + List<Product> findByCategory(@Param("category") String category); + + // Find products with stock greater than a given amount + List<Product> findByStockGreaterThan(@Param("stock") Integer stock); + + // Find products within a price range + List<Product> findByPriceBetween(@Param("min") java.math.BigDecimal min, @Param("max") java.math.BigDecimal max); +} diff --git a/backend/src/main/java/kisbe32/backend/User.java b/backend/src/main/java/kisbe32/backend/User.java index b2fbba2..5d0139f 100644 --- a/backend/src/main/java/kisbe32/backend/User.java +++ b/backend/src/main/java/kisbe32/backend/User.java @@ -1,41 +1,40 @@ package kisbe32.backend; -import jakarta.persistence.*; -import java.time.LocalDateTime; +import jakarta.persistence.Entity; +import jakarta.persistence.GeneratedValue; +import jakarta.persistence.GenerationType; +import jakarta.persistence.Id; +import jakarta.persistence.Table; @Entity @Table(name = "users") public class User { @Id - @GeneratedValue(strategy = GenerationType.SEQUENCE) - @Column(name = "id", nullable = false) + @GeneratedValue(strategy = GenerationType.IDENTITY) private Integer id; - - @Column(name = "username", nullable = false, length = 50, unique = true) private String username; - - @Column(name = "email", nullable = false, length = 100, unique = true) private String email; - - @Column(name = "password", nullable = false, length = 255) private String password; - @Column(name = "created_at") - private LocalDateTime createdAt; - - protected User() {} + // Default constructor required by JPA + public User() {} public User(String username, String email, String password) { this.username = username; this.email = email; this.password = password; - this.createdAt = LocalDateTime.now(); } - // Getters + // Getters and setters public Integer getId() { return id; } + public void setId(Integer id) { this.id = id; } + public String getUsername() { return username; } + public void setUsername(String username) { this.username = username; } + public String getEmail() { return email; } + public void setEmail(String email) { this.email = email; } + public String getPassword() { return password; } - public LocalDateTime getCreatedAt() { return createdAt; } + public void setPassword(String password) { this.password = password; } } \ No newline at end of file diff --git a/backend/src/main/java/kisbe32/backend/UserRepository.java b/backend/src/main/java/kisbe32/backend/UserRepository.java new file mode 100644 index 0000000..11146df --- /dev/null +++ b/backend/src/main/java/kisbe32/backend/UserRepository.java @@ -0,0 +1,9 @@ +package kisbe32.backend; + +import org.springframework.data.jpa.repository.JpaRepository; +import org.springframework.stereotype.Repository; + +@Repository +public interface UserRepository extends JpaRepository<User, Integer> { + User findByUsername(String username); +} \ No newline at end of file diff --git a/backend/src/main/java/kisbe32/backend/WebConfig.java b/backend/src/main/java/kisbe32/backend/WebConfig.java new file mode 100644 index 0000000..826898c --- /dev/null +++ b/backend/src/main/java/kisbe32/backend/WebConfig.java @@ -0,0 +1,15 @@ +package kisbe32.backend; + +import org.springframework.context.annotation.Configuration; +import org.springframework.web.servlet.config.annotation.CorsRegistry; +import org.springframework.web.servlet.config.annotation.WebMvcConfigurer; + +@Configuration +public class WebConfig implements WebMvcConfigurer { + @Override + public void addCorsMappings(CorsRegistry registry) { + registry.addMapping("/**") // Allow all endpoints + .allowedOrigins("*") + .allowedMethods("GET", "POST", "PUT", "DELETE", "OPTIONS"); + } +} \ No newline at end of file diff --git a/backend/src/main/resources/application.properties b/backend/src/main/resources/application.properties new file mode 100644 index 0000000..f975b96 --- /dev/null +++ b/backend/src/main/resources/application.properties @@ -0,0 +1,9 @@ +spring.application.name=backend + +spring.datasource.url=jdbc:postgresql://jokaiter.duckdns.org:34821/szofttech_project +spring.datasource.username=szofttech_project +spring.datasource.password=Utopia_Gown_Duo4 +spring.jpa.hibernate.ddl-auto=update +spring.jpa.properties.hibernate.dialect=org.hibernate.dialect.PostgreSQLDialect + +spring.data.rest.base-path=/api \ No newline at end of file diff --git a/backend/src/test/java/kisbe32/backend/ApiIntegrationTest.java b/backend/src/test/java/kisbe32/backend/ApiIntegrationTest.java new file mode 100644 index 0000000..6003157 --- /dev/null +++ b/backend/src/test/java/kisbe32/backend/ApiIntegrationTest.java @@ -0,0 +1,126 @@ +package kisbe32.backend; + +import org.junit.jupiter.api.Test; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.boot.test.context.SpringBootTest; +import org.springframework.boot.test.web.client.TestRestTemplate; +import org.springframework.core.ParameterizedTypeReference; +import org.springframework.http.*; +import org.springframework.test.context.ActiveProfiles; + +import java.math.BigDecimal; +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +import static org.assertj.core.api.Assertions.assertThat; + +@SpringBootTest(webEnvironment = SpringBootTest.WebEnvironment.RANDOM_PORT) +@ActiveProfiles("test") +public class ApiIntegrationTest { + + @Autowired + private TestRestTemplate restTemplate; + + @Autowired + private ProductRepository productRepository; + + @Autowired + private UserRepository userRepository; + + @Autowired + private OrderRepository orderRepository; + + @Autowired + private OrderItemRepository orderItemRepository; + + @Test + public void testGetProducts() { + // Given: Add test products + Product product1 = new Product("Product 1", "Description 1", new BigDecimal("10.99"), 5, "Category 1"); + Product product2 = new Product("Product 2", "Description 2", new BigDecimal("15.99"), 10, "Category 2"); + productRepository.saveAll(List.of(product1, product2)); + + // When: Request is made to get all products + ResponseEntity<Map<String, Object>> response = restTemplate.exchange( + "/api/products", + HttpMethod.GET, + null, + new ParameterizedTypeReference<Map<String, Object>>() {} + ); + + // Then: The response should contain products + assertThat(response.getStatusCode()).isEqualTo(HttpStatus.OK); + assertThat(response.getBody()).isNotNull(); + List<Map<String, Object>> products = (List<Map<String, Object>>) response.getBody().get("_embedded"); + assertThat(products).isNotEmpty(); + } + + @Test + public void testCreateOrderAndOrderItems() { + // Given: A user and products exist + User user = new User("testuser", "test@example.com", "password"); + userRepository.save(user); + + Product product = new Product("Test Product", "Description", new BigDecimal("19.99"), 20, "Test Category"); + productRepository.save(product); + + // When: Create an order + Map<String, Object> orderRequest = new HashMap<>(); + orderRequest.put("userId", user.getId()); + orderRequest.put("status", "PENDING"); + + ResponseEntity<Map> orderResponse = restTemplate.postForEntity( + "/api/orders", orderRequest, Map.class); + + // Then: The order should be created + assertThat(orderResponse.getStatusCode()).isEqualTo(HttpStatus.CREATED); + String orderLocation = orderResponse.getHeaders().getLocation().toString(); + + // When: Create an order item + Map<String, Object> orderItemRequest = new HashMap<>(); + orderItemRequest.put("orderId", getIdFromLocation(orderLocation)); + orderItemRequest.put("productId", product.getId()); + orderItemRequest.put("quantity", 2); + orderItemRequest.put("price", product.getPrice()); + + ResponseEntity<Map> orderItemResponse = restTemplate.postForEntity( + "/api/orderitems", orderItemRequest, Map.class); + + // Then: The order item should be created + assertThat(orderItemResponse.getStatusCode()).isEqualTo(HttpStatus.CREATED); + } + + @Test + public void testLogin() { + // Given: A user exists + String username = "loginuser"; + String password = "password123"; + User user = new User(username, "login@example.com", password); + userRepository.save(user); + + // When: Login with valid credentials + AuthController.LoginRequest loginRequest = new AuthController.LoginRequest(username, password); + + ResponseEntity<AuthController.LoginResponse> response = restTemplate.postForEntity( + "/api/login", loginRequest, AuthController.LoginResponse.class); + + // Then: Should get successful response + assertThat(response.getStatusCode()).isEqualTo(HttpStatus.OK); + assertThat(response.getBody().isSuccess()).isTrue(); + assertThat(response.getBody().getUserId()).isEqualTo(user.getId()); + + // When: Login with invalid password + loginRequest.setPassword("wrongpassword"); + ResponseEntity<AuthController.LoginResponse> failedResponse = restTemplate.postForEntity( + "/api/login", loginRequest, AuthController.LoginResponse.class); + + // Then: Should get unauthorized response + assertThat(failedResponse.getStatusCode()).isEqualTo(HttpStatus.UNAUTHORIZED); + assertThat(failedResponse.getBody().isSuccess()).isFalse(); + } + + private Integer getIdFromLocation(String location) { + return Integer.parseInt(location.substring(location.lastIndexOf("/") + 1)); + } +} \ No newline at end of file diff --git a/docker-compose.yml b/docker-compose.yml new file mode 100644 index 0000000..2ae14cb --- /dev/null +++ b/docker-compose.yml @@ -0,0 +1,43 @@ +version: '3.8' +services: + backend: + build: ./backend + ports: + - "8080:8080" + environment: + - SPRING_DATASOURCE_URL=jdbc:postgresql://db:5432/szofttech_project + - SPRING_DATASOURCE_USERNAME=szofttech_project + - SPRING_DATASOURCE_PASSWORD=Utopia_Gown_Duo4 + - SPRING_JPA_HIBERNATE_DDL_AUTO=update + depends_on: + - db + networks: + - app-network + + frontend: + build: ./frontend + ports: + - "80:80" + depends_on: + - backend + networks: + - app-network + + db: + image: postgres:17 + ports: + - "5432:5432" + environment: + - POSTGRES_DB=szofttech_project + - POSTGRES_USER=szofttech_project + - POSTGRES_PASSWORD=Utopia_Gown_Duo4 + volumes: + - postgres-data:/var/lib/postgresql/data + networks: + - app-network + +networks: + app-network: + +volumes: + postgres-data: diff --git a/frontend/.idea/vcs.xml b/frontend/.idea/vcs.xml index 94a25f7..288b36b 100644 --- a/frontend/.idea/vcs.xml +++ b/frontend/.idea/vcs.xml @@ -1,6 +1,7 @@ <?xml version="1.0" encoding="UTF-8"?> <project version="4"> <component name="VcsDirectoryMappings"> + <mapping directory="$PROJECT_DIR$/.." vcs="Git" /> <mapping directory="$PROJECT_DIR$" vcs="Git" /> </component> </project> \ No newline at end of file diff --git a/frontend/Dockerfile b/frontend/Dockerfile new file mode 100644 index 0000000..c448e2f --- /dev/null +++ b/frontend/Dockerfile @@ -0,0 +1,27 @@ +# Build stage +FROM node:lts-alpine as build-stage +WORKDIR /app +COPY package*.json ./ +RUN npm install +COPY . . +RUN npm run build + +# Production stage +FROM nginx:stable-alpine as production-stage +COPY --from=build-stage /app/dist /usr/share/nginx/html +# Create nginx config to handle SPA routing and API proxying +RUN echo 'server { \ + listen 80; \ + location / { \ + root /usr/share/nginx/html; \ + index index.html; \ + try_files $uri $uri/ /index.html; \ + } \ + location /api/ { \ + proxy_pass http://backend:8080/api/; \ + proxy_set_header Host $host; \ + proxy_set_header X-Real-IP $remote_addr; \ + } \ +}' > /etc/nginx/conf.d/default.conf +EXPOSE 80 +CMD ["nginx", "-g", "daemon off;"] diff --git a/frontend/src/components/Cart.vue b/frontend/src/components/Cart.vue index 3ab48cc..27b4443 100644 --- a/frontend/src/components/Cart.vue +++ b/frontend/src/components/Cart.vue @@ -3,6 +3,7 @@ import { ref, computed, onMounted } from 'vue'; import cartStore from '../stores/cartStore.js'; import axios from 'axios'; import { defineEmits } from 'vue'; +import {API_BASE_URL} from "@/config/api.js"; const emit = defineEmits(['navigate']); @@ -59,7 +60,7 @@ async function checkout() { })) }; - const response = await axios.post('http://localhost:3000/orders', orderData, { + const response = await axios.post(`${API_BASE_URL}/orders`, orderData, { headers: { Authorization: `Bearer ${token}` } diff --git a/frontend/src/components/Login.vue b/frontend/src/components/Login.vue index 8445158..b99afe7 100644 --- a/frontend/src/components/Login.vue +++ b/frontend/src/components/Login.vue @@ -1,6 +1,7 @@ <script setup> import { defineEmits, ref } from 'vue' import axios from "axios"; +import { API_BASE_URL } from "@/config/api.js"; const emit = defineEmits(['navigate']) @@ -14,7 +15,7 @@ async function onLogin() { error.value = '' isLoading.value = true - const response = await axios.post('http://localhost:3000/login', { + const response = await axios.post(`${API_BASE_URL}/login`, { username: username.value, password: password.value }); diff --git a/frontend/src/components/MainContent.vue b/frontend/src/components/MainContent.vue index 0a8d242..ac91181 100644 --- a/frontend/src/components/MainContent.vue +++ b/frontend/src/components/MainContent.vue @@ -2,6 +2,7 @@ import { ref, computed, onMounted } from 'vue'; import axios from 'axios'; import cartStore from '../stores/cartStore.js' +import {API_BASE_URL} from "@/config/api.js"; function goToCart() { emit('navigate', 'cart') @@ -95,7 +96,7 @@ const fetchProducts = async () => { error.value = null; // Itt történik az API hívás a backend felé - const response = await axios.get('http://localhost:3000/products'); + const response = await axios.get(`${API_BASE_URL}/products`); // Az adatbázisból érkező adatok formázása products.value = response.data.map(product => ({ diff --git a/frontend/src/components/Orders.vue b/frontend/src/components/Orders.vue index 49b16eb..8413367 100644 --- a/frontend/src/components/Orders.vue +++ b/frontend/src/components/Orders.vue @@ -1,6 +1,7 @@ <script setup> import { ref, onMounted } from 'vue' import axios from 'axios' +import {API_BASE_URL} from "@/config/api.js"; const emit = defineEmits(['navigate']) const orders = ref([]) @@ -23,7 +24,7 @@ async function fetchOrders() { return } - const response = await axios.get('http://localhost:3000/user/orders', { + const response = await axios.get(`${API_BASE_URL}/orders`, { headers: { Authorization: `Bearer ${token}` } diff --git a/frontend/src/components/Register.vue b/frontend/src/components/Register.vue index ab6f3cc..76f08d8 100644 --- a/frontend/src/components/Register.vue +++ b/frontend/src/components/Register.vue @@ -1,6 +1,7 @@ <script setup> import { defineEmits, ref } from 'vue' import axios from "axios"; +import {API_BASE_URL} from "@/config/api.js"; const emit = defineEmits(['navigate']) @@ -21,7 +22,7 @@ async function onRegister() { error.value = '' isLoading.value = true - const response = await axios.post('http://localhost:3000/register', { + const response = await axios.post(`${API_BASE_URL}/register`, { username: username.value, password: password.value }); diff --git a/frontend/src/config/api.js b/frontend/src/config/api.js new file mode 100644 index 0000000..b7e6565 --- /dev/null +++ b/frontend/src/config/api.js @@ -0,0 +1 @@ +export const API_BASE_URL = 'http://localhost:8080/api'; \ No newline at end of file diff --git a/frontend/vite.config.js b/frontend/vite.config.js index 1947791..4eafdca 100644 --- a/frontend/vite.config.js +++ b/frontend/vite.config.js @@ -18,9 +18,8 @@ export default defineConfig({ server: { proxy: { '/api': { - target: 'http://localhost:5000', - changeOrigin: true, - secure: false + target: 'http://localhost:8080', + changeOrigin: true } } } -- GitLab