├── .gitignore ├── README.md ├── build.gradle ├── gradle └── wrapper │ ├── gradle-wrapper.jar │ └── gradle-wrapper.properties ├── gradlew ├── gradlew.bat ├── settings.gradle └── src └── main ├── java └── com │ └── java │ └── sahil │ └── minio │ ├── ServletInitializer.java │ ├── SpringBootMinioApplication.java │ ├── client │ └── DummyClient.java │ ├── config │ ├── MinioConfiguration.java │ ├── SwaggerConfiguration.java │ └── WebConfig.java │ ├── controller │ ├── SwaggerApiResponse.java │ └── UserController.java │ ├── dto │ ├── FileDto.java │ ├── PageableResponseDto.java │ ├── UserRequestDto.java │ └── UserResponseDto.java │ ├── entity │ └── User.java │ ├── error │ ├── ApiError.java │ ├── EntityNotFoundException.java │ ├── ErrorHandler.java │ ├── ExtensionNotAcceptableException.java │ └── FileCantUploadException.java │ ├── mapper │ └── UserMapper.java │ ├── repo │ └── UserRepo.java │ ├── service │ ├── FileService.java │ ├── UserService.java │ └── impl │ │ ├── FileServiceImpl.java │ │ └── UserServiceImpl.java │ └── util │ └── FileUtil.java └── resources ├── application.yml ├── banner.txt ├── base64Contract.txt ├── base64Signature.txt ├── data.sql ├── docker-compose.yml └── messages.properties /.gitignore: -------------------------------------------------------------------------------- 1 | HELP.md 2 | .gradle 3 | build/ 4 | !gradle/wrapper/gradle-wrapper.jar 5 | !**/src/main/**/build/ 6 | !**/src/test/**/build/ 7 | 8 | ### STS ### 9 | .apt_generated 10 | .classpath 11 | .factorypath 12 | .project 13 | .settings 14 | .springBeans 15 | .sts4-cache 16 | bin/ 17 | !**/src/main/**/bin/ 18 | !**/src/test/**/bin/ 19 | 20 | ### IntelliJ IDEA ### 21 | .idea 22 | *.iws 23 | *.iml 24 | *.ipr 25 | out/ 26 | !**/src/main/**/out/ 27 | !**/src/test/**/out/ 28 | 29 | ### NetBeans ### 30 | /nbproject/private/ 31 | /nbbuild/ 32 | /dist/ 33 | /nbdist/ 34 | /.nb-gradle/ 35 | 36 | ### VS Code ### 37 | .vscode/ 38 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # **Spring Boot App With MinIO File Storage Server** 2 | 3 | * Docker must be installed on your computer to run the application! 4 | 5 | * Before starting app you have to run Gradle Docker composeUp stage to create MinIo server which 6 | you can access using following url: http://127.0.0.1:9000/ (access key: test, secret key: test1234) 7 | * Create "user" bucket if it not exists 8 | 9 | -------------------------------------------------------------------------------- /build.gradle: -------------------------------------------------------------------------------- 1 | buildscript { 2 | repositories { 3 | maven { 4 | url "https://plugins.gradle.org/m2/" 5 | } 6 | } 7 | dependencies { 8 | classpath "gradle.plugin.com.avast.gradle:gradle-docker-compose-plugin:0.14.3" 9 | } 10 | } 11 | 12 | plugins { 13 | id 'org.springframework.boot' version '2.5.0' 14 | id 'io.spring.dependency-management' version '1.0.11.RELEASE' 15 | id 'java' 16 | id 'war' 17 | id "com.avast.gradle.docker-compose" version "0.14.3" 18 | } 19 | 20 | apply plugin: "com.avast.gradle.docker-compose" 21 | 22 | 23 | group = 'com.java.sahil.minio' 24 | version = '0.0.1-SNAPSHOT' 25 | sourceCompatibility = '11' 26 | 27 | configurations { 28 | compileOnly { 29 | extendsFrom annotationProcessor 30 | } 31 | } 32 | 33 | repositories { 34 | mavenCentral() 35 | } 36 | 37 | dependencies { 38 | implementation 'org.springframework.boot:spring-boot-starter-data-jpa' 39 | implementation 'org.springframework.boot:spring-boot-starter-web' 40 | implementation 'org.springframework.boot:spring-boot-starter-validation' 41 | runtimeOnly 'com.h2database:h2' 42 | compileOnly 'org.projectlombok:lombok' 43 | annotationProcessor 'org.projectlombok:lombok' 44 | implementation group: 'net.logstash.logback', name: 'logstash-logback-encoder', version: '6.6' 45 | implementation group: 'io.minio', name: 'minio', version: '8.2.1' 46 | implementation group: 'org.apache.commons', name: 'commons-lang3', version: '3.11' 47 | // implementation group: 'org.springdoc', name: 'springdoc-openapi-ui', version: '1.5.9' 48 | implementation('io.springfox:springfox-swagger-ui:2.9.2') 49 | implementation("io.springfox:springfox-swagger2:2.9.2") { 50 | exclude module: 'mapstruct' 51 | } 52 | implementation "io.springfox:springfox-bean-validators:2.9.2" 53 | implementation group: 'org.mapstruct', name: 'mapstruct', version: '1.4.2.Final' 54 | annotationProcessor group: 'org.mapstruct', name: 'mapstruct-processor', version: '1.4.2.Final' 55 | annotationProcessor 'org.springframework.boot:spring-boot-configuration-processor' 56 | providedRuntime 'org.springframework.boot:spring-boot-starter-tomcat' 57 | testImplementation 'org.springframework.boot:spring-boot-starter-test' 58 | implementation group: 'org.springframework.cloud', name: 'spring-cloud-starter-openfeign', version: '3.0.3' 59 | // https://mvnrepository.com/artifact/commons-codec/commons-codec 60 | implementation group: 'commons-codec', name: 'commons-codec', version: '1.15' 61 | 62 | 63 | } 64 | 65 | dockerCompose { 66 | useComposeFiles = ['src/main/resources/docker-compose.yml'] 67 | } 68 | 69 | test { 70 | useJUnitPlatform() 71 | } -------------------------------------------------------------------------------- /gradle/wrapper/gradle-wrapper.jar: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/sahilappayev/SpringBootMinIo/362f7714e6bfeb3df603aec7ad7f7b0c1af632ce/gradle/wrapper/gradle-wrapper.jar -------------------------------------------------------------------------------- /gradle/wrapper/gradle-wrapper.properties: -------------------------------------------------------------------------------- 1 | distributionBase=GRADLE_USER_HOME 2 | distributionPath=wrapper/dists 3 | distributionUrl=https\://services.gradle.org/distributions/gradle-7.0.2-bin.zip 4 | zipStoreBase=GRADLE_USER_HOME 5 | zipStorePath=wrapper/dists 6 | -------------------------------------------------------------------------------- /gradlew: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env sh 2 | 3 | # 4 | # Copyright 2015 the original author or authors. 5 | # 6 | # Licensed under the Apache License, Version 2.0 (the "License"); 7 | # you may not use this file except in compliance with the License. 8 | # You may obtain a copy of the License at 9 | # 10 | # https://www.apache.org/licenses/LICENSE-2.0 11 | # 12 | # Unless required by applicable law or agreed to in writing, software 13 | # distributed under the License is distributed on an "AS IS" BASIS, 14 | # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 15 | # See the License for the specific language governing permissions and 16 | # limitations under the License. 17 | # 18 | 19 | ############################################################################## 20 | ## 21 | ## Gradle start up script for UN*X 22 | ## 23 | ############################################################################## 24 | 25 | # Attempt to set APP_HOME 26 | # Resolve links: $0 may be a link 27 | PRG="$0" 28 | # Need this for relative symlinks. 29 | while [ -h "$PRG" ] ; do 30 | ls=`ls -ld "$PRG"` 31 | link=`expr "$ls" : '.*-> \(.*\)$'` 32 | if expr "$link" : '/.*' > /dev/null; then 33 | PRG="$link" 34 | else 35 | PRG=`dirname "$PRG"`"/$link" 36 | fi 37 | done 38 | SAVED="`pwd`" 39 | cd "`dirname \"$PRG\"`/" >/dev/null 40 | APP_HOME="`pwd -P`" 41 | cd "$SAVED" >/dev/null 42 | 43 | APP_NAME="Gradle" 44 | APP_BASE_NAME=`basename "$0"` 45 | 46 | # Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. 47 | DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"' 48 | 49 | # Use the maximum available, or set MAX_FD != -1 to use that value. 50 | MAX_FD="maximum" 51 | 52 | warn () { 53 | echo "$*" 54 | } 55 | 56 | die () { 57 | echo 58 | echo "$*" 59 | echo 60 | exit 1 61 | } 62 | 63 | # OS specific support (must be 'true' or 'false'). 64 | cygwin=false 65 | msys=false 66 | darwin=false 67 | nonstop=false 68 | case "`uname`" in 69 | CYGWIN* ) 70 | cygwin=true 71 | ;; 72 | Darwin* ) 73 | darwin=true 74 | ;; 75 | MINGW* ) 76 | msys=true 77 | ;; 78 | NONSTOP* ) 79 | nonstop=true 80 | ;; 81 | esac 82 | 83 | CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar 84 | 85 | 86 | # Determine the Java command to use to start the JVM. 87 | if [ -n "$JAVA_HOME" ] ; then 88 | if [ -x "$JAVA_HOME/jre/sh/java" ] ; then 89 | # IBM's JDK on AIX uses strange locations for the executables 90 | JAVACMD="$JAVA_HOME/jre/sh/java" 91 | else 92 | JAVACMD="$JAVA_HOME/bin/java" 93 | fi 94 | if [ ! -x "$JAVACMD" ] ; then 95 | die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME 96 | 97 | Please set the JAVA_HOME variable in your environment to match the 98 | location of your Java installation." 99 | fi 100 | else 101 | JAVACMD="java" 102 | which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. 103 | 104 | Please set the JAVA_HOME variable in your environment to match the 105 | location of your Java installation." 106 | fi 107 | 108 | # Increase the maximum file descriptors if we can. 109 | if [ "$cygwin" = "false" -a "$darwin" = "false" -a "$nonstop" = "false" ] ; then 110 | MAX_FD_LIMIT=`ulimit -H -n` 111 | if [ $? -eq 0 ] ; then 112 | if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then 113 | MAX_FD="$MAX_FD_LIMIT" 114 | fi 115 | ulimit -n $MAX_FD 116 | if [ $? -ne 0 ] ; then 117 | warn "Could not set maximum file descriptor limit: $MAX_FD" 118 | fi 119 | else 120 | warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT" 121 | fi 122 | fi 123 | 124 | # For Darwin, add options to specify how the application appears in the dock 125 | if $darwin; then 126 | GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\"" 127 | fi 128 | 129 | # For Cygwin or MSYS, switch paths to Windows format before running java 130 | if [ "$cygwin" = "true" -o "$msys" = "true" ] ; then 131 | APP_HOME=`cygpath --path --mixed "$APP_HOME"` 132 | CLASSPATH=`cygpath --path --mixed "$CLASSPATH"` 133 | 134 | JAVACMD=`cygpath --unix "$JAVACMD"` 135 | 136 | # We build the pattern for arguments to be converted via cygpath 137 | ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null` 138 | SEP="" 139 | for dir in $ROOTDIRSRAW ; do 140 | ROOTDIRS="$ROOTDIRS$SEP$dir" 141 | SEP="|" 142 | done 143 | OURCYGPATTERN="(^($ROOTDIRS))" 144 | # Add a user-defined pattern to the cygpath arguments 145 | if [ "$GRADLE_CYGPATTERN" != "" ] ; then 146 | OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)" 147 | fi 148 | # Now convert the arguments - kludge to limit ourselves to /bin/sh 149 | i=0 150 | for arg in "$@" ; do 151 | CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -` 152 | CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option 153 | 154 | if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition 155 | eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"` 156 | else 157 | eval `echo args$i`="\"$arg\"" 158 | fi 159 | i=`expr $i + 1` 160 | done 161 | case $i in 162 | 0) set -- ;; 163 | 1) set -- "$args0" ;; 164 | 2) set -- "$args0" "$args1" ;; 165 | 3) set -- "$args0" "$args1" "$args2" ;; 166 | 4) set -- "$args0" "$args1" "$args2" "$args3" ;; 167 | 5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;; 168 | 6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;; 169 | 7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;; 170 | 8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;; 171 | 9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;; 172 | esac 173 | fi 174 | 175 | # Escape application args 176 | save () { 177 | for i do printf %s\\n "$i" | sed "s/'/'\\\\''/g;1s/^/'/;\$s/\$/' \\\\/" ; done 178 | echo " " 179 | } 180 | APP_ARGS=`save "$@"` 181 | 182 | # Collect all arguments for the java command, following the shell quoting and substitution rules 183 | eval set -- $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS "\"-Dorg.gradle.appname=$APP_BASE_NAME\"" -classpath "\"$CLASSPATH\"" org.gradle.wrapper.GradleWrapperMain "$APP_ARGS" 184 | 185 | exec "$JAVACMD" "$@" 186 | -------------------------------------------------------------------------------- /gradlew.bat: -------------------------------------------------------------------------------- 1 | @rem 2 | @rem Copyright 2015 the original author or authors. 3 | @rem 4 | @rem Licensed under the Apache License, Version 2.0 (the "License"); 5 | @rem you may not use this file except in compliance with the License. 6 | @rem You may obtain a copy of the License at 7 | @rem 8 | @rem https://www.apache.org/licenses/LICENSE-2.0 9 | @rem 10 | @rem Unless required by applicable law or agreed to in writing, software 11 | @rem distributed under the License is distributed on an "AS IS" BASIS, 12 | @rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 | @rem See the License for the specific language governing permissions and 14 | @rem limitations under the License. 15 | @rem 16 | 17 | @if "%DEBUG%" == "" @echo off 18 | @rem ########################################################################## 19 | @rem 20 | @rem Gradle startup script for Windows 21 | @rem 22 | @rem ########################################################################## 23 | 24 | @rem Set local scope for the variables with windows NT shell 25 | if "%OS%"=="Windows_NT" setlocal 26 | 27 | set DIRNAME=%~dp0 28 | if "%DIRNAME%" == "" set DIRNAME=. 29 | set APP_BASE_NAME=%~n0 30 | set APP_HOME=%DIRNAME% 31 | 32 | @rem Resolve any "." and ".." in APP_HOME to make it shorter. 33 | for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi 34 | 35 | @rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. 36 | set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m" 37 | 38 | @rem Find java.exe 39 | if defined JAVA_HOME goto findJavaFromJavaHome 40 | 41 | set JAVA_EXE=java.exe 42 | %JAVA_EXE% -version >NUL 2>&1 43 | if "%ERRORLEVEL%" == "0" goto execute 44 | 45 | echo. 46 | echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. 47 | echo. 48 | echo Please set the JAVA_HOME variable in your environment to match the 49 | echo location of your Java installation. 50 | 51 | goto fail 52 | 53 | :findJavaFromJavaHome 54 | set JAVA_HOME=%JAVA_HOME:"=% 55 | set JAVA_EXE=%JAVA_HOME%/bin/java.exe 56 | 57 | if exist "%JAVA_EXE%" goto execute 58 | 59 | echo. 60 | echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% 61 | echo. 62 | echo Please set the JAVA_HOME variable in your environment to match the 63 | echo location of your Java installation. 64 | 65 | goto fail 66 | 67 | :execute 68 | @rem Setup the command line 69 | 70 | set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar 71 | 72 | 73 | @rem Execute Gradle 74 | "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %* 75 | 76 | :end 77 | @rem End local scope for the variables with windows NT shell 78 | if "%ERRORLEVEL%"=="0" goto mainEnd 79 | 80 | :fail 81 | rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of 82 | rem the _cmd.exe /c_ return code! 83 | if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1 84 | exit /b 1 85 | 86 | :mainEnd 87 | if "%OS%"=="Windows_NT" endlocal 88 | 89 | :omega 90 | -------------------------------------------------------------------------------- /settings.gradle: -------------------------------------------------------------------------------- 1 | rootProject.name = 'spring-boot-minio' 2 | -------------------------------------------------------------------------------- /src/main/java/com/java/sahil/minio/ServletInitializer.java: -------------------------------------------------------------------------------- 1 | package com.java.sahil.minio; 2 | 3 | import org.springframework.boot.builder.SpringApplicationBuilder; 4 | import org.springframework.boot.web.servlet.support.SpringBootServletInitializer; 5 | 6 | public class ServletInitializer extends SpringBootServletInitializer { 7 | 8 | @Override 9 | protected SpringApplicationBuilder configure(SpringApplicationBuilder application) { 10 | return application.sources(SpringBootMinioApplication.class); 11 | } 12 | 13 | } 14 | -------------------------------------------------------------------------------- /src/main/java/com/java/sahil/minio/SpringBootMinioApplication.java: -------------------------------------------------------------------------------- 1 | package com.java.sahil.minio; 2 | 3 | import com.java.sahil.minio.client.DummyClient; 4 | import org.springframework.boot.SpringApplication; 5 | import org.springframework.boot.autoconfigure.SpringBootApplication; 6 | import org.springframework.cloud.openfeign.EnableFeignClients; 7 | 8 | /** 9 | * @author SahilAppayev 10 | * @since 23.05.2021 11 | */ 12 | 13 | @SpringBootApplication 14 | //@OpenAPIDefinition(info = @Info(title = "SpringBoot - MinIO", version = "1.0", 15 | // description = "Spring Boot App With MinIO File Storage Server")) 16 | public class SpringBootMinioApplication { 17 | 18 | public static void main(String[] args) { 19 | SpringApplication.run(SpringBootMinioApplication.class, args); 20 | } 21 | 22 | } 23 | -------------------------------------------------------------------------------- /src/main/java/com/java/sahil/minio/client/DummyClient.java: -------------------------------------------------------------------------------- 1 | package com.java.sahil.minio.client; 2 | 3 | import lombok.SneakyThrows; 4 | import org.apache.commons.io.IOUtils; 5 | 6 | import java.io.FileInputStream; 7 | 8 | public class DummyClient { 9 | 10 | @SneakyThrows 11 | public static String getBase64Contract(){ 12 | FileInputStream fis = new FileInputStream("src/main/resources/base64Contract.txt"); 13 | return IOUtils.toString(fis, "UTF-8"); 14 | } 15 | 16 | @SneakyThrows 17 | public static String getBase64Signature(){ 18 | FileInputStream fis = new FileInputStream("src/main/resources/base64Signature.txt"); 19 | return IOUtils.toString(fis, "UTF-8"); 20 | } 21 | } 22 | -------------------------------------------------------------------------------- /src/main/java/com/java/sahil/minio/config/MinioConfiguration.java: -------------------------------------------------------------------------------- 1 | package com.java.sahil.minio.config; 2 | 3 | import io.minio.MinioClient; 4 | import org.springframework.beans.factory.annotation.Value; 5 | import org.springframework.context.annotation.Bean; 6 | import org.springframework.context.annotation.Configuration; 7 | 8 | @Configuration 9 | public class MinioConfiguration { 10 | @Value("${minio.access-key}") 11 | private String accessKey; 12 | @Value("${minio.secret-key}") 13 | private String secretKet; 14 | @Value("${minio.url}") 15 | private String url; 16 | 17 | @Bean 18 | public MinioClient minioClient() { 19 | return MinioClient.builder() 20 | .credentials(accessKey, secretKet) 21 | .endpoint(url).build(); 22 | } 23 | } 24 | -------------------------------------------------------------------------------- /src/main/java/com/java/sahil/minio/config/SwaggerConfiguration.java: -------------------------------------------------------------------------------- 1 | package com.java.sahil.minio.config; 2 | 3 | import org.springframework.beans.factory.annotation.Value; 4 | import org.springframework.context.annotation.Bean; 5 | import org.springframework.context.annotation.Configuration; 6 | import org.springframework.http.MediaType; 7 | import org.springframework.web.servlet.config.annotation.ResourceHandlerRegistry; 8 | import org.springframework.web.servlet.config.annotation.WebMvcConfigurationSupport; 9 | import springfox.documentation.builders.ApiInfoBuilder; 10 | import springfox.documentation.builders.PathSelectors; 11 | import springfox.documentation.builders.RequestHandlerSelectors; 12 | import springfox.documentation.service.ApiInfo; 13 | import springfox.documentation.spi.DocumentationType; 14 | import springfox.documentation.spring.web.plugins.Docket; 15 | import springfox.documentation.swagger2.annotations.EnableSwagger2; 16 | 17 | import java.util.Collections; 18 | 19 | @Configuration 20 | @EnableSwagger2 21 | public class SwaggerConfiguration extends WebMvcConfigurationSupport { 22 | @Value("${application.swagger.title}") 23 | private String apiTitle; 24 | @Value("${application.swagger.description}") 25 | private String apiDescription; 26 | @Value("${application.swagger.version}") 27 | private String apiVersion; 28 | 29 | @Bean 30 | public Docket api() { 31 | return new Docket(DocumentationType.SWAGGER_2) 32 | .select() 33 | .apis(RequestHandlerSelectors.basePackage("com.java.sahil.minio")) 34 | .paths(PathSelectors.any()) 35 | .build().apiInfo(apiInfo()) 36 | .produces(Collections.singleton(MediaType.APPLICATION_JSON_VALUE)) 37 | .consumes(Collections.singleton(MediaType.APPLICATION_JSON_VALUE)); 38 | } 39 | 40 | private ApiInfo apiInfo() { 41 | return new ApiInfoBuilder() 42 | .title(apiTitle) 43 | .version(apiVersion) 44 | .description(apiDescription) 45 | .build(); 46 | } 47 | 48 | @Override 49 | public void addResourceHandlers(ResourceHandlerRegistry registry) { 50 | registry.addResourceHandler("swagger-ui.html") 51 | .addResourceLocations("classpath:/META-INF/resources/"); 52 | registry.addResourceHandler("/webjars/**") 53 | .addResourceLocations("classpath:/META-INF/resources/webjars/"); 54 | } 55 | } 56 | 57 | -------------------------------------------------------------------------------- /src/main/java/com/java/sahil/minio/config/WebConfig.java: -------------------------------------------------------------------------------- 1 | package com.java.sahil.minio.config; 2 | 3 | import org.springframework.context.annotation.Configuration; 4 | import org.springframework.web.servlet.config.annotation.CorsRegistry; 5 | import org.springframework.web.servlet.config.annotation.EnableWebMvc; 6 | import org.springframework.web.servlet.config.annotation.WebMvcConfigurer; 7 | 8 | @Configuration 9 | @EnableWebMvc 10 | public class WebConfig implements WebMvcConfigurer { 11 | 12 | @Override 13 | public void addCorsMappings(CorsRegistry registry) { 14 | registry.addMapping("/**") 15 | .allowedHeaders("*") 16 | .allowedMethods("*") 17 | .allowedOrigins("*") 18 | .maxAge(600); 19 | } 20 | } -------------------------------------------------------------------------------- /src/main/java/com/java/sahil/minio/controller/SwaggerApiResponse.java: -------------------------------------------------------------------------------- 1 | package com.java.sahil.minio.controller; 2 | 3 | import com.java.sahil.minio.error.ApiError; 4 | import io.swagger.annotations.ApiResponse; 5 | import io.swagger.annotations.ApiResponses; 6 | 7 | import java.lang.annotation.ElementType; 8 | import java.lang.annotation.Retention; 9 | import java.lang.annotation.RetentionPolicy; 10 | import java.lang.annotation.Target; 11 | 12 | @Retention(RetentionPolicy.RUNTIME) 13 | @Target(ElementType.METHOD) 14 | @ApiResponses(value = { 15 | @ApiResponse(code = 500, message = "Server Error", response = ApiError.class), 16 | @ApiResponse(code = 404, message = "Not Found", response = ApiError.class), 17 | }) 18 | public @interface SwaggerApiResponse { 19 | 20 | } 21 | -------------------------------------------------------------------------------- /src/main/java/com/java/sahil/minio/controller/UserController.java: -------------------------------------------------------------------------------- 1 | package com.java.sahil.minio.controller; 2 | 3 | import com.java.sahil.minio.dto.FileDto; 4 | import com.java.sahil.minio.dto.PageableResponseDto; 5 | import com.java.sahil.minio.dto.UserRequestDto; 6 | import com.java.sahil.minio.dto.UserResponseDto; 7 | import com.java.sahil.minio.service.UserService; 8 | import io.swagger.annotations.ApiOperation; 9 | import lombok.RequiredArgsConstructor; 10 | import lombok.SneakyThrows; 11 | import org.springframework.beans.factory.annotation.Value; 12 | import org.springframework.core.io.InputStreamResource; 13 | import org.springframework.http.HttpHeaders; 14 | import org.springframework.http.HttpStatus; 15 | import org.springframework.http.MediaType; 16 | import org.springframework.http.ResponseEntity; 17 | import org.springframework.web.bind.annotation.DeleteMapping; 18 | import org.springframework.web.bind.annotation.GetMapping; 19 | import org.springframework.web.bind.annotation.PathVariable; 20 | import org.springframework.web.bind.annotation.PostMapping; 21 | import org.springframework.web.bind.annotation.PutMapping; 22 | import org.springframework.web.bind.annotation.RequestBody; 23 | import org.springframework.web.bind.annotation.RequestMapping; 24 | import org.springframework.web.bind.annotation.RequestParam; 25 | import org.springframework.web.bind.annotation.ResponseStatus; 26 | import org.springframework.web.bind.annotation.RestController; 27 | import org.springframework.web.multipart.MultipartFile; 28 | 29 | import javax.validation.Valid; 30 | import java.io.ByteArrayInputStream; 31 | import java.io.File; 32 | import java.io.InputStream; 33 | import java.nio.file.Files; 34 | import java.nio.file.Path; 35 | import java.util.List; 36 | import java.util.Map; 37 | 38 | @RestController 39 | @RequestMapping("users") 40 | @RequiredArgsConstructor 41 | public class UserController { 42 | 43 | private final UserService userService; 44 | @Value("${minio.image-folder}") 45 | private String imageFolder; 46 | @Value("${minio.video-folder}") 47 | private String videoFolder; 48 | @Value("${minio.resume-folder}") 49 | private String resumeFolder; 50 | 51 | @GetMapping 52 | @ApiOperation(value = "Get User List as Pageable") 53 | @SwaggerApiResponse 54 | public ResponseEntity>> getAllPaged( 55 | @RequestParam(value = "page", defaultValue = "0") int page, 56 | @RequestParam(value = "size", defaultValue = "5") int size) { 57 | return ResponseEntity.status(HttpStatus.OK).body(userService.findAll(page, size)); 58 | } 59 | 60 | 61 | @GetMapping("{id}") 62 | @ApiOperation(value = "Get User by Id") 63 | @SwaggerApiResponse 64 | public ResponseEntity getById(@PathVariable("id") Long id) { 65 | return ResponseEntity.status(HttpStatus.OK).body(userService.findById(id)); 66 | } 67 | 68 | @PostMapping 69 | @ApiOperation(value = "Add User") 70 | @SwaggerApiResponse 71 | public ResponseEntity create(@Valid @RequestBody UserRequestDto userRequestDto) { 72 | return ResponseEntity.status(HttpStatus.CREATED).body(userService.create(userRequestDto)); 73 | } 74 | 75 | @PutMapping("{id}") 76 | @ApiOperation(value = "Update User") 77 | @SwaggerApiResponse 78 | public ResponseEntity update(@PathVariable("id") Long id, 79 | @Valid @RequestBody UserRequestDto userRequestDto) { 80 | return ResponseEntity.status(HttpStatus.CREATED).body(userService.update(userRequestDto, id)); 81 | } 82 | 83 | @DeleteMapping("{id}") 84 | @ApiOperation(value = "Delete User") 85 | @SwaggerApiResponse 86 | public ResponseEntity delete(@PathVariable("id") Long id) { 87 | return ResponseEntity.status(HttpStatus.OK).body(userService.delete(id)); 88 | } 89 | 90 | /** 91 | * FILE URIs 92 | */ 93 | @PostMapping(value = "/image/{id}") 94 | @ApiOperation(value = "Add User Image") 95 | public ResponseEntity createImage(@PathVariable Long id, @Valid @RequestParam MultipartFile file) { 96 | return ResponseEntity.status(HttpStatus.CREATED).body(new FileDto(userService.uploadImage(file, id))); 97 | } 98 | 99 | @PutMapping(value = "/image/{id}") 100 | @ApiOperation(value = "Update User Image") 101 | public ResponseEntity updateImage(@PathVariable Long id, @Valid @RequestParam MultipartFile file) { 102 | return ResponseEntity.status(HttpStatus.CREATED).body(new FileDto(userService.updateImage(file, id))); 103 | } 104 | 105 | @PutMapping(value = "/image/contract/{dummyPin}") 106 | @ApiOperation(value = "Update User Image") 107 | public ResponseEntity uploadContractByPin(@PathVariable String dummyPin) { 108 | return ResponseEntity.status(HttpStatus.CREATED).body(new FileDto(userService.uploadContractByPin(dummyPin))); 109 | } 110 | 111 | @GetMapping(value = "/image/{fileName}", produces = {MediaType.IMAGE_JPEG_VALUE, MediaType.IMAGE_PNG_VALUE}) 112 | @ApiOperation(value = "Get User Image") 113 | @ResponseStatus(HttpStatus.OK) 114 | public byte[] getImage(@PathVariable("fileName") String fileName) { 115 | return userService.getFile(fileName, imageFolder); 116 | } 117 | 118 | @DeleteMapping("/image/{id}") 119 | @ApiOperation(value = "Delete User Image") 120 | public void deleteUserImage(@PathVariable("id") Long id) { 121 | userService.deleteUserImage(id); 122 | } 123 | 124 | @PostMapping(value = "/video/{id}") 125 | @ApiOperation(value = "Add User Video") 126 | public ResponseEntity createVideo(@PathVariable Long id, @Valid @RequestParam MultipartFile file) { 127 | return ResponseEntity.status(HttpStatus.CREATED).body(new FileDto(userService.uploadVideo(file, id))); 128 | } 129 | 130 | @PutMapping(value = "/video/{id}") 131 | @ApiOperation(value = "Update User Video") 132 | public ResponseEntity updateVideo(@PathVariable Long id, @Valid @RequestParam MultipartFile file) { 133 | return ResponseEntity.status(HttpStatus.CREATED).body(new FileDto(userService.updateVideo(file, id))); 134 | } 135 | 136 | @SneakyThrows 137 | @GetMapping(value = "/video/{fileName}") 138 | @ApiOperation(value = "Get User Video") 139 | public ResponseEntity getVideo(@PathVariable("fileName") String fileName) { 140 | byte[] videoData = userService.getFile(fileName, videoFolder); 141 | try (InputStream file = new ByteArrayInputStream(videoData)) { 142 | Path path = new File(fileName).toPath(); 143 | String mimeType = Files.probeContentType(path); 144 | HttpHeaders headers = new HttpHeaders(); 145 | headers.set("Accept-Ranges", "bytes"); 146 | headers.set("Content-Type", mimeType); 147 | headers.set("Content-Range", "bytes 50-1025/17839845"); 148 | headers.set("Content-Length", String.valueOf(videoData.length)); 149 | return new ResponseEntity<>(new InputStreamResource(file), headers, HttpStatus.OK); 150 | } 151 | } 152 | 153 | @DeleteMapping("/video/{id}") 154 | @ApiOperation(value = "Delete User Video") 155 | public void deleteUserVideo(@PathVariable("id") Long id) { 156 | userService.deleteUserVideo(id); 157 | } 158 | 159 | 160 | @PostMapping("/resume/{id}") 161 | @ApiOperation(value = "Add User Pdf Resume") 162 | public ResponseEntity uploadResume(@PathVariable Long id, @Valid @RequestParam MultipartFile file) { 163 | return ResponseEntity.status(HttpStatus.CREATED).body(new FileDto(userService.uploadResume(file, id))); 164 | } 165 | 166 | @PutMapping(value = "/resume/{id}") 167 | @ApiOperation(value = "Update User Pdf Resume") 168 | public ResponseEntity updateResume(@PathVariable Long id, @Valid @RequestParam MultipartFile file) { 169 | return ResponseEntity.status(HttpStatus.CREATED).body(new FileDto(userService.updateResume(file, id))); 170 | } 171 | 172 | @GetMapping(value = "/resume/{fileName}", produces = {MediaType.APPLICATION_PDF_VALUE}) 173 | @ApiOperation(value = "Get User Resume") 174 | @ResponseStatus(HttpStatus.OK) 175 | public byte[] getResume(@PathVariable("fileName") String fileName) { 176 | return userService.getFile(fileName, resumeFolder); 177 | } 178 | 179 | @PostMapping("/{id}/upload-photos") 180 | public Map uploadPhotos(@PathVariable Long id, 181 | @RequestParam String[] names, 182 | @RequestParam MultipartFile[] files) { 183 | return userService.uploadPhotos(id, names, files); 184 | } 185 | 186 | } 187 | -------------------------------------------------------------------------------- /src/main/java/com/java/sahil/minio/dto/FileDto.java: -------------------------------------------------------------------------------- 1 | package com.java.sahil.minio.dto; 2 | 3 | import lombok.AccessLevel; 4 | import lombok.AllArgsConstructor; 5 | import lombok.Getter; 6 | import lombok.Setter; 7 | import lombok.experimental.FieldDefaults; 8 | 9 | @Getter 10 | @Setter 11 | @AllArgsConstructor 12 | @FieldDefaults(level = AccessLevel.PRIVATE) 13 | public class FileDto { 14 | String fileName; 15 | } 16 | -------------------------------------------------------------------------------- /src/main/java/com/java/sahil/minio/dto/PageableResponseDto.java: -------------------------------------------------------------------------------- 1 | package com.java.sahil.minio.dto; 2 | 3 | import lombok.AllArgsConstructor; 4 | import lombok.Data; 5 | import lombok.NoArgsConstructor; 6 | 7 | import java.io.Serializable; 8 | 9 | @Data 10 | @AllArgsConstructor 11 | @NoArgsConstructor 12 | public class PageableResponseDto implements Serializable { 13 | 14 | private T responseDto; 15 | private long totalElementsOnDatabase; 16 | private long returnedElementsToUI; 17 | } 18 | -------------------------------------------------------------------------------- /src/main/java/com/java/sahil/minio/dto/UserRequestDto.java: -------------------------------------------------------------------------------- 1 | package com.java.sahil.minio.dto; 2 | 3 | import lombok.AllArgsConstructor; 4 | import lombok.Data; 5 | import lombok.NoArgsConstructor; 6 | 7 | import javax.validation.constraints.NotBlank; 8 | 9 | @Data 10 | @NoArgsConstructor 11 | @AllArgsConstructor 12 | public class UserRequestDto { 13 | 14 | @NotBlank 15 | private String name; 16 | @NotBlank 17 | private String surname; 18 | 19 | } 20 | -------------------------------------------------------------------------------- /src/main/java/com/java/sahil/minio/dto/UserResponseDto.java: -------------------------------------------------------------------------------- 1 | package com.java.sahil.minio.dto; 2 | 3 | import lombok.AllArgsConstructor; 4 | import lombok.Data; 5 | import lombok.NoArgsConstructor; 6 | 7 | import java.util.Map; 8 | 9 | @Data 10 | @NoArgsConstructor 11 | @AllArgsConstructor 12 | public class UserResponseDto { 13 | 14 | private Long id; 15 | private String name; 16 | private String surname; 17 | private String photo; 18 | private String video; 19 | private String resume; 20 | 21 | private Map photos; 22 | 23 | } 24 | -------------------------------------------------------------------------------- /src/main/java/com/java/sahil/minio/entity/User.java: -------------------------------------------------------------------------------- 1 | package com.java.sahil.minio.entity; 2 | 3 | import lombok.EqualsAndHashCode; 4 | import lombok.Getter; 5 | import lombok.Setter; 6 | 7 | import javax.persistence.ElementCollection; 8 | import javax.persistence.Entity; 9 | import javax.persistence.GeneratedValue; 10 | import javax.persistence.GenerationType; 11 | import javax.persistence.Id; 12 | import javax.persistence.Table; 13 | import java.io.Serializable; 14 | import java.util.HashMap; 15 | import java.util.Map; 16 | 17 | @Getter 18 | @Setter 19 | @EqualsAndHashCode(of = "id") 20 | @Entity 21 | @Table(name = "user") 22 | public class User implements Serializable { 23 | private static final long serialVersionUID = 1L; 24 | 25 | @Id 26 | @GeneratedValue(strategy = GenerationType.IDENTITY) 27 | private Long id; 28 | private String name; 29 | private String surname; 30 | private String photo; 31 | private String video; 32 | private String resume; 33 | 34 | @ElementCollection 35 | private Map photos = new HashMap<>(); 36 | 37 | } 38 | -------------------------------------------------------------------------------- /src/main/java/com/java/sahil/minio/error/ApiError.java: -------------------------------------------------------------------------------- 1 | package com.java.sahil.minio.error; 2 | 3 | import lombok.AllArgsConstructor; 4 | import lombok.Data; 5 | 6 | @Data 7 | @AllArgsConstructor 8 | public class ApiError { 9 | private int code; 10 | private String message; 11 | } 12 | -------------------------------------------------------------------------------- /src/main/java/com/java/sahil/minio/error/EntityNotFoundException.java: -------------------------------------------------------------------------------- 1 | package com.java.sahil.minio.error; 2 | 3 | public class EntityNotFoundException extends RuntimeException { 4 | public EntityNotFoundException(Class entity) { 5 | super(entity.getSimpleName() + " "); 6 | } 7 | 8 | public EntityNotFoundException(Class entity, Object id) { 9 | super(entity.getSimpleName() + " with ID: " + id + " "); 10 | } 11 | } 12 | -------------------------------------------------------------------------------- /src/main/java/com/java/sahil/minio/error/ErrorHandler.java: -------------------------------------------------------------------------------- 1 | package com.java.sahil.minio.error; 2 | 3 | import lombok.extern.slf4j.Slf4j; 4 | import org.springframework.beans.factory.annotation.Value; 5 | import org.springframework.context.annotation.PropertySource; 6 | import org.springframework.http.HttpHeaders; 7 | import org.springframework.http.HttpStatus; 8 | import org.springframework.http.ResponseEntity; 9 | import org.springframework.http.converter.HttpMessageNotReadableException; 10 | import org.springframework.validation.FieldError; 11 | import org.springframework.web.bind.MethodArgumentNotValidException; 12 | import org.springframework.web.bind.annotation.ExceptionHandler; 13 | import org.springframework.web.bind.annotation.ResponseStatus; 14 | import org.springframework.web.bind.annotation.RestControllerAdvice; 15 | import org.springframework.web.context.request.WebRequest; 16 | import org.springframework.web.servlet.mvc.method.annotation.ResponseEntityExceptionHandler; 17 | 18 | import java.util.HashMap; 19 | import java.util.Map; 20 | 21 | import static net.logstash.logback.argument.StructuredArguments.kv; 22 | 23 | @RestControllerAdvice 24 | @Slf4j 25 | @PropertySource(value = "classpath:messages.properties", encoding = "UTF-8") 26 | public class ErrorHandler extends ResponseEntityExceptionHandler { 27 | @Value("${error.notFound}") 28 | private String notFound; 29 | @Value("${error.extensionNotAcceptable}") 30 | private String extensionError; 31 | 32 | 33 | @Override 34 | protected ResponseEntity handleMethodArgumentNotValid(MethodArgumentNotValidException ex, 35 | HttpHeaders headers, 36 | HttpStatus status, 37 | WebRequest request) { 38 | Map errors = new HashMap<>(); 39 | var ref = new Object() { 40 | String fieldName = null; 41 | String message = null; 42 | }; 43 | ex.getBindingResult().getAllErrors().forEach((error) -> { 44 | ref.fieldName = ((FieldError) error).getField(); 45 | ref.message = error.getDefaultMessage(); 46 | errors.put(ref.fieldName, ref.message); 47 | }); 48 | errors.forEach((fieldName, message) -> { 49 | log.error("Api error, {}, message: {} ", 50 | kv("errorCode", 1011), fieldName + " - " + message); 51 | }); 52 | return new ResponseEntity<>(new ApiError(1011, ref.fieldName + " - " + ref.message), 53 | HttpStatus.BAD_REQUEST); 54 | } 55 | 56 | @Override 57 | protected ResponseEntity handleHttpMessageNotReadable(HttpMessageNotReadableException ex, 58 | HttpHeaders headers, 59 | HttpStatus status, 60 | WebRequest request) { 61 | log.error("Api error, {}, message: {} ", 62 | kv("errorCode", 1011), ex.getMessage()); 63 | return new ResponseEntity<>(new ApiError(1011, ex.getCause().getMessage()), 64 | HttpStatus.BAD_REQUEST); 65 | } 66 | 67 | 68 | @ExceptionHandler(ExtensionNotAcceptableException.class) 69 | @ResponseStatus(HttpStatus.BAD_REQUEST) 70 | public ApiError extensionError(final Exception e) { 71 | log.error("Api error, {}, message: {}, {} ", 72 | kv("errorCode", 1001), e.getMessage(), extensionError); 73 | return new ApiError(1001, e.getMessage() + extensionError); 74 | } 75 | 76 | @ExceptionHandler(EntityNotFoundException.class) 77 | @ResponseStatus(HttpStatus.NOT_FOUND) 78 | public ApiError entityNotFound(final Exception e) { 79 | log.error("Api error, {}, message: {}, {} ", 80 | kv("errorCode", 1002), e.getMessage(), notFound); 81 | return new ApiError(1002, e.getMessage() + notFound); 82 | } 83 | 84 | @ExceptionHandler(FileCantUploadException.class) 85 | @ResponseStatus(HttpStatus.BAD_REQUEST) 86 | public ApiError fileUploadError(final Exception e) { 87 | log.error("Api error, {}, message: {}", 88 | kv("errorCode", 1001), e.getMessage()); 89 | return new ApiError(1001, e.getMessage()); 90 | } 91 | 92 | } -------------------------------------------------------------------------------- /src/main/java/com/java/sahil/minio/error/ExtensionNotAcceptableException.java: -------------------------------------------------------------------------------- 1 | package com.java.sahil.minio.error; 2 | 3 | public class ExtensionNotAcceptableException extends RuntimeException { 4 | public ExtensionNotAcceptableException(String extension) { 5 | super("." + extension + " "); 6 | } 7 | } 8 | -------------------------------------------------------------------------------- /src/main/java/com/java/sahil/minio/error/FileCantUploadException.java: -------------------------------------------------------------------------------- 1 | package com.java.sahil.minio.error; 2 | 3 | public class FileCantUploadException extends RuntimeException { 4 | public FileCantUploadException(String fileName) { 5 | super(fileName + ",this file cant upload!"); 6 | } 7 | } 8 | -------------------------------------------------------------------------------- /src/main/java/com/java/sahil/minio/mapper/UserMapper.java: -------------------------------------------------------------------------------- 1 | package com.java.sahil.minio.mapper; 2 | 3 | import com.java.sahil.minio.dto.UserRequestDto; 4 | import com.java.sahil.minio.dto.UserResponseDto; 5 | import com.java.sahil.minio.entity.User; 6 | import org.mapstruct.Mapper; 7 | import org.mapstruct.ReportingPolicy; 8 | 9 | import java.util.List; 10 | 11 | @Mapper(unmappedSourcePolicy = ReportingPolicy.IGNORE, 12 | unmappedTargetPolicy = ReportingPolicy.IGNORE, 13 | componentModel = "spring") 14 | public interface UserMapper { 15 | 16 | UserResponseDto toUserDto(User user); 17 | 18 | User toUserEntity(UserRequestDto userRequestDto); 19 | 20 | List toUserDtoList(List userList); 21 | 22 | } 23 | -------------------------------------------------------------------------------- /src/main/java/com/java/sahil/minio/repo/UserRepo.java: -------------------------------------------------------------------------------- 1 | package com.java.sahil.minio.repo; 2 | 3 | import com.java.sahil.minio.entity.User; 4 | import org.springframework.data.jpa.repository.JpaRepository; 5 | 6 | public interface UserRepo extends JpaRepository { 7 | } 8 | -------------------------------------------------------------------------------- /src/main/java/com/java/sahil/minio/service/FileService.java: -------------------------------------------------------------------------------- 1 | package com.java.sahil.minio.service; 2 | 3 | import org.springframework.web.multipart.MultipartFile; 4 | 5 | public interface FileService { 6 | 7 | byte[] getFile(String fileName, String folder); 8 | 9 | String uploadImage(MultipartFile file, String folder, boolean isResize); 10 | 11 | void deleteFile(String fileName, String folder); 12 | 13 | String uploadVideo(MultipartFile file, String folder); 14 | 15 | String uploadPdf(MultipartFile file, String folder); 16 | } 17 | -------------------------------------------------------------------------------- /src/main/java/com/java/sahil/minio/service/UserService.java: -------------------------------------------------------------------------------- 1 | package com.java.sahil.minio.service; 2 | 3 | import com.java.sahil.minio.dto.PageableResponseDto; 4 | import com.java.sahil.minio.dto.UserRequestDto; 5 | import com.java.sahil.minio.dto.UserResponseDto; 6 | import org.springframework.web.multipart.MultipartFile; 7 | 8 | import java.util.List; 9 | import java.util.Map; 10 | 11 | public interface UserService { 12 | 13 | UserResponseDto create(UserRequestDto userRequestDto); 14 | 15 | UserResponseDto update(UserRequestDto userRequestDto, Long id); 16 | 17 | UserResponseDto findById(Long id); 18 | 19 | PageableResponseDto> findAll(int page, int size); 20 | 21 | UserResponseDto delete(Long id); 22 | 23 | String uploadImage(MultipartFile file, Long id); 24 | 25 | String updateImage(MultipartFile file, Long id); 26 | 27 | void deleteUserImage(Long id); 28 | 29 | void deleteFile(String fileName, String folder); 30 | 31 | byte[] getFile(String fileName, String folder); 32 | 33 | String uploadVideo(MultipartFile file, Long id); 34 | 35 | String updateVideo(MultipartFile file, Long id); 36 | 37 | void deleteUserVideo(Long id); 38 | 39 | String uploadContractByPin(String dummyPin); 40 | 41 | String uploadResume(MultipartFile file, Long id); 42 | 43 | String updateResume(MultipartFile file, Long id); 44 | 45 | Map uploadPhotos(Long id, String[] names, MultipartFile[] files); 46 | } 47 | -------------------------------------------------------------------------------- /src/main/java/com/java/sahil/minio/service/impl/FileServiceImpl.java: -------------------------------------------------------------------------------- 1 | package com.java.sahil.minio.service.impl; 2 | 3 | import com.java.sahil.minio.service.FileService; 4 | import com.java.sahil.minio.util.FileUtil; 5 | import io.minio.GetObjectArgs; 6 | import io.minio.MinioClient; 7 | import io.minio.PutObjectArgs; 8 | import io.minio.RemoveObjectArgs; 9 | import io.minio.errors.ErrorResponseException; 10 | import io.minio.messages.ErrorResponse; 11 | import lombok.RequiredArgsConstructor; 12 | import lombok.SneakyThrows; 13 | import lombok.extern.slf4j.Slf4j; 14 | import org.springframework.beans.factory.annotation.Value; 15 | import org.springframework.stereotype.Service; 16 | import org.springframework.web.multipart.MultipartFile; 17 | 18 | import javax.imageio.ImageIO; 19 | import java.awt.image.BufferedImage; 20 | import java.io.ByteArrayInputStream; 21 | import java.io.ByteArrayOutputStream; 22 | import java.io.File; 23 | import java.io.InputStream; 24 | import java.nio.file.Files; 25 | import java.nio.file.Path; 26 | 27 | import static net.logstash.logback.argument.StructuredArguments.kv; 28 | 29 | @Service 30 | @RequiredArgsConstructor 31 | @Slf4j 32 | public class FileServiceImpl implements FileService { 33 | private final MinioClient minioClient; 34 | private final FileUtil fileUtil; 35 | @Value("${minio.bucket}") 36 | private String bucketName; 37 | private final String VIDEO_MEDIA_TYPE = "video"; 38 | private final String IMAGE_MEDIA_TYPE = "image"; 39 | private final String PDF_MEDIA_TYPE = "pdf"; 40 | 41 | @SneakyThrows 42 | @Override 43 | public byte[] getFile(String fileName, String folder) { 44 | String objectName = folder + fileName; 45 | GetObjectArgs minioRequest = GetObjectArgs.builder().bucket(bucketName).object(objectName).build(); 46 | byte[] bytes = null; 47 | try { 48 | bytes = minioClient.getObject(minioRequest).readAllBytes(); 49 | } catch (ErrorResponseException e) { 50 | ErrorResponse response = e.errorResponse(); 51 | log.error("Minio error occurred with: {}, {}, {}", 52 | kv("code", response.code()), kv("message", response.message()), 53 | kv("objectName", response.objectName())); 54 | } 55 | return bytes; 56 | } 57 | 58 | @SneakyThrows 59 | @Override 60 | public String uploadImage(MultipartFile file, String folder, boolean isResize) { 61 | String fileExtension = fileUtil.getFileExtensionIfAcceptable(file, IMAGE_MEDIA_TYPE); 62 | String fileName = fileUtil.generateUniqueName(fileExtension); 63 | String objectName = folder + fileName; 64 | InputStream inputStream = file.getInputStream(); 65 | 66 | if (isResize){ 67 | BufferedImage image = ImageIO.read(inputStream); 68 | ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream(); 69 | ImageIO.write(fileUtil.resizeImage(image, image.getWidth(), image.getHeight()), fileExtension, byteArrayOutputStream); 70 | inputStream = new ByteArrayInputStream(byteArrayOutputStream.toByteArray()); 71 | } 72 | 73 | minioClient.putObject(PutObjectArgs.builder().bucket(bucketName).object(objectName).stream( 74 | inputStream, inputStream.available(), -1) 75 | .contentType(file.getContentType()) 76 | .build()); 77 | return fileName; 78 | } 79 | 80 | 81 | @SneakyThrows 82 | @Override 83 | public void deleteFile(String fileName, String folder) { 84 | String objectName = folder + fileName; 85 | minioClient.removeObject(RemoveObjectArgs.builder().bucket(bucketName).object(objectName).build()); 86 | } 87 | 88 | @SneakyThrows 89 | @Override 90 | public String uploadVideo(MultipartFile file, String folder) { 91 | String fileExtension = fileUtil.getFileExtensionIfAcceptable(file, VIDEO_MEDIA_TYPE); 92 | String fileName = fileUtil.generateUniqueName(fileExtension); 93 | String objectName = folder + fileName; 94 | minioClient.putObject(PutObjectArgs.builder().bucket(bucketName).object(objectName).stream( 95 | file.getInputStream(), file.getInputStream().available(), -1) 96 | .contentType(file.getContentType()) 97 | .build()); 98 | return fileName; 99 | } 100 | 101 | @SneakyThrows 102 | @Override 103 | public String uploadPdf(MultipartFile file, String folder) { 104 | String fileExtension = fileUtil.getFileExtensionIfAcceptable(file, PDF_MEDIA_TYPE); 105 | String fileName = fileUtil.generateUniqueName(fileExtension); 106 | String objectName = folder + fileName; 107 | InputStream inputStream = file.getInputStream(); 108 | minioClient.putObject(PutObjectArgs.builder().bucket(bucketName).object(objectName).stream( 109 | inputStream, inputStream.available(), -1) 110 | .contentType(file.getContentType()) 111 | .build()); 112 | return fileName; 113 | } 114 | 115 | @SneakyThrows 116 | public String uploadInputStreamImage(InputStream file, String folder) { 117 | String fileName = fileUtil.generateUniqueName(fileUtil.getFileExtensionFromInputStream(file)); 118 | String fileExtension = fileUtil.getFileExtensionIfAcceptable(fileName, IMAGE_MEDIA_TYPE); 119 | Path path = new File(fileName).toPath(); 120 | String mimeType = Files.probeContentType(path); 121 | fileName = fileUtil.generateUniqueName(fileExtension); 122 | String objectName = folder + fileName; 123 | 124 | minioClient.putObject(PutObjectArgs.builder().bucket(bucketName).object(objectName).stream( 125 | file, file.available(), -1) 126 | .contentType(mimeType) 127 | .build()); 128 | return fileName; 129 | } 130 | 131 | } 132 | -------------------------------------------------------------------------------- /src/main/java/com/java/sahil/minio/service/impl/UserServiceImpl.java: -------------------------------------------------------------------------------- 1 | package com.java.sahil.minio.service.impl; 2 | 3 | import com.java.sahil.minio.dto.PageableResponseDto; 4 | import com.java.sahil.minio.dto.UserRequestDto; 5 | import com.java.sahil.minio.dto.UserResponseDto; 6 | import com.java.sahil.minio.entity.User; 7 | import com.java.sahil.minio.error.EntityNotFoundException; 8 | import com.java.sahil.minio.error.FileCantUploadException; 9 | import com.java.sahil.minio.mapper.UserMapper; 10 | import com.java.sahil.minio.repo.UserRepo; 11 | import com.java.sahil.minio.service.UserService; 12 | import com.java.sahil.minio.util.FileUtil; 13 | import lombok.RequiredArgsConstructor; 14 | import lombok.SneakyThrows; 15 | import lombok.extern.slf4j.Slf4j; 16 | import org.apache.commons.codec.binary.Base64; 17 | import org.springframework.beans.factory.annotation.Autowired; 18 | import org.springframework.beans.factory.annotation.Value; 19 | import org.springframework.data.domain.PageRequest; 20 | import org.springframework.data.domain.Pageable; 21 | import org.springframework.stereotype.Service; 22 | import org.springframework.transaction.PlatformTransactionManager; 23 | import org.springframework.transaction.TransactionStatus; 24 | import org.springframework.transaction.annotation.Transactional; 25 | import org.springframework.transaction.support.DefaultTransactionDefinition; 26 | import org.springframework.web.multipart.MultipartFile; 27 | 28 | import javax.imageio.ImageIO; 29 | import java.awt.Color; 30 | import java.awt.Font; 31 | import java.awt.Graphics2D; 32 | import java.awt.image.BufferedImage; 33 | import java.io.ByteArrayInputStream; 34 | import java.io.ByteArrayOutputStream; 35 | import java.io.InputStream; 36 | import java.util.List; 37 | import java.util.Map; 38 | 39 | import static com.java.sahil.minio.client.DummyClient.getBase64Contract; 40 | import static com.java.sahil.minio.client.DummyClient.getBase64Signature; 41 | import static net.logstash.logback.argument.StructuredArguments.kv; 42 | 43 | @Service 44 | @Slf4j 45 | @RequiredArgsConstructor 46 | public class UserServiceImpl implements UserService { 47 | 48 | private final UserRepo userRepo; 49 | private final FileServiceImpl fileServiceImpl; 50 | private final UserMapper userMapper; 51 | private final FileUtil fileUtil; 52 | @Value("${minio.image-folder}") 53 | private String imageFolder; 54 | @Value("${minio.video-folder}") 55 | private String videoFolder; 56 | @Value("${minio.resume-folder}") 57 | private String resumeFolder; 58 | 59 | @Override 60 | @Transactional 61 | public UserResponseDto create(UserRequestDto userRequestDto) { 62 | log.info("create User started with: {}", kv("userRequestDto", userRequestDto)); 63 | User user = userRepo.save(userMapper.toUserEntity(userRequestDto)); 64 | UserResponseDto userResponseDto = userMapper.toUserDto(user); 65 | log.info("create User completed successfully with: {}", kv("userRequestDto", userRequestDto)); 66 | return userResponseDto; 67 | } 68 | 69 | @Override 70 | @Transactional 71 | public UserResponseDto update(UserRequestDto userRequestDto, Long id) { 72 | log.info("update User started with: {}, {}", kv("id", id), 73 | kv("userRequestDto", userRequestDto)); 74 | User user = userRepo.findById(id).orElseThrow(() -> { 75 | throw new EntityNotFoundException(User.class, id); 76 | }); 77 | user.setName(userRequestDto.getName()); 78 | user.setSurname(userRequestDto.getSurname()); 79 | User saved = userRepo.save(user); 80 | UserResponseDto userResponseDto = userMapper.toUserDto(saved); 81 | log.info("update User completed successfully with: {}, {}", kv("id", id), 82 | kv("userRequestDto", userRequestDto)); 83 | return userResponseDto; 84 | } 85 | 86 | @Override 87 | @Transactional(readOnly = true) 88 | public UserResponseDto findById(Long id) { 89 | log.info("findById User started with: {}", kv("id", id)); 90 | User user = userRepo.findById(id).orElseThrow(() -> { 91 | throw new EntityNotFoundException(User.class, id); 92 | }); 93 | UserResponseDto userResponseDto = userMapper.toUserDto(user); 94 | log.info("findById User completed successfully with: {}", kv("id", id)); 95 | return userResponseDto; 96 | } 97 | 98 | @Override 99 | @Transactional(readOnly = true) 100 | public PageableResponseDto> findAll(int page, int size) { 101 | log.info("findAll User started"); 102 | Pageable pageable = PageRequest.of(page, size); 103 | List userList = userRepo.findAll(pageable).getContent(); 104 | log.info("findAll User completed successfully"); 105 | return new PageableResponseDto<>(userMapper.toUserDtoList(userList), 106 | userRepo.count(), userList.size()); 107 | } 108 | 109 | @Override 110 | @Transactional 111 | public UserResponseDto delete(Long id) { 112 | log.info("delete User started with: {}", kv("id", id)); 113 | User user = userRepo.findById(id).orElseThrow(() -> { 114 | throw new EntityNotFoundException(User.class, id); 115 | }); 116 | if (user.getPhoto() != null) { 117 | deleteFile(user.getPhoto(), imageFolder); 118 | } 119 | if (user.getVideo() != null) { 120 | deleteFile(user.getVideo(), videoFolder); 121 | } 122 | userRepo.delete(user); 123 | UserResponseDto userResponseDto = userMapper.toUserDto(user); 124 | log.info("delete User completed successfully with: {}", kv("id", id)); 125 | return userResponseDto; 126 | } 127 | 128 | /** 129 | * FILE METHODS 130 | */ 131 | 132 | @Autowired 133 | private PlatformTransactionManager transactionManager; 134 | 135 | @Override 136 | public String uploadImage(MultipartFile file, Long id) { 137 | log.info("uploadImage to User started with, {}", 138 | kv("partnerId", id)); 139 | DefaultTransactionDefinition definition = new DefaultTransactionDefinition(); 140 | // definition.setIsolationLevel(TransactionDefinition.PROPAGATION_REQUIRED); 141 | definition.setTimeout(3); 142 | TransactionStatus status = transactionManager.getTransaction(definition); 143 | String fileName = ""; 144 | try { 145 | User user = getUser(id); 146 | if (user.getPhoto() == null) { 147 | fileName = fileServiceImpl.uploadImage(file, imageFolder, true); 148 | user.setPhoto(fileName); 149 | userRepo.save(user); 150 | // if (!fileName.equals("")) throw new RuntimeException("AHAAAAAAAAAAAAAAAAAAA"); 151 | transactionManager.commit(status); 152 | log.info("uploadImage to User completed successfully with {}", 153 | kv("partnerId", id)); 154 | return fileName; 155 | } 156 | } catch (Exception e) { 157 | log.info("Filename: ", fileName); 158 | fileServiceImpl.deleteFile(fileName, imageFolder); 159 | transactionManager.rollback(status); 160 | } 161 | throw new FileCantUploadException(file.getOriginalFilename()); 162 | } 163 | 164 | @Override 165 | @Transactional 166 | public String updateImage(MultipartFile file, Long id) { 167 | log.info("updateImage to User started with, {}", 168 | kv("partnerId", id)); 169 | User user = getUser(id); 170 | deleteFile(user.getPhoto(), imageFolder); 171 | String fileName = fileServiceImpl.uploadImage(file, imageFolder, true); 172 | user.setPhoto(fileName); 173 | userRepo.save(user); 174 | log.info("updateImage to User completed successfully with {}", 175 | kv("partnerId", user)); 176 | return fileName; 177 | } 178 | 179 | @Override 180 | @Transactional 181 | public void deleteUserImage(Long id) { 182 | log.info("deleteUserImage started from User with {}", kv("id", id)); 183 | User user = getUser(id); 184 | if (user.getPhoto() != null) { 185 | fileServiceImpl.deleteFile(user.getPhoto(), imageFolder); 186 | user.setPhoto(null); 187 | userRepo.save(user); 188 | } 189 | log.info("deleteUserImage completed successfully from User with {} ", kv("id", id)); 190 | } 191 | 192 | @Override 193 | public void deleteFile(String fileName, String folder) { 194 | log.info("deleteFile started from User with {}", kv("fileName", fileName)); 195 | fileServiceImpl.deleteFile(fileName, folder); 196 | log.info("deleteFile completed successfully from User with {} ", kv("fileName", fileName)); 197 | } 198 | 199 | @Override 200 | public byte[] getFile(String fileName, String folder) { 201 | log.info("getFile started with {}", kv("fileName", fileName)); 202 | return fileServiceImpl.getFile(fileName, folder); 203 | } 204 | 205 | @Override 206 | @Transactional 207 | public String uploadVideo(MultipartFile file, Long id) { 208 | log.info("uploadVideo to User started with, {}", 209 | kv("partnerId", id)); 210 | User user = getUser(id); 211 | if (user.getVideo() == null) { 212 | String fileName = fileServiceImpl.uploadVideo(file, videoFolder); 213 | user.setVideo(fileName); 214 | userRepo.save(user); 215 | log.info("uploadFile to User completed successfully with {}", 216 | kv("partnerId", id)); 217 | return fileName; 218 | } 219 | throw new FileCantUploadException(file.getOriginalFilename()); 220 | } 221 | 222 | @Override 223 | @Transactional 224 | public String updateVideo(MultipartFile file, Long id) { 225 | log.info("updateVideo to User started with, {}", 226 | kv("partnerId", id)); 227 | User user = getUser(id); 228 | deleteFile(user.getVideo(), videoFolder); 229 | String fileName = fileServiceImpl.uploadVideo(file, videoFolder); 230 | user.setVideo(fileName); 231 | userRepo.save(user); 232 | log.info("updateVideo to User completed successfully with {}", 233 | kv("partnerId", user)); 234 | return fileName; 235 | } 236 | 237 | @Override 238 | @Transactional 239 | public void deleteUserVideo(Long id) { 240 | log.info("deleteUserVideo started from User with {}", kv("id", id)); 241 | User user = getUser(id); 242 | if (user.getPhoto() != null) { 243 | fileServiceImpl.deleteFile(user.getVideo(), videoFolder); 244 | user.setVideo(null); 245 | userRepo.save(user); 246 | } 247 | log.info("deleteUserVideo completed successfully from User with {} ", kv("id", id)); 248 | } 249 | 250 | @SneakyThrows 251 | @Override 252 | @Transactional 253 | public String uploadContractByPin(String dummyPin) { 254 | User user = getUser(1L); 255 | BufferedImage contract = base64ToBufferedImage(getBase64Contract()); 256 | BufferedImage signature = base64ToBufferedImage(getBase64Signature()); 257 | signature = fileUtil.resizeImage(signature, 70, 25); 258 | 259 | BufferedImage combinedImage = new BufferedImage(contract.getWidth(), contract.getHeight(), 260 | BufferedImage.TYPE_INT_ARGB); 261 | Graphics2D g = combinedImage.createGraphics(); 262 | g.drawImage(contract, 0, 0, null); 263 | g.drawImage(signature, 90, 264 | (contract.getHeight() - 40 - signature.getHeight()), null); 265 | g.setPaint(Color.BLACK); 266 | g.setFont(new Font("Serif", Font.ITALIC, 14)); 267 | g.drawString("Sahil Appayev", 60, 82); 268 | g.dispose(); 269 | 270 | log.info("Contract size: {} {}", combinedImage.getWidth(), combinedImage.getHeight()); 271 | log.info("Signature size: {} {}", signature.getWidth(), signature.getHeight()); 272 | 273 | ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream(); 274 | ImageIO.write(combinedImage, "PNG", byteArrayOutputStream); 275 | byte[] bytes = byteArrayOutputStream.toByteArray(); 276 | InputStream file = new ByteArrayInputStream(bytes); 277 | String fileName = fileServiceImpl.uploadInputStreamImage(file, imageFolder); 278 | user.setPhoto(fileName); 279 | userRepo.save(user); 280 | return fileName; 281 | } 282 | 283 | @SneakyThrows 284 | private BufferedImage base64ToBufferedImage(String base64) { 285 | base64 = base64.split("[,]")[1]; 286 | byte[] bytes = Base64.decodeBase64(base64); 287 | return ImageIO.read(new ByteArrayInputStream(bytes)); 288 | } 289 | 290 | @Transactional 291 | public String uploadResume(MultipartFile file, Long id) { 292 | log.info("uploadResume to User started with, {}", 293 | kv("id", id)); 294 | User user = getUser(id); 295 | if (user.getVideo() == null) { 296 | String fileName = fileServiceImpl.uploadPdf(file, resumeFolder); 297 | user.setResume(fileName); 298 | userRepo.save(user); 299 | log.info("uploadResume to User completed successfully with {}", 300 | kv("id", id)); 301 | return fileName; 302 | } 303 | throw new FileCantUploadException(file.getOriginalFilename()); 304 | } 305 | 306 | @Override 307 | @Transactional 308 | public String updateResume(MultipartFile file, Long id) { 309 | log.info("updateResume to User started with, {}", 310 | kv("id", id)); 311 | User user = getUser(id); 312 | deleteFile(user.getResume(), resumeFolder); 313 | String fileName = fileServiceImpl.uploadPdf(file, resumeFolder); 314 | user.setResume(fileName); 315 | userRepo.save(user); 316 | log.info("updateResume to User completed successfully with {}", 317 | kv("id", user)); 318 | return fileName; 319 | } 320 | 321 | private User getUser(Long id) { 322 | return userRepo.findById(id).orElseThrow( 323 | () -> new EntityNotFoundException(User.class, id)); 324 | } 325 | 326 | @Override 327 | @Transactional 328 | public Map uploadPhotos(Long id, String[] names, MultipartFile[] files) { 329 | User user = getUser(id); 330 | int i = 0; 331 | for (MultipartFile file : files) { 332 | String uploadImageName = fileServiceImpl.uploadImage(file, imageFolder, false); 333 | user.getPhotos().put(names[i], uploadImageName); 334 | i++; 335 | } 336 | return user.getPhotos(); 337 | } 338 | 339 | } 340 | -------------------------------------------------------------------------------- /src/main/java/com/java/sahil/minio/util/FileUtil.java: -------------------------------------------------------------------------------- 1 | package com.java.sahil.minio.util; 2 | 3 | import com.java.sahil.minio.error.ExtensionNotAcceptableException; 4 | import lombok.RequiredArgsConstructor; 5 | import lombok.extern.slf4j.Slf4j; 6 | import org.springframework.beans.factory.annotation.Value; 7 | import org.springframework.stereotype.Service; 8 | import org.springframework.web.multipart.MultipartFile; 9 | 10 | import javax.validation.constraints.NotNull; 11 | import java.awt.Image; 12 | import java.awt.image.BufferedImage; 13 | import java.io.IOException; 14 | import java.io.InputStream; 15 | import java.net.URLConnection; 16 | import java.util.Date; 17 | 18 | @Slf4j 19 | @Service 20 | @RequiredArgsConstructor 21 | public class FileUtil { 22 | @Value("${file.upload.acceptableVideoExtensions}") 23 | private String[] acceptableVideoExtensions; 24 | @Value("${file.upload.acceptableImageExtensions}") 25 | private String[] acceptableImageExtensions; 26 | @Value("${file.upload.acceptablePdfExtensions}") 27 | private String[] acceptablePdfExtensions; 28 | 29 | private boolean isExtensionAcceptable(String extension, String mediaType) { 30 | if (mediaType.equals("image")) { 31 | for (String s : acceptableImageExtensions) { 32 | if (s.equalsIgnoreCase(extension)) { 33 | return true; 34 | } 35 | } 36 | } else if (mediaType.equals("video")) { 37 | for (String s : acceptableVideoExtensions) { 38 | if (s.equalsIgnoreCase(extension)) { 39 | return true; 40 | } 41 | } 42 | } else if (mediaType.equals("pdf")) { 43 | for (String s : acceptablePdfExtensions) { 44 | if (s.equalsIgnoreCase(extension)) { 45 | return true; 46 | } 47 | } 48 | } 49 | return false; 50 | } 51 | 52 | public String getFileExtensionIfAcceptable(@NotNull MultipartFile file, String mediaType) { 53 | String extension = file.getContentType().split("[/]")[1]; 54 | if (isExtensionAcceptable(extension, mediaType)) { 55 | return extension; 56 | } else { 57 | throw new ExtensionNotAcceptableException(extension); 58 | } 59 | } 60 | 61 | public String getFileExtensionIfAcceptable(String fileName, String mediaType) { 62 | String extension = fileName.split("\\.")[1]; 63 | if (isExtensionAcceptable(extension, mediaType)) { 64 | return extension; 65 | } else { 66 | throw new ExtensionNotAcceptableException(extension); 67 | } 68 | } 69 | 70 | public String generateUniqueName(String extension) { 71 | Date date = new Date(); 72 | return date.getTime() + "." + extension; 73 | } 74 | 75 | public String getFileExtensionFromInputStream(InputStream inputStream) { 76 | String fileExtension = null; 77 | try { 78 | String mimeType = URLConnection.guessContentTypeFromStream(inputStream); 79 | String[] tokens = mimeType.split("[/]"); 80 | fileExtension = tokens[1]; 81 | } catch (IOException ioException) { 82 | log.error("Error occurred: {}", ioException.getMessage()); 83 | } 84 | return fileExtension; 85 | } 86 | 87 | public BufferedImage resizeImage(BufferedImage originalImage, int targetWidth, int targetHeight) { 88 | Image resultingImage = originalImage.getScaledInstance(targetWidth, targetHeight, Image.SCALE_DEFAULT); 89 | BufferedImage outputImage = new BufferedImage(targetWidth, targetHeight, BufferedImage.TYPE_INT_RGB); 90 | outputImage.getGraphics().drawImage(resultingImage, 0, 0, null); 91 | return outputImage; 92 | } 93 | 94 | } 95 | -------------------------------------------------------------------------------- /src/main/resources/application.yml: -------------------------------------------------------------------------------- 1 | spring: 2 | h2: 3 | console: 4 | enabled: true 5 | jpa: 6 | open-in-view: false 7 | defer-datasource-initialization: true 8 | show-sql: true 9 | servlet: 10 | multipart: 11 | enabled: true 12 | file-size-threshold: 2KB 13 | max-file-size: 500MB 14 | max-request-size: 1000MB 15 | file: 16 | upload: 17 | acceptableVideoExtensions: mp4,mov,avi,wmv 18 | acceptableImageExtensions: jpg,png,jpeg 19 | acceptablePdfExtensions: pdf 20 | 21 | minio: 22 | url: http://127.0.0.1:9000/ 23 | bucket: user 24 | access-key: test 25 | secret-key: test1234 26 | image-folder: profpic/ 27 | video-folder: video/ 28 | resume-folder: resume/ 29 | application: 30 | swagger: 31 | title: MinIo Photo API 32 | description: Spring Boot file uploading with MinIo file storage. 33 | version: 0.0.2 -------------------------------------------------------------------------------- /src/main/resources/banner.txt: -------------------------------------------------------------------------------- 1 | 2 | ____ _ ____ _ __ __ _ ___ ___ 3 | / ___| _ __ _ __(_)_ __ __ _ | __ ) ___ ___ | |_ | \/ (_)_ __ |_ _/ _ \ 4 | \___ \| '_ \| '__| | '_ \ / _` | | _ \ / _ \ / _ \| __| _____ | |\/| | | '_ \ | | | | | 5 | ___) | |_) | | | | | | | (_| | | |_) | (_) | (_) | |_ |_____| | | | | | | | || | |_| | 6 | |____/| .__/|_| |_|_| |_|\__, | |____/ \___/ \___/ \__| |_| |_|_|_| |_|___\___/ 7 | |_| |___/ 8 | 9 | ${AnsiColor.BRIGHT_RED}:: Running Spring Boot ${spring-boot.version} :: 10 | -------------------------------------------------------------------------------- /src/main/resources/base64Contract.txt: -------------------------------------------------------------------------------- 1 |  -------------------------------------------------------------------------------- /src/main/resources/base64Signature.txt: -------------------------------------------------------------------------------- 1 |  -------------------------------------------------------------------------------- /src/main/resources/data.sql: -------------------------------------------------------------------------------- 1 | insert into user (name, surname) 2 | values ('Sahil', 'Appayev'); -------------------------------------------------------------------------------- /src/main/resources/docker-compose.yml: -------------------------------------------------------------------------------- 1 | version: '3' 2 | services: 3 | minio: 4 | image: minio/minio 5 | ports: 6 | - "9000:9000" 7 | - "9001:9001" 8 | environment: 9 | MINIO_ACCESS_KEY: test 10 | MINIO_SECRET_KEY: test1234 11 | command: server --console-address :9001 /data 12 | createbuckets: 13 | image: minio/mc 14 | depends_on: 15 | - minio 16 | entrypoint: > 17 | /bin/sh -c " 18 | /usr/bin/mc alias set myminio http://minio:9000 test test1234; 19 | /usr/bin/mc mb myminio/user; 20 | /usr/bin/mc policy set public myminio/user; 21 | exit 0; 22 | " -------------------------------------------------------------------------------- /src/main/resources/messages.properties: -------------------------------------------------------------------------------- 1 | error.notFound=is not found! 2 | error.extensionNotAcceptable=extension is not acceptable! --------------------------------------------------------------------------------