◼︎ 환경
- Model : MacBook Pro (14-inch, 2021)
- CPU : Apple M1 Pro
- MENORY : 16GB
- DISK : 512 GB SSD
- OS : macOS 13.2.4 (22F66)
- TOOLS : Visual Studio Code, Java 11, Gradle, Docker
- Version Control : GitHub
- Programming Language : Java, Vue3
- Front-End Framework : Vue 3.4.30, Vuetify 3.6.10, ag-grid-vue3 31.3.2
- Back-End Framework : Spring Boot 2.7.12, Spring Security 5.7.7
- DBMS : MySql 8.0.33
- Cloud : OCI (free tier account
1. 업로드 라이브러리 & 모듈 조사
Vue 3 환경에서 파일업로드 구현을 위한 인기있는 라이브러리 또는 모듈들에는 여러 가지가 있다. 추가로 사용성 확인을 위하여 npmjs.com 사이트를 활용하였다.
- Weekly Downloads: 363,089
- Version: 6.0.0-beta.2
- Last Publish: 3년 전
- 특징: 가장 많이 사용되는 파일 업로드 라이브러리 중 하나이며, Vue.js 에서도 널리 사용되고 있음. 하지만, Vue 3 공식 지원이 부족할 수 있음.
- Weekly Downloads: 10,754
- Version: 2.2.1
- Last Publish: 8개월 전
- 특징: Dropzone.js의 Vue 3 통합 버전으로, Vue 3 프로젝트에서 드래그 앤 드롭 파일 업로드를 쉽게 구현할 수 있음.
- Weekly Downloads: 82,508
- Version: 2.8.9
- Last Publish: 2달 전
- 특징: 이미지 크롭 기능을 제공하는 파일 업로드 라이브러리로, Vue 3와의 호환성이 높고, 비교적 최근에 업데이트되었음.
➃ Uppy
- Weekly Downloads: 23,383
- Version: 4.1.0
- Last Publish: 10일 전
- 특징: 강력한 모듈화 파일 업로드 라이브러리로, 여러 소스에서 파일을 업로드할 수 있으며, Vue 3와의 호환성도 좋음.
- Weekly Downloads: 21,154
- Version: 7.0.4
- Last Publish: 1년 전
- 특징: 사용자 친화적인 인터페이스를 제공하며, 다양한 파일 포맷을 지원. 다만, 최근에 업데이트가 없음.
➅ Vue Uploader
- Weekly Downloads: 115
- Version: 3.37.1
- Last Publish: 1년 전
- 특징: 다운로드 수가 비교적 적으며, 최근 업데이트가 없는 편임. 작은 프로젝트나 간단한 용도로 적합할 수 있음.
간단하게 요약해보면 아래와 같다.
- 가장 많이 사용되는 라이브러리: Dropzone은 여전히 가장 많은 다운로드 수를 기록하고 있으며, 널리 사용되고 있다. 하지만 Vue 3에 대한 직접적인 지원이 부족할 수 있다.
- Vue 3 에 적합한 라이브러리: Vue Advanced Cropper, Uppy, Vue FilePond, Vue3 Dropzone 등은 Vue 3에서 사용할 수 있는 좋은 선택이다.
개인적으로는 Vue3 Dropzone, Vue FilePond 은 사용해 보았고 이번에는 대용량 파일 업로드가 강점이라고 하는 Uppy 을 사용해 보았다.
2. Uppy
Uppy는 강력하고 모듈화된 파일 업로드 라이브러리로, 다양한 파일 소스에서 업로드를 지원하며 파일 업로드 경험을 쉽게 관리할 수 있도록 설계되었다. Uppy는 최신 웹 기술을 활용하며, 파일 업로드를 직관적으로 구현할 수 있도록 다양한 플러그인을 제공한다. Vue.js와의 통합도 지원하여, Vue 프로젝트에서 Uppy의 기능을 쉽게 사용할 수 있다.
- 모듈식 구조: 필요한 기능만 선택적으로 사용할 수 있도록 다양한 플러그인으로 구성되어 있다. 예를 들어, 파일 드래그 앤 드롭, 웹캠, Google Drive, Dropbox 등 다양한 소스를 지원한다.
다양한 파일 소스: 로컬 파일뿐만 아니라 원격 소스(예: Google Drive, Dropbox, Instagram)에서도 파일을 업로드할 수 있다.
Vue.js 통합:
@uppy/vue
패키지를 통해 Vue.js 프로젝트에 쉽게 통합할 수 있다.커스터마이징 가능: 업로드 UI와 동작을 프로젝트의 요구에 맞게 커스터마이징할 수 있다. 또한, 업로드 진행률 표시, 취소, 재시도 등의 기능도 제공. ⇨ UI 커스터마이징이 비교적 용의했다.
3. Uppy 설치
Uppy와 Vue 통합 패키지를 설치한다.
npm install @uppy/core @uppy/dashboard @uppy/drag-drop @uppy/file-input @uppy/progress-bar
@uppy/vue @uppy/xhr-upload --save
4. Uppy 을 이용한 파일 업로드
업로드 구현은 아래와 같은 순서로 코딩하면 된다.
❶ Uppy 컴포넌트 임포트
<script setup lang="ts">
// import upload component(uppy)
import Uppy from '@uppy/core';
import { Dashboard } from '@uppy/vue';
import XHRUpload from '@uppy/xhr-upload';
import '@uppy/core/dist/style.css';
import '@uppy/dashboard/dist/style.css';
❷ Vue 컴포넌트에서
Dashboard
사용<div class="dashboard-container">
<Dashboard :uppy="uppy"
inline="true"
:height="200"
:note="'Images only, up to 10MB'"
:metaFields="metaFields" />
</div>
@uppy/vue에서 가져온 Dashboard 컴포넌트를 사용하여 Vue 템플릿에서 Uppy 대시보드를 직접 렌더링. inline 속성을 사용 대시보드를 <div> 요소에 직접 렌더링 되도록 설정.
metaFields 을 사용하여 업로드할 파일과 관련된 추가적인 메타데이터를 수집하고, 사용자로부터 입력받을 수 있게 한다. 이를 통해 파일에 대한 추가 정보를 서버로 전송하거나, 파일 처리 로직에서 메타데이터를 활용할 수 있다. → 이부분은 확인 할 수 없었다.
❸ Uppy 인스턴스를 생성하고 파일업로드 플러그인을 설정
const fileUploadUrl = computed(() => {
return `${import.meta.env.VITE_API_URL}/data/secure/mgmt/resources/images/${props.imageId}/upload`
})
const uppy = new Uppy({
autoProceed: false, // 수동으로 업로드 시작
restrictions: {
maxFileSize: 10000000, // 1최대 파일 크기: 10MB
maxNumberOfFiles: 5, // 업로드 가능한 최대 파일 수
minNumberOfFiles: 1, // 업로드 가능한 최소 파일 수
allowedFileTypes: ['image/*']
}
})
.use(XHRUpload, {
endpoint: fileUploadUrl.value, // 업로드할 서버 엔드포인트
fieldName: 'file', // 서버에 전송되는 파일의 필드 이름
formData: true, // FormData를 사용하여 파일을 업로드 (멀티파트 사용)
headers: {
...authHeader(),
},
retryDelays: [0, 1000, 3000, 5000]
});
참로로 authHeader 는 JWT 토큰 관련 값을 정의하는 내용이다. 이런 방식으로 헤더에 값을 추가할 수 있다.
Uppy 는 여러가지 다수의 업로드 방식을 지원하고 있고 여기에서는 가장 일반적인 XHRUpload 을 적용했다.
- XHRUpload: 가장 일반적인 파일 업로드 방식으로, 대부분의 서버에서 지원.
- Tus: 대규모 파일 업로드 및 네트워크 문제로 인한 업로드 중단 시 재개를 지원하는 프로토콜
- Multipart: multipart/form-data 형식으로 서버에 파일을 전송.
- S3 Multipart: AWS S3에 대규모 파일을 멀티파트 방식으로 업로드.
- Transloadit: 클라우드 기반 업로드 서비스와 통합
파일을 업로드 할때 추가 정보를 전송하려면 uppy 의 upload 이벤트를 사용하여 추가하면된다. 이경우 멀티파트 파마메터 형식으로 추가된 값들이 전달된다.
uppy.on('upload', (data) => {
uppy.setMeta({
objectId: dataRef.value.objectId ,
objectType: dataRef.value.objectType,
link: createSharedLink.value,
description: dataRef.value.description
});
});
❺ onBeforeUnmount 을 사용 컴포넌트가 파괴되기 전에 Uppy 인스턴스를 정리
onBeforeUnmount(() => {
uppy.destroy();
});
❻ 업로드 영역 크기 조정하기
Dashboard 의 height 을 사용해도 높이가 조정 되지 않기 떄문에 CSS 을 아래와 같은 방식으로 수정해 적용해야 한다.
<style scoped>
::v-deep .uppy-Dashboard {
height: 250px; /* 원하는 높이로 설정 */
max-height: 100%;
display: flex;
flex-direction: column;
}
</style>
![]() |
그림2. 파일을 추가한 화면 |
![]() |
그림3. 파일 업로드가 성공한 경우 |
➐ 서버 프로그램
XHRUpload 업로드 모듈의 경우 formData 옵션을 사용하면 multipart/form-data 형식으로 서버에 전송되기 때문에 기존 업로드 프로그램을 사용하여 업로드를 구현할 수 있다.
@PostMapping(value = { "/images/{imageId:[\\p{Digit}]+}/upload" }, produces = MediaType.APPLICATION_JSON_VALUE)
public List<Image> upload(
@PathVariable Long imageId,
@RequestParam(value = "objectType", defaultValue = "-1", required = false) Integer objectType,
@RequestParam(value = "objectId", defaultValue = "-1", required = false) Long objectId,
@RequestParam(value = "description", required = false) String description,
@RequestParam(value = "link", defaultValue = "false", required = false) Boolean createLink,
@RequestParam("file") List<MultipartFile> files) throws NotFoundException, IOException, UnAuthorizedException {
User user = SecurityHelper.getUser();
List<Image> list = new ArrayList<>();
for (MultipartFile mpf : files) {
String fileName = StringUtils.cleanPath(mpf.getOriginalFilename());
InputStream is = mpf.getInputStream();
log.debug("upload <file name:{}, size:{}, type:{}>", fileName, mpf.getSize(), mpf.getContentType());
// 업로드 파일을 처리한다.
list.add(newImage);
}
return list;
}
5. 대용량 파일 업로드
Uppy 에서 대용량 파일 업로드는 Tus 프로토콜을 사용하여 구현 할 수 있다. Tus는 대규모 파일 업로드를 효율적으로 처리할 수 있도록 설계된 오픈 프로토콜로 Uppy는 Tus를 통해 파일을 청크 단위로 업로드하고, 중단된 업로드를 재개하는 등의 기능을 제공한다.npm install @uppy/tus
Uppy 에서는 간단하게 XHRUpload 파일 업로드 모듈을 Tus 로 변경해주면 된다.
import Tus from '@uppy/tus';
const tusFileUploadUrl = computed(()=>{
return `${import.meta.env.VITE_API_URL}/data/secure/mgmt/resources/files/${props.imageId}/tus`;
})
uppy.use(Tus, {
endpoint: tusFileUploadUrl.value, // Tus 서버 엔드포인트
chunkSize: 5 * 1024 * 1024, // 5MB 청크 사이즈
headers: {
...authHeader(),
},
retryDelays: [0, 1000, 3000, 5000]
})
;
클라이언트 부분과 다르게 서버는 Tus 프로토콜에 따른 서버 프로그램을 새로 구현해야한다.
Tus 프로토콜
Tus 프로토콜은 클라이언트와 서버 간 4단계로 파일을 업로드합니다.
❶ 파일 생성(Create):
- 클라이언트는 서버에 새로운 업로드를 생성하도록 요청한다.
- 이 요청에는 파일의 크기와 메타데이터가 포함될 수 있다.
- 서버는 업로드에 고유한 URL을 생성하고 클라이언트에게 반환한다.
- 파일 및 메타 정보는 이때만 전송된다. 이후에는 데이터만 전송된다.
❷ 파일 업로드(Upload):
- 클라이언트는 생성된 URL을 사용하여 파일의 특정 바이트를 서버에 업로드한다.
- 서버는 클라이언트가 업로드한 바이트를 확인하고, 다음 업로드 시 어디서부터 시작할지 오프셋(Upload-Offset)을 반환한다.
- 업로드 중단 시 클라이언트는 서버에서 제공한 오프셋을 사용하여 중단된 지점부터 업로드를 재개할 수 있다.
- 모든 바이트가 성공적으로 업로드되면 업로드가 완료된다.
프로토콜 메시지
- OPTIONS: 서버가 Tus 프로토콜을 지원하는지 확인하고, 지원하는 기능을 확인하기 위해 클라이언트가 전송. 서버는 지원하는 버전, 최대 파일 크기 등의 정보를 응답.
- POST: 새로운 업로드를 생성하기 위해 사용. 서버는 업로드 URL을 응답합니다. ❶
- HEAD:업로드 상태를 확인하기 위해 사용. 서버는 현재 업로드된 바이트의 오프셋을 응답.
- PATCH: 파일의 특정 바이트를 업로드하기 위해 사용. 서버는 업로드된 바이트를 저장하고, 새로운 오프셋을 응답. ❷❸
- DELETE:업로드를 취소하거나 삭제하기 위해 사용. 서버는 업로드된 데이터를 삭제합니다.
다음은 파일이 업로드되면 유니크한 아이디를 생성하고 메타 정보는 meta.json 형태로 저장하여 청크파일 업로드를 구현하는 예이다.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
package architecture.web.spring.web.controller.data.upload; | |
import java.io.File; | |
import java.io.FileOutputStream; | |
import java.io.IOException; | |
import java.util.Base64; | |
import java.util.HashMap; | |
import java.util.Map; | |
import org.apache.commons.io.FileUtils; | |
import org.apache.commons.lang3.RandomStringUtils; | |
import org.springframework.http.HttpStatus; | |
import org.springframework.http.ResponseEntity; | |
import com.fasterxml.jackson.databind.ObjectMapper; | |
import architecture.ee.exception.NotFoundException; | |
import architecture.web.attachments.AttachmentService; // 업로드를 위한 내부 컴포넌트로 환경에 맞게 구현하여 사용하면 됨. | |
import lombok.extern.slf4j.Slf4j; | |
@Slf4j | |
public abstract class AbstractTusUploadDataController { | |
private static final String TUS_META_FILENAME = "meta.json"; | |
private static final String TUS_RESUMABLE_VERSION = "1.0.0"; | |
private static final long MAX_FILE_SIZE = 10L * 1024L * 1024L * 1024L; // 10GB | |
private final ObjectMapper objectMapper = new ObjectMapper(); | |
protected abstract AttachmentService getAttachmentService(); | |
protected abstract String getUrlPrefix(); | |
protected abstract void doUploadComplete(File file, TusFileUploadStatus status) throws NotFoundException, IOException; | |
/** | |
* Handle creation post request for Tus protoclo. | |
* @param attachmentId | |
* @param fileSize | |
* @param uploadMetadata | |
* @return | |
* @throws IOException | |
*/ | |
protected ResponseEntity<?> handleCreationPostRequest(Long attachmentId, long fileSize, String uploadMetadata) throws IOException { | |
String fileName = getFileNameFromMetadata(uploadMetadata); | |
if (fileSize > MAX_FILE_SIZE) { | |
return ResponseEntity.status(HttpStatus.BAD_REQUEST).body("File size exceeds the maximum limit"); | |
} | |
if (fileName == null) | |
return ResponseEntity.status(HttpStatus.BAD_REQUEST).body("Filename is missing in metadata"); | |
Map<String, String> meta = parseMetadata(uploadMetadata); | |
TusFileUploadStatus status = createUpload(fileName, fileSize, meta); | |
String location = getUrlPrefix() + "/" + attachmentId + "/tus/" + status.getUploadUid(); | |
log.debug("create upload : {} - {}", fileName, status.getUploadUid()); | |
return handleCreationPostRequest(status, location); | |
} | |
protected ResponseEntity<?> handleCreationPostRequest(TusFileUploadStatus status, String location) throws IOException { | |
// status 나 location 이 없는 경우 에러 처리 | |
if (status == null || location == null) { | |
return ResponseEntity.status(HttpStatus.BAD_REQUEST).body("Invalid request"); | |
} | |
return ResponseEntity.status(HttpStatus.CREATED) | |
.header(TusHttpHeader.LOCATION, location ) | |
.header(TusHttpHeader.TUS_VERSION, TUS_RESUMABLE_VERSION) | |
.build(); | |
} | |
protected ResponseEntity<?> handleCreationPatchRequest( | |
String encodedFileName, | |
long offset, | |
String tusVersion, | |
byte[] data) throws IOException { | |
if (!TUS_RESUMABLE_VERSION.equals(tusVersion)) { | |
return ResponseEntity.status(HttpStatus.PRECONDITION_FAILED).body("Unsupported TUS version"); | |
} | |
try { | |
updateUpload(encodedFileName, offset, data); | |
} catch (IllegalStateException e) { | |
return ResponseEntity.status(HttpStatus.CONFLICT).body(e.getMessage()); | |
} | |
TusFileUploadStatus fileUploadStatus = getFileUploadStatus(encodedFileName); | |
log.debug("uploading file : {} , offset : {} , size : {} - {}", fileUploadStatus.getFileName(), | |
fileUploadStatus.getUploadOffset(), | |
fileUploadStatus.getFileSize(), fileUploadStatus.isComplete()); | |
// Check if the upload is complete | |
if (fileUploadStatus.isComplete()) { | |
try{ | |
uploadComplete(fileUploadStatus); | |
} catch (NotFoundException e) { | |
return ResponseEntity.status(HttpStatus.CONFLICT).body(e.getMessage()); | |
} | |
} | |
return ResponseEntity.noContent() | |
.header(TusHttpHeader.UPLOAD_OFFSET, String.valueOf(fileUploadStatus.getUploadOffset())) | |
.header(TusHttpHeader.TUS_RESUMABLE, TUS_RESUMABLE_VERSION) | |
.build(); | |
} | |
protected ResponseEntity<?> handleCreationHeadRequest ( | |
String encodedFileName) throws IOException { | |
TusFileUploadStatus fileUploadStatus = getFileUploadStatus(encodedFileName); | |
log.debug("File upload offset : {} - {}/{}", fileUploadStatus.isComplete(), fileUploadStatus.getUploadOffset(), fileUploadStatus.getFileSize()); | |
long uploadOffset = fileUploadStatus.isComplete() ? fileUploadStatus.getFileSize() : fileUploadStatus.getUploadOffset(); | |
return ResponseEntity.noContent() | |
.header(TusHttpHeader.UPLOAD_OFFSET, String.valueOf(uploadOffset)) | |
.header(TusHttpHeader.TUS_RESUMABLE, TUS_RESUMABLE_VERSION) | |
.build(); | |
} | |
private TusFileUploadStatus createUpload(String filename, long fileSize, Map<String, String> meta) throws IOException { | |
String uuid = RandomStringUtils.random(64, true, true); | |
File file = getTempFile(uuid); | |
TusFileUploadStatus fileUploadStatus = new TusFileUploadStatus(uuid, filename, fileSize, meta, 0, false); | |
saveMetaFile(file, fileUploadStatus); | |
return fileUploadStatus; | |
} | |
private void updateUpload(String uploadUid, long offset, byte[] data) throws IOException { | |
TusFileUploadStatus fileUploadStatus = getFileUploadStatus(uploadUid); | |
if (fileUploadStatus == null) { | |
throw new IllegalStateException("File not found"); | |
} | |
if (fileUploadStatus.isComplete()) { | |
throw new IllegalStateException("File is already completely uploaded"); | |
} | |
if (fileUploadStatus.getUploadOffset() != offset) { | |
throw new IllegalStateException("Invalid upload offset"); | |
} | |
File dir = getTempFile(uploadUid); | |
File file = new File(dir, fileUploadStatus.getFileName()); | |
try (FileOutputStream fos = new FileOutputStream(file, true)) { | |
fos.write(data); | |
} | |
fileUploadStatus.setUploadOffset(offset + data.length); | |
if (fileUploadStatus.getUploadOffset() >= fileUploadStatus.getFileSize()) { | |
fileUploadStatus.setComplete(true); | |
} | |
saveMetaFile(dir, fileUploadStatus); | |
} | |
private TusFileUploadStatus getFileUploadStatus(String uploadUid) throws IOException { | |
File dir = getTempFile(uploadUid); | |
File metaFile = new File(dir, TUS_META_FILENAME); | |
log.debug("get file upload status : {} - {}", metaFile, metaFile.exists()); | |
if (metaFile.exists()) { | |
return objectMapper.readValue(metaFile, TusFileUploadStatus.class); | |
} | |
return null; | |
} | |
private void saveMetaFile(File dir, TusFileUploadStatus fileUploadStatus) throws IOException { | |
File metaFile = new File(dir, TUS_META_FILENAME); | |
log.debug("save meta file : {} - {}", metaFile, fileUploadStatus); | |
objectMapper.writeValue(metaFile, fileUploadStatus); | |
} | |
private File getTempFile(String uploadUid) throws IOException { | |
File root = getAttachmentService().getTempDir(); | |
File dir = new File(root, uploadUid); | |
if (!dir.exists()) | |
dir.mkdirs(); | |
return dir; | |
} | |
private void uploadComplete(TusFileUploadStatus fileUploadStatus) throws NotFoundException, IOException { | |
// Add your logic to handle the completed upload | |
// For example, move the file from temp storage to permanent storage | |
log.info("Upload complete for file: {}", fileUploadStatus.getFileName()); | |
// Implement additional logic as needed | |
// clearup temp files | |
try { | |
File dir = getTempFile(fileUploadStatus.getUploadUid()); | |
doUploadComplete(new File(dir, fileUploadStatus.getFileName()), fileUploadStatus); | |
FileUtils.deleteDirectory(dir); | |
} catch (IOException e) { | |
log.error("Error while deleting temp files", e); | |
} | |
} | |
private String getFileNameFromMetadata(String metadata) { | |
// 예시로 metadata를 파싱하여 파일 이름을 추출 | |
String[] pairs = metadata.split(","); | |
for (String pair : pairs) { | |
String[] kv = pair.split(" "); | |
if (kv[0].equals("filename")) { | |
return new String(Base64.getDecoder().decode(kv[1])); | |
} | |
} | |
return null; | |
} | |
private Map<String, String> parseMetadata(String metadata) { | |
Map<String, String> metadataMap = new HashMap<>(); | |
String[] pairs = metadata.split(","); | |
for (String pair : pairs) { | |
String[] kv = pair.split(" "); | |
if (kv.length == 2) { | |
metadataMap.put(kv[0], new String(Base64.getDecoder().decode(kv[1]))); | |
} | |
} | |
return metadataMap; | |
} | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
package architecture.web.spring.web.controller.data.upload; | |
import java.util.Map; | |
import lombok.AllArgsConstructor; | |
import lombok.Builder; | |
import lombok.Data; | |
import lombok.NoArgsConstructor; | |
@AllArgsConstructor | |
@NoArgsConstructor | |
@Data | |
@Builder | |
public class TusFileUploadStatus { | |
private String uploadUid; | |
private String fileName; | |
private long fileSize; | |
private Map<String, String> meta; | |
private long uploadOffset; | |
private boolean complete; | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
package architecture.web.spring.web.controller.data.upload; | |
public class TusHttpHeader { | |
public static final String LOCATION = "Location"; | |
/** | |
* The Upload-Offset request and response header indicates a byte offset within a resource. The | |
* value MUST be a non-negative integer. | |
*/ | |
public static final String UPLOAD_OFFSET = "Upload-Offset"; | |
public static final String UPLOAD_METADATA = "Upload-Metadata"; | |
/** | |
* The Upload-Checksum request header contains information about the checksum of the current body | |
* payload. The header MUST consist of the name of the used checksum algorithm and the Base64 | |
* encoded checksum separated by a space. | |
*/ | |
public static final String UPLOAD_CHECKSUM = "Upload-Checksum"; | |
/** | |
* The Upload-Length request and response header indicates the size of the entire upload in bytes. | |
* The value MUST be a non-negative integer. | |
*/ | |
public static final String UPLOAD_LENGTH = "Upload-Length"; | |
/** | |
* The Upload-Expires response header indicates the time after which the unfinished upload | |
* expires. The value of the Upload-Expires header MUST be in RFC 7231 | |
* (https://tools.ietf.org/html/rfc7231#section-7.1.1.1) datetime format. | |
*/ | |
public static final String UPLOAD_EXPIRES = "Upload-Expires"; | |
/** | |
* The Upload-Defer-Length request and response header indicates that the size of the upload is | |
* not known currently and will be transferred later. Its value MUST be 1. If the length of an | |
* upload is not deferred, this header MUST be omitted. | |
*/ | |
public static final String UPLOAD_DEFER_LENGTH = "Upload-Defer-Length"; | |
/** | |
* The Upload-Concat request and response header MUST be set in both partial and upload creation | |
* requests. It indicates whether the upload is either a partial or upload. | |
*/ | |
public static final String UPLOAD_CONCAT = "Upload-Concat"; | |
/** | |
* The Tus-Version response header MUST be a comma-separated list of protocol versions supported | |
* by the Server. The list MUST be sorted by Server’s preference where the first one is the most | |
* preferred one. | |
*/ | |
public static final String TUS_VERSION = "Tus-Version"; | |
/** | |
* The Tus-Resumable header MUST be included in every request and response except for OPTIONS | |
* requests. The value MUST be the version of the protocol used by the Client or the Server. | |
*/ | |
public static final String TUS_RESUMABLE = "Tus-Resumable"; | |
/** | |
* The Tus-Extension response header MUST be a comma-separated list of the extensions supported by | |
* the Server. If no extensions are supported, the Tus-Extension header MUST be omitted. | |
*/ | |
public static final String TUS_EXTENSION = "Tus-Extension"; | |
/** | |
* The Tus-Max-Size response header MUST be a non-negative integer indicating the maximum allowed | |
* size of an entire upload in bytes. The Server SHOULD set this header if there is a known hard | |
* limit. | |
*/ | |
public static final String TUS_MAX_SIZE = "Tus-Max-Size"; | |
private TusHttpHeader() { | |
// This is an utility class to hold constants | |
} | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
package architecture.web.spring.web.controller.data.secure.mgmt.resources; | |
import java.io.File; | |
import java.io.FileInputStream; | |
import java.io.IOException; | |
import java.util.Optional; | |
import org.springframework.beans.factory.annotation.Autowired; | |
import org.springframework.beans.factory.annotation.Qualifier; | |
import org.springframework.http.ResponseEntity; | |
import org.springframework.web.bind.annotation.PatchMapping; | |
import org.springframework.web.bind.annotation.PathVariable; | |
import org.springframework.web.bind.annotation.PostMapping; | |
import org.springframework.web.bind.annotation.RequestBody; | |
import org.springframework.web.bind.annotation.RequestHeader; | |
import org.springframework.web.bind.annotation.RequestMapping; | |
import org.springframework.web.bind.annotation.RequestMethod; | |
import org.springframework.web.bind.annotation.RestController; | |
import architecture.ee.exception.NotFoundException; | |
import architecture.user.User; | |
import architecture.user.util.SecurityHelper; | |
import architecture.web.attachments.Attachment; | |
import architecture.web.attachments.AttachmentService; | |
import architecture.web.attachments.DefaultAttachment; | |
import architecture.web.attachments.dao.jpa.LocalAttachment; | |
import architecture.web.model.Models; | |
import architecture.web.share.SharedLinkService; | |
import architecture.web.spring.web.controller.data.upload.AbstractTusUploadDataController; | |
import architecture.web.spring.web.controller.data.upload.TusFileUploadStatus; | |
import architecture.web.spring.web.controller.data.upload.TusHttpHeader; | |
import io.swagger.v3.oas.annotations.tags.Tag; | |
import lombok.extern.slf4j.Slf4j; | |
@Slf4j | |
@RequestMapping(TusUploadDataController.TUS_REQUEST_URL) | |
@Tag(name = "mgmt files", description = "File Management APIs") | |
@RestController("controllers:mgmt:tus-upload-data-controller") | |
public class TusUploadDataController extends AbstractTusUploadDataController { | |
public static final String TUS_REQUEST_URL = "/data/secure/mgmt/resources/files"; | |
final AttachmentService attachmentService; | |
final SharedLinkService sharedLinkService; | |
TusUploadDataController( | |
@Qualifier(AttachmentService.SERVICE_NAME) AttachmentService attachmentService, | |
@Autowired(required = false) @Qualifier(SharedLinkService.SERVICE_NAME) SharedLinkService sharedLinkService ) { | |
this.attachmentService = attachmentService; | |
this.sharedLinkService = sharedLinkService; | |
} | |
protected AttachmentService getAttachmentService() { | |
return attachmentService; | |
} | |
protected String getUrlPrefix() { | |
return TUS_REQUEST_URL; | |
} | |
protected void doUploadComplete(File file, TusFileUploadStatus status) throws NotFoundException, IOException { | |
String contentType = Optional.ofNullable(status.getMeta().get("filetype")) | |
.map(Object::toString) | |
.orElse("application/octet-stream"); // 기본값을 application/octet-stream 설정 | |
long attachmentId = Optional.ofNullable(status.getMeta().get("attachmentId")) | |
.map(Object::toString) | |
.map(Long::parseLong) | |
.orElse(0L); // 기본값을 0 설정 | |
int objectType = Optional.ofNullable(status.getMeta().get("objectType")) | |
.map(Object::toString) | |
.map(Integer::parseInt) | |
.orElse(0); // 기본값을 0 설정 | |
long objectId = Optional.ofNullable(status.getMeta().get("objectId")) | |
.map(Object::toString) | |
.map(Long::parseLong) | |
.orElse(0L); // 기본값을 0 설정 | |
boolean link = Optional.ofNullable(status.getMeta().get("link")) | |
.map(Object::toString) | |
.map(Boolean::parseBoolean) | |
.orElse(false); // 기본값을 false 설정 | |
FileInputStream inputStream = new FileInputStream(file); | |
Attachment attachment; | |
boolean isUpdate = attachmentId > 0 ? true : false; | |
if (isUpdate) { | |
attachment = attachmentService.getAttachment(attachmentId); | |
attachment.setContentType(contentType); | |
attachment.setSize((int)status.getFileSize() ); | |
attachment.setInputStream(inputStream); | |
attachment.setName(status.getFileName()); | |
setObjectTypeAndObjectId(attachment, objectType, objectId); | |
} else { | |
attachment = attachmentService.createAttachment(objectType, objectId, status.getFileName(), contentType, inputStream, (int) status.getFileSize()); | |
} | |
User user = SecurityHelper.getUser(); | |
attachment.setUser(user); | |
attachmentService.saveAttachment(attachment); | |
if (link & !isUpdate) { | |
sharedLinkService.getSharedLink(Models.ATTACHMENT.getObjectType(), attachment.getAttachmentId(), true); | |
} | |
} | |
private void setObjectTypeAndObjectId(Attachment attachment, int objectType, long objectId) { | |
if (attachment instanceof LocalAttachment){ | |
((LocalAttachment) attachment).setObjectType(objectType); | |
((LocalAttachment) attachment).setObjectId(objectId); | |
}else{ | |
((DefaultAttachment) attachment).setObjectType(objectType); | |
((DefaultAttachment) attachment).setObjectId(objectId); | |
} | |
} | |
/** | |
* Create a new upload for tus protocol | |
* | |
* @param attachmentId | |
* @param fileSize | |
* @param uploadMetadata | |
* @return | |
* @throws IOException | |
*/ | |
@PostMapping("/{attachmentId:[\\p{Digit}]+}/tus") | |
public ResponseEntity<?> createUpload( | |
@PathVariable Long attachmentId, | |
@RequestHeader(TusHttpHeader.UPLOAD_LENGTH) long fileSize, | |
@RequestHeader(TusHttpHeader.UPLOAD_METADATA) String uploadMetadata) throws IOException { | |
return handleCreationPostRequest(attachmentId, fileSize, uploadMetadata); | |
} | |
/** | |
* Upload a file chunk for tus protocol | |
* | |
* @param attachmentId | |
* @param encodedFileName | |
* @param offset | |
* @param tusVersion | |
* @param data | |
* @return | |
* @throws IOException | |
*/ | |
@PatchMapping("/{attachmentId:[\\p{Digit}]+}/tus/{encodedFileName}") | |
public ResponseEntity<?> uploadFile(@PathVariable Long attachmentId, | |
@PathVariable String encodedFileName, | |
@RequestHeader(TusHttpHeader.UPLOAD_OFFSET) long offset, | |
@RequestHeader(TusHttpHeader.TUS_RESUMABLE) String tusVersion, | |
@RequestBody byte[] data) throws IOException { | |
return handleCreationPatchRequest(encodedFileName, offset, tusVersion, data); | |
} | |
/** | |
* Get the upload offset for the file | |
* | |
* @param attachmentId | |
* @param encodedFileName | |
* @return | |
* @throws IOException | |
*/ | |
@RequestMapping(value = "/{attachmentId:[\\p{Digit}]+}/tus/{encodedFileName}", method = RequestMethod.HEAD) | |
public ResponseEntity<?> getFileUploadOffset( | |
@PathVariable Long attachmentId, | |
@PathVariable String encodedFileName) throws IOException { | |
return handleCreationHeadRequest(encodedFileName); | |
} | |
} |
댓글 없음:
댓글 쓰기