파일 업로드와 다중 insert
저번주에 골머리를 썩혔던 파일 업로드 부분과
데이터 insert 부분을 진행했다.
각설하고 시작하자면 파일업로드 부분에서 multipartFile은 일단 버려두었다.
기존에 js단에서 업로드된 이미지를 base64 포맷으로 컨버팅하여 UI단에 보여주는 것을 이용하여
base64 스트림을 사용한 이미지 업로드를 구현하였다.
@Data
public class Base64File{
private FileCode fileCode;
private String base64Str;
@Builder
public Base64File(FileCode fileCode, String base64Str) {
super();
this.fileCode = fileCode;
this.base64Str = base64Str;
}
}
일단 편의를 위해 lombok data 어노테이션 사용..
DTO
@Data
public class CompanySaveRequestDto {
private String comName;
private String comOwnerName;
private String comBuildDate;
private String comAddr;
private String comAddrCity;
private String comAddrState;
private CityEnum city;
private String comBizNum;
private String comBizTypeCode;
private String comBizTypeName;
private String comAddrPostalCode;
private List<Base64File> files;
@Builder
public CompanySaveRequestDto(String comName, String comOwnerName, String comBuildDate, String comAddr,
String comAddrCity, String comAddrState, CityEnum city, String comBizNum, String comBizTypeCode,
String comBizTypeName, String comAddrPostalCode, List<Base64File> files) {
super();
this.comName = comName;
this.comOwnerName = comOwnerName;
this.comBuildDate = comBuildDate;
this.comAddr = comAddr;
this.comAddrCity = comAddrCity;
this.comAddrState = comAddrState;
this.city = city;
this.comBizNum = comBizNum;
this.comBizTypeCode = comBizTypeCode;
this.comBizTypeName = comBizTypeName;
this.comAddrPostalCode = comAddrPostalCode;
this.files = files;
}
public Company toModel(List<DocFile> fileList) {
List<City> cityList = new ArrayList<City>();
cityList.add(City.builder().city(city).build());
return Company.builder()
.comName(comName)
.comOwnerName(comOwnerName)
.comBuildDate(comBuildDate)
.comAddr(comAddr)
.comAddrCity(comAddrCity)
.comAddrState(comAddrState)
.comBizNum(comBizNum)
.files(fileList)
.citys(cityList)
.build();
}
}
리스트 형태로 Base64File 필드를 갖는다.
ajax 부분
put : function(){
var tempform = JSON.stringify($('#frm').serializeObject());
var jsonForm = JSON.parse(tempform);
var fileList = new Array();
$(".files").each(function(){
if(!$(this).next('img').attr('src'))
return;
var temp= new Object();
temp.fileCode = $(this).attr('id');
temp.base64Str = $(this).next('img').attr('src');
fileList.push(temp);
});
jsonForm.files = fileList;
console.log(jsonForm);
$.ajax({
url : "/api/v1/company",
data: JSON.stringify(jsonForm),
dataType : 'json',
type : 'PUT',
contentType: 'application/json',
success : function(data) {
var result = data;
if (result) {
alert('저장하였습니다.');
} else {
alert('저장 실패');
}
},
error : function(request, status, error) {
console.log("code:" + request.status + "\n" + "message:"
+ request.responseText + "\n" + "error:" + error);
}
});
}
조금 지저분해서 안타깝지만 대략 이러하다.
이미지를 업로드할 때 자동으로 img 태그에 src로 주어진 base64를 파일코드와 함께 오브젝트화 한다.
UI의 텍스트 필드를 json 오브젝트로 파싱한 후 fileList 배열을 추가
다시 json string으로 전송한다.
파일 유틸부분
@Autowired
FileUtils(S3Uploader s3Uploader , ServletContext context){
this.s3Uploader = s3Uploader;
this.context = context;
}
public static List<DocFile> uploadBase64Image(List<Base64File> base64File,String target) throws Exception{
String prefixPath = context.getRealPath("/");
String filePath = prefixPath+"/"+target;
List<DocFile> fileList = new ArrayList<DocFile>();
/*
List<DocFile> list = base64File.stream()
.map(x-> Base64ToImgDecoder.decoder(x.getBase64Str(), filePath))
.peek(x-> s3Uploader.uploadToS3(x, target + x.getName()))
.map(x-> DocFile.builder().fileCode())
.collect(Collectors.toList());
*/
for(Base64File item : base64File) {
File file = Base64ToImgDecoder.decoder(item.getBase64Str(), filePath);
s3Uploader.uploadToS3(file, target + file.getName());
DocFile temp = DocFile.builder()
.fileCode(item.getFileCode())
.url(target + file.getName())
.build();
fileList.add(temp);
}
return fileList;
}
public class Base64ToImgDecoder {
public static File decoder(String base64, String target) {
String[] strings = base64.split(",");
String extension;
String rdStr = CommonUtils.getRandomString();
switch (strings[0]) {//check image's extension
case "data:image/jpeg;base64":
extension = ".jpeg";
break;
case "data:image/png;base64":
extension = ".png";
break;
default://should write cases for more images types
extension = ".jpg";
break;
}
//convert base64 string to binary data
byte[] data = DatatypeConverter.parseBase64Binary(strings[1]);
String path = target + rdStr + extension;
File file = new File(path);
try (OutputStream outputStream = new BufferedOutputStream(new FileOutputStream(file))) {
outputStream.write(data);
} catch (IOException e) {
e.printStackTrace();
}
return file;
}
}
FileUtils
context path를 가져오는 부분을 더 간단히 하고 싶었으나 실패.. fileutils 클래스에 의존성 주입하였다.
uploadBase64Image
스트림을 써서 리턴하고 싶었으나 File 형태로 컨버팅되기전 fileCode를 어떻게 지니고 있을지 몰라 일단 진행..
Base64ToImgDecoder
인터넷에서 긁어온 base64toFile 부분 알아서 확장자도 확정해줘서 좋다.
서비스 부분
@Transactional(rollbackFor = Exception.class)
public boolean putCompany(CompanySaveRequestDto request) throws Exception{
List<DocFile> fileList = FileUtils.uploadBase64Image(request.getFiles(), companyUploadDir);
Company company = request.toModel(fileList);
companyMapper.putCompany(company);
companyMapper.putCompanyCity(company);
companyMapper.putCompanyFee(company);
companyMapper.putCompanyDoc(company);
return true;
}
트랜잭셔널 어노테이션 추가..
toModel 부분에 fileList 파라미터가 들어간게 아쉽다. 파일업로드를 toModel 부분에서 진행하는게 좋을까싶다.
(realPath를 구하는데 문제가 있지만 생각해보니 Context를 사용해 bean에 realPath를 구해놓고 계속 가져다 쓰면 되지 싶다.)
아니면 company 모델에서 역으로 Base64File리스트를 받아 업로드한 후 내부의 DocFile list로 변환하는게 맞을거 같기도하다.
쿼리부분
<insert id="putCompany" useGeneratedKeys="true" keyColumn="COM_IDX" keyProperty="comIdx" parameterType="com.guivingAdmin.domain.Company">
INSERT INTO TB_COMPANY
(
COM_NAME,
COM_OWNER_NAME,
COM_BUILD_DATE,
COM_BIZ_NUM,
COM_ADDR,
COM_AUTH_CODE,
COM_ADDR_CITY,
COM_ADDR_STATE
)
VALUES(
#{comName},
#{comOwnerName},
#{comBuildDate},
#{comBizNum},
#{comAddr},
SUBSTRING(MD5(RAND()) FROM 1 FOR 8),
#{comAddrCity},
#{comAddrState}
);
</insert>
키 칼럼을 통해 insert 된 id들 가져온다.
company 모델에 setter가 없었는데 작동한걸 보니 내부적으로 다르게 동작하는가 싶다.
(insert후 다시 select한다던가..)
중간에 랜덤 스트링을 반환하는 부분은 날짜나 시퀀스등과 혼합사용해야 혹시모를 중복을 피할수 있겠다.
(이부분이 company 모델에서 내부 비지니스 처리 로직으로 들어가면 좋겠다.)
테스트 코드
@Test
public void putCompanyTest() throws Exception {
List<Base64File> files = new ArrayList<Base64File>();
files.add(Base64File.builder()
.fileCode(FileCode.MPT)
.base64Str("")
.build());
CompanySaveRequestDto request = CompanySaveRequestDto.builder()
.files(files)
.build();
cs.putCompany(request);
}
MockMvc와 restTemplate를 사용하여 컨트롤러 단부터 서비스까지의 테스트 코드 숙지가 필요하겠다.
service까지는 대략적으로 테스트 코드를 작성하는데
controller부터 (특히 파일업로드 쪽) 에서 조금 복잡해보여 제대로 시도해보지 않았다.
좀 지치게하는 일이지만
그로인해 이제까지 고민했던부분의 문제점과 개선방향에 대해서 다시 생각해볼 필요가 있다.
뛰지말고 걷더라도 멈추지 않는걸로