Development/Spring & Spring Boot
[Spring Boot] Kotlin 기반 Spring Boot 프로젝트에서 파일 업로드하기
개발하는 곰돌이
2022. 12. 23. 17:00
목차
기본 환경
- IntelliJ Ultimate 2022.3
- Spring Boot 2.7.6
- Kotlin 1.7.21(JDK 11)
- Oracle 11g
- Swagger 3.0.0
- mybaitis 2.3.0
Spring Boot 프로젝트에 Swagger 3.0.0을 적용하는 방법은 아래 포스트를 참고하면 된다.
[Spring Boot] Kotlin 기반 Spring Boot 프로젝트에 Swagger 3.0.0 적용하기
개요
예제로 블로그에 글을 작성할 때 제목과 내용, 카테고리 일련번호와 파일 하나를 업로드하여 작성하는 API를 구현한다. Swagger를 통해 테스트하기 때문에 별도의 화면은 구성하지 않는다.
Domain
블로그의 글과 관련된 BoardDTO를 작성한다.
data class BoardDTO(
var seq: Int?,
@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss", timezone = "Asia/Seoul")
var reg_time: LocalDateTime?,
var category_id: Int?,
var subject: String,
var content: String,
var thumbnail: String?,
var view_cnt: Int?,
var like_cnt: Int?
) {
constructor(
subject: String,
content: String,
category_id: Int?,
thumbnail: String?
) : this(
null,
null,
category_id,
subject,
content,
thumbnail,
null,
null
)
}
Controller
파일은 선택적으로 업로드해야 하므로 uploadFile은 required = false
로 지정한다. 글 제목과 내용, 글을 분류할 카테고리 번호와 함께 업로드할 파일을 파라미터로 받는 write()
메소드를 작성한다. 글을 작성한 이후엔 작성한 글을 불러올 것이기 때문에 반환 타입은 BoardDTO로 한다.
@Api(tags = ["블로그 API"], description = "블로그 글 관련 API")
@RestController
class BoardController(val boardMapper: BoardMapper) {
@ApiOperation("글 작성")
@ApiImplicitParams(
ApiImplicitParam(name = "subject", value = "글 제목", required = true),
ApiImplicitParam(name = "content", value = "글 내용", required = true),
ApiImplicitParam(name = "category_id", value = "글을 분류할 카테고리의 일련번호")
)
@PostMapping("/write")
fun write(
@RequestParam subject: String,
@RequestParam content: String,
@RequestParam(required = false) category_id: Int?,
@ApiParam("업로드 할 썸네일") @RequestPart(required = false) uploadFile: MultipartFile?
): BoardDTO? {
val path = "D:\\blog\\img\\"
Files.createDirectories(Paths.get(path))
var filename: String? = null
if (uploadFile?.isEmpty == false) {
val extension = FilenameUtils.getExtension(uploadFile.originalFilename)
val timemillis = System.currentTimeMillis()
val random = IntRange(10000, 99999).random()
filename = "${random}${timemillis}.$extension"
uploadFile.transferTo(File("${path}${filename}"))
}
val document = BoardDTO(subject, content, category_id ?: 0, filename)
boardMapper.write(document)
return boardMapper.getDocument(document.seq ?: 0)
}
}
write() 메소드의 동작 과정
- 업로드한 파일을 저장할 경로를 설정하고, 해당 경로가 존재하지 않으면 새로운 디렉토리를 만든다.
- uploadFile이 null이거나 빈 파일인지 체크하고 업로드한 파일이 빈 파일이 아니라면 업로드 과정을 수행한다.
- 업로드한 파일의 확장자를 추출하고 현재 시간을 기준으로 무작위 파일명을 부여한 후 지정 경로에 파일을 전송한다. 무작위 파일명은 5자리의 무작위 숫자 뒤에 작성시간의 밀리초 단위를 붙여서 설정한다.
- BoardDTO 객체를 생성하여 글을 작성하고 작성된 글을 반환한다.
Swagger로 테스트
테스트는 설정해둔 Swagger에서 진행한다. Swagger 설정에 대해선 다음 링크를 참고하면 된다.
글 작성 API를 테스트해보면 정상적으로 작성된 글이 반환된 것을 볼 수 있다.
파일도 정상적으로 업로드 되었다.