원글 페이지 : 바로가기
S3 업로드 기능 구현하기 현재 진행하는 프로젝트에는 이미지를 업로드할 일이 굉장히 많다 아티스트나 소속사의 프로필부터 시작해서 소통을 위한 아티스트 커뮤니티, 공지사항을 생성할 때도 필요하다 확인 필요 enterfeed의 경우 공지에 이미지가 들어가는 건 알겠는데, 스케줄도 이미지 들어가는지? * 둘 다 들어갈 경우 파일 저장 경로를 다르게 해야하므로 create 메서드 수정 필요 -> 아니오 이미지 최대 용량? -> 대포카메라 4000*6000 기준 7~8MB 이므로 장 당 10MB, 여러장 받을 경우 최대 용량 100MB 이미지만 받을 건지? * 그렇다면 이미지파일이 아닌 경우 업로드 불가 안내 (필터링) -> ok 한 장만 받을 건지, 여러장 받을 건지? -> 프로필/로고 한 장(10MB), 게시글 여러 장( 100MB ) -> 설정 필요 // 파일 장수 제한 (예: 최대 10장)
if (files != null && files.size() > 10) {
throw new FileUploadException(“최대 5개의 파일만 업로드할 수 있습니다.”);
}
// 파일 장수 제한 (예: 최대 1장) //프로필이나 로고
if (files != null && files.size() > 10) {
throw new FileUploadException(“최대 5개의 파일만 업로드할 수 있습니다.”);
}
// 파일 용량 제한 (예: 개별 파일 최대 10MB)
if (files != null) {
for (MultipartFile file : files) {
if (file.getSize() > 10 * 1024 * 1024) {
throw new FileUploadException(“파일 크기는 2MB를 초과할 수 없습니다.”);
}
}
} 이미지 확장자 어디까지 받을 것인지? jpg, jpeg, png, webp, gif, bmp, tiff, ppm, pgm, pbm, pnm -> 모두 받기로 함 절차 요약 메서드상 이미지 받기 -> 객체에 기본정보 저장 -> s3에 이미지 저장 -> 객체에도 그 경로 저장 결정 사항 파일명 중복방지를 위한 방법 파일명을 uuid 를 사용할 것인지 vs. 날짜데이터 등으로 넣을 것인지 -> uuid 파일을 Dto 로 받을 것인지 @RequestPart 로 받을 것인지 -> @RequestPart 저장할 파일 경로 -> 각 패키지명 Dir 더 세부적인 파일 경로 설정필요 enter 의 로고가 Group1, Group2 와 같은 위치에 놓이도록 할지 아니면 enter 끼리만 모아놓도록 할지?? -> 이런 꼴로 만들기로 결정 업로드 원본 이름을 저장할 것인지 말 것인지 -> 저장하기로 함 유저프로필의 경우 디렉토리가 USER 로 시작은 하는데, 모든 유저를 한 폴더에 모아 놓을 수 없는 상황.. 어떻게 할 것인지 결정해야 함 (가입 날짜별 / 프로필 사진 등록 날짜 / 기타) -> 정하기 힘들어서 튜터님께 조언 받음 불러올 때 부하가 있을 거라는 걱정은 안 해도 된다고 하심 20만개까지도 성능상 큰 이슈가 없는 경험이 있다고 하셔서 그대로 User 디렉토리 하나에 다 모아 놓기로 결정함 필수 내용 업로드 파일이름과 저장된 파일이름 구별하여 저장?? -> 특수문자가 들어있을 수도 있어 원본 이름은 삭제해야할지도 이미지 파일인지 다른 파일인지 확인 1. 제목으로 ok 2. contentType 추출로 리스트로 받았을 때, 처리하기 -> 프로필 제외하면 다 리스트로 받기 @RequestPart 로 받을 때 주의점 업로드한 이미지에 대해 읽거나 쓸 수 있는 권한 제한 public enum CannedAccessControlList {
Private(“private”), //기본설정. 객체 소유자만 읽기/쓰기 가능
PublicRead(“public-read”), //모든 사용자가 읽기 가능, 소유자만 쓰기 가능
PublicReadWrite(“public-read-write”), //모든 사용자가 읽기/쓰기 가능 (권장x)
AuthenticatedRead(“authenticated-read”), // AWS 계정에 인증된 사용자만 읽기 가능
LogDeliveryWrite(“log-delivery-write”), // S3 서버 접근 로그를 버킷에 기록하는 데 사용
BucketOwnerRead(“bucket-owner-read”),//객체 소유자 달라도 버킷 소유자가 읽을 수 있음
BucketOwnerFullControl(“bucket-owner-full-control”), //버킷 소유자가 객체에 대해 읽기/쓰기 가능
AwsExecRead(“aws-exec-read”); //AWS에서 실행권한 갖음 -> publicRead 로 결정 공부가 필요한 사항 인터페이스의 메서드 사용이 그냥 가능한 건지 -> 누군가 만들어둔 결과를 보니 그냥 사용하고 있어서.. 예) Multipartfile.getContentType(); Spring Framework는 MultipartFile 인터페이스의 여러 구현체를 제공하며, 가장 일반적으로 사용하는 구현체는 CommonsMultipartFile과 StandardMultipartFile입니다. 이 구현체들은 Spring의 파일 업로드 처리 로직에서 자동으로 생성됩니다. Spring이 자동으로 MultipartFile 인터페이스를 구현한 인스턴스를 제공합니다. 따라서 file.getContentType() 메서드를 직접 호출할 수 있습니다. MultipartFile의 getInputStream() 메서드 파일의 데이터에 접근하기 위해 InputStream 객체를 얻는 데 사용됩니다. InputStream은 파일, 네트워크, 메모리 등 다양한 데이터 소스에서 데이터를 바이트 스트림 형태로 읽을 수 있게 해줍니다. 파일의 정보를 읽어서 그 내용을 다른 곳에서 사용(조작)하고자 할 때 사용함 예) 해상도를 줄여서 썸네일을 만들 때 등 Key 란? putObjectRequest 는 key 를 매개 변수로 받는다 key 는 객체를 버킷 내에서 식별하는 고유한 키로, 객체의 경로와 유사한 역할을 하며, 버킷 내에서 객체를 찾는 데 사용 예를 들어, “folder1/folder2/myfile.txt”와 같이 설정하면, folder1이라는 폴더 안에 folder2라는 하위 폴더가 있고, 그 안에 myfile.txt라는 파일이 있는 구조를 표현할 수 있습니다. new PutObjectRequest(bucketName, key, file); 의 경우, bucketName 버킷의 key 위치에 file을 업로드 함 public PutObjectRequest(String bucketName, String key, File file) {
super(bucketName, key, file);
}
public PutObjectRequest(String bucketName, String key, String redirectLocation) {
super(bucketName, key, redirectLocation);
}
public PutObjectRequest(String bucketName, String key, InputStream input, ObjectMetadata metadata) {
super(bucketName, key, input, metadata); PutObjectRequest 생성자는 세 가지가 있다 로컬 파일을 직접 S3 버킷으로 업로드할 때 사용 – 간결함, 파일 스트림의 직접 조작 없을 때 S3 객체를 리디렉션할 때 사용되며, 이 객체를 요청하면 지정된 redirectLocation으로 리디렉션됨 – 리디렉션 할 때 스트림 형태로 파일 내용을 전송하며, 파일의 메타데이터를 설정할 때 사용 – 대용량이나 스트리밍 방식에 유용, 파일 내용을 직접 읽는 대신 스트림을 사용 -> 첫 번째로 결정 그런데 매개변수부분에 오류가 난다 그래서 찾아보니, pubObject() 메소드에 file을 매개변수로 넘겨주는 경우에는 실제 파일이 존재해야 하기 때문에 file을 create해주어야 합니다. 이 방식은 큰 문제를 가지고 있는데요. 첫째로 파일쓰기 작업이 일어난다는 점입니다. 파일 쓰기 작업은 매우 무거운 작업입니다. 두번째로 업로드할 곳은 S3인데 로컬에도 파일이 저장된다는 점입니다. 이는 쓸데없는 자원을 낭비하며, 이 때문에, 별도로 file을 지우는 작업에 대한 고려를 하게됩니다 라고 한다 그래서 경로 수정 -> 번째로 방식으로 진행하기로 했다 만약 그대로 첫 번째를 이용할 경우에는 임시파일을 이용해야 한다고 한다 //윗부분 생략
File tempFile = convertMultipartFileToFile(file);
try {
amazonS3Client.putObject(new PutObjectRequest(bucket, uploadName, tempFile)
.withCannedAcl(CannedAccessControlList.PublicRead));
} catch (Exception e) {
throw new S3Exception(UPLOAD_ERROR);
} finally {
// 업로드 후 임시 파일 삭제 tempFile.delete();
}
}
private File convertMultipartFileToFile(MultipartFile file) throws IOException {
File tempFile = File.createTempFile(“temp”, null);
file.transferTo(tempFile); return tempFile;
} 이와같이 임시파일을 삭제하는 로직까지 필요하다 boolean allMatch(Predicate super T> predicate); //전부 같을 경우 참
boolean anyMatch(Predicate super T> predicate); //하나라도 같을경우 참 @RequestPart + @RequestBody 는 불가. @RequestPart + @RequestPart 로 받아야 함 따라서 @PathVariable final String groupName,
@RequestPart(value = “file”, required = false) List
@AuthenticationPrincipal UserDetailsImpl userDetails,
@Valid @RequestBody final CreateFeedRequestDto requestDto 에서 @PathVariable final String groupName,
@RequestPart(value = “file”, required = false) List
@AuthenticationPrincipal UserDetailsImpl userDetails, //변경된 부분
@Valid @RequestPart final CreateFeedRequestDto requestDto 로 변경함 난관 이미 feed 가 생성되는 기능을 구현해 놓은 상태이며, 이미지 업로드 기능을 추가하여, feed 를 생성할 때 이미지도 들어가게 하려고 하는 중이다 그런데, feedId 에 맞춰서 feed의 이미지가 저장되는 Dir 경로를 설정해 두었더니 1. 이미지를 먼저 받고자 했더니, feedId가 있어야 이미지를 저장할 수 있고, 그 url을 받을 수가 없다 2. feed 먼저 받고자 했더니, setter 를 feed에 사용해야 url을 넣을 수 있다 -> feedId 가 먼저 생성되는게 여러장의 이미지를 올렸을 때, 한 곳에 모아두기 좋다 feedId 를 먼저 받고싶고, setter 는 사용하고 싶지 않았다 그래서 찾아보다가 객체를 하나 새로 생성하는 방법을 알게 되었다 어떤 패키지에 만들어줄까 하다가 dto 패키지에 support로 집어넣었다 왜냐하면 Dto와 역할이 유사하기 때문이다 => support 는 보통 service 패키지에서 service에서 분리된 메서드들을 담는 곳으로 쓰이므로 다른 역할을 가진 지금의 것에는 어울리지 않아서 Carrier 라고 이미지url 과 이미지를 넣는 객체의 id를 넣어 전달하는 역할을 하는 객체로 변경함 feed 엔티티에는 List
private List
return Artist.builder()
.artistName(requestDto.getArtistName())
.artistProfileUrl(url)
.user(loginUser)
.build(); file 타입이 바로 url이 만들어질 수가 없으니 말이다 처음 생성되는 필드는 null을 채우기로 했다 어차피 불러와서 사용하는 일이 없으므로 괜찮기 때문이다 프론트 연결 정보 참고 http://dev-gorany.tistory.com/125 [Spring] 업로드한 파일의 Content-Type 첨부파일을 다루다 보면, 파일의 종류, 확장자에 따라 다르게 처리해야할 경우가 있다. 이 때 업로드된 파일의 종류를 알아보려면 Content-Type이 무엇인지 알면 간단하게 판단할 수 있다. Content-Type dev-gorany.tistory.com https://jj-yi.tistory.com/30 [Spring Boot] S3 이미지 업로드 Spring Boot로 S3에 이미지 업로드 하기 Ver.1 의존성을 먼저 추가해주어야 합니다 // AWS S3 compile group: ‘org.springframework.cloud’, name: ‘spring-cloud-aws’, version: ‘2.2.1.RELEASE’, ext: ‘pom’ // https://mvnrepository.com/artifac jj-yi.tistory.com https://galid1.tistory.com/591 AWS SDK – JAVA를 이용해 S3에 파일 업로드시 로컬에 저장되지 않도록 하기 이번 포스팅에는 사용자의 업로드 요청시 로컬 스토리지가 아닌 AWS의 S3에 저장하는데, 로컬에는 파일이 저장되지 않은 채로 바로 업로드 하는 방법을 알아보도록 하겠습니다 이번 포스팅에는 galid1.tistory.com https://velog.io/@zvyg1023/Spring-Boot-RequestPart%EB%A5%BC-%EC%9D%B4%EC%9A%A9%ED%95%9C-MultipartFile-DTO-%EC%B2%98%EB%A6%AC-%EB%B0%8F-%ED%85%8C%EC%8A%A4%ED%8A%B8 [Spring] @RequestPart를 이용한 MultipartFile, DTO 처리 및 테스트 문제 – @RequestParam + @RequestBody 쇼핑몰 토이 프로젝트 중 해당 품목의 이미지를 첨부하고 품목 정보를 입력해 아이템을 등록하는 기능을 구현하려고* MultipartFile은 @RequestParam으로 Dto는 @RequestBody를 velog.io https://velog.io/@kseysh/ElementCollection%EC%9D%98-%EA%B0%9C%EB%85%90%EA%B3%BC-%EC%A3%BC%EC%9D%98%ED%95%A0-%EC%A0%90 @ElementCollection의 개념과 주의할 점! @ElementCollection에서 발생한 문제들을 해결하고 개념을 확실히 잡아보아요 velog.io https://dangdangee.tistory.com/entry/JPA-ElementCollection-CollectionTable%EC%9D%84-%ED%86%B5%ED%95%9C-%EA%B0%92-%ED%83%80%EC%9E%85-%EC%BB%AC%EB%A0%89%EC%85%98-%EC%82%AC%EC%9A%A9%EB%B2%95 [JPA] @ElementCollection, @CollectionTable을 통한 값 타입 컬렉션 사용법 값 타입 int, Integer, String처럼 단순히 값으로 사용하는 자바 기본 타입이나 객체이다. 값타입 분류 기본값 타입 자바 기본 타입(int, double) 래퍼 클래스(Integer, Long) String 임베디드 타입(embedded type, dangdangee.tistory.com https://velog.io/@rookieand/MIME-type%EC%9D%80-%EB%AD%90%EA%B3%A0-Content-type%EC%9D%80-%EB%AD%94%EB%8D%B0 MIME type은 뭐고, Content-type은 뭔데? 내가 아는 MIME 이라고는 인터넷에서 나도는 그 밈밖에 몰랐는데… velog.io https://velog.io/@marbea6282/%EC%9D%B4%EB%AF%B8%EC%A7%80-%EC%97%85%EB%A1%9C%EB%93%9C 이미지 업로드 기능 구현 파일 업로드를 위해서 사용하는 라이브러리는 COS.jar, Apache Commons FileUpload, Servlet(3.0이후 version)의 Part(API) 등이 있으며 본 프로젝트에서는 Apache Commons FileUpload를 사용하였다.프로젝트 빌 velog.io https://inpa.tistory.com/entry/REDIS-%F0%9F%93%9A-Window10-%ED%99%98%EA%B2%BD%EC%97%90-Redis-%EC%84%A4%EC%B9%98%ED%95%98%EA%B8%B0 [REDIS] 📚 Window10 환경에 Redis 설치 & 설정 Redis 윈도우 설치 Redis 다운로드 페이지로 이동하여 설치 프로그램을 다운로드하고 설치를 진행한다. Releases · microsoftarchive/redis Redis is an in-memory database that persists on disk. The data model is key-value, but inpa.tistory.com https://emoney96.tistory.com/258 Postman을 이용한 File, Dto 동시 Post요청 보통 Controller에서 Dto를 받을 때는 @RequestBody를 주로 사용합니다. 그리고 File을 받을 때는 MultipartFile 객체를 사용하며, @RequestParam을 사용합니다. 하지만 File과 Dto를 같이 받기 위해서는 @RequestPart라 whyeskang.com https://hatpub.tistory.com/81 Redis 간단한 명령어 (추가, 삭제) 저는 Redis 를 Lock 을 걸때 .. 그러니까 하나의 프로그램을 실행 하고 마칠 때 까지 중복 실행이 안되게끔 하기 위해서 Redis 를 쓰고 있습니다. 1. 서버 접속 명령어 redis-cli docker (docker-compose) 에서 red hatpub.tistory.com https://chb2005.tistory.com/200#3.4.%20%ED%8C%8C%EC%9D%BC%20%EC%97%85%EB%A1%9C%EB%93%9C%20%EA%B5%AC%ED%98%84 [Spring Boot] AWS S3를 이용한 파일 업로드 AWS S3 란? AWS Simple Storage Service의 줄임말로 파일 서버의 역할을 하는 서비스 프로젝트 개발 중 파일을 저장하고 불러오는 작업이 필요한 경우에 프로젝트 내부 폴더에 저장할 수 있지만, AWS S3를 chb2005.tistory.com