1. 요약
- 목적
- 이미지 전송 원리 이해
- 동기
- 현재 제작하고 있는 프로젝트의 더 깊은 이해
- 현 프로젝트에서는 이미지를 GCS에 저장하여 사용하고 있음
- 해당 이미지 전송에 대한 이해도가 부족하다고 생각되어, 이미지 전송 과정을 공부하고자 함
- 현재 제작하고 있는 프로젝트의 더 깊은 이해
- 과정
- FE → BE로 이미지 전송하는 과정을 그리기
2. FE
- 요약
- 통신 방법
- XML
- XHL
- XMLHttpRequest
- AJAX
- 데이터 형식
- MIME-TYPE
- Content-Type
- multipart/*
- multipart/form-data
- 통신 방법
- XML
- 데이터 표기 방법 중 하나
- 데이터 전송 시, xml 형식으로 데이터 정보 기술 가능
- 예시
- 데이터 표기 방법 중 하나
<book>
<title>Harry Potter</title>
<author>J.K. Rowling</author>
<year>1997</year>
</book>
- XHL
- 비동기 통신 방법
- XML, JSON 등의 데이터를 주고받게 됨
- 이점
- 페이지 새로고침 없이 최신 정보로 업데이트
※ 동기 / 비동기 통신
- 동기
- 상대방의 응답이 올때가지 일을 하지 않음
- 예시
- 상대방에게 요청 보냄 → 상대방에게 응답 받음 → 다음 일 수행
- 비동기
- 상대방의 응답을 기다리지 않고 다음 작업 수행
- 목적
- 일부 페이지만 리렌더링 시키는 것
- 전체 페이지 새로고침 없이 일부만 비동기적으로 처리해서 페이지를 사용하는 걸 목적으로 함
- 예시
- 상대방에게 요청 보냄 → 다음 일 수행 → 상대방에게 응답 받음
- XMLHttpRequest
- XHL 형식의 통신을 가능하게 해주는 객체
- 해당 객체로 통신 시, 상대방의 응답을 기다리지 않고 내 작업을 계속 수행
- 예시
- XHR : GET 요청 보내기
var xhr = new XMLHttpRequest();
xhr.open('GET', 'https://api.example.com/data', true); // true -> 비동기 요청
xhr.onreadystatechange = function() {
if (xhr.readyState === 4 && xhr.status === 200) {
// 서버 응답을 처리
console.log(xhr.responseText);
}
};
xhr.send();
//send() 이후, 응답을 기다리지 않고 다음 작업 수행
- AJAX
- XHL을 이용해서 비동기 통신을 수행함
- XMLHttpRequest로 XHL을 사용함
- 이점
- 웹페이지 동적 업데이트
- XHL만 이용해서는 새로고침 없이 웹페이지를 동적으로 업데이트하기에는 난이도가 있음
- AJAX는 XHL을 편리하게 사용할 수 있도록 구성함으로써, 동적 업데이트를 쉽게 구현 가능
- 웹페이지 동적 업데이트
- XHL을 이용해서 비동기 통신을 수행함
- MIME-TYPE
- 요약
- 인터넷 통신에서 데이터 타입을 정하는 표준
- (인터넷 통신 : HTTP 통신, 이메일 통신, …)
- 인터넷 통신에서 데이터 타입을 정하는 표준
- 예시
- 웹 페이지: text/html
- 이미지 파일: image/png, image/jpeg
- JSON 데이터: application/json
- PDF 파일: application/pdf
- 요약
- Content-Type
- 요약
- HTTP 통신에서 Request Body 데이터에 대한 MIME-type 지정
- 설명
- POST 요청 시, Request Body에 요청 데이터를 첨부한다.
- 요청을 받는 입장에서는 Content-Type을 통해 요청 데이터의 형식을 이해할 수 있다.
- 예시
- XHL
- HTTP Method : POST
- Content-Type : JSON
- XHL
- 요약
var xhr = new XMLHttpRequest();
xhr.open('POST', 'https://example.com/upload', true);
// 요청 헤더에 MIME-TYPE 설정
xhr.setRequestHeader('Content-Type', 'application/json'); // JSON 데이터 전송
// 요청 보내기
var data = JSON.stringify({ name: 'John', age: 30 });
xhr.send(data);
- multipart/*
- 요약
- 여러 타입의 데이터를 한 번에 보내기 위한 Content-Type
- 설명
- image, text 등의 한정된 Content-Type으로는 Form Request를 보내기 어렵다.
- 예를 들어, 사진 및 설명 글을 백엔드에 요청 시, 여러 번 나누어서 Request를 보내야 한다.
- 그때, multipart는 여러 타입의 데이터를 한 번에 보낼 수 있다.
- multipart는 boundary라는 문자열로 각 타입의 데이터를 분리함
- 종류
- multipart/form-data
- 텍스트 및 파일 동시 전송
- 각 타입의 데이터들은 name 필드로 구분됨
- HTML Form 데이터 전송할 때 좋음
- 예시
- 받는 입장에서는 name을 통해 각 데이터를 가져오는게 가능함
- multipart/form-data
- 요약
POST /upload HTTP/1.1
Host: example.com
Content-Type: multipart/form-data; boundary=----Boundary123
------Boundary
Content-Disposition: form-data; name="username"
john_doe
------Boundary
Content-Disposition: form-data; name="file"; filename="example.jpg"
Content-Type: image/jpeg
(binary data)
------Boundary--
- multipart/alternative
- 동일한 내용을 서로 다른 타입으로 보냄
- 응답을 받는 입장에서는 자기가 원하는 타입으로 골라 받음
- multipart/mixed
- 텍스트 및 파일 동시 전송
- form-data처럼 name 필드는 없음
- name 필드 구분없이 데이터 보내도 좋을 때 사용함
- ex) 파일 묶음 전송할때 좋음
- 예시
Content-Type: multipart/mixed; boundary=----Boundary123
------Boundary
Content-Type: text/plain
This is plain text.
------Boundary
Content-Type: image/jpeg
Content-Disposition: attachment; filename="image.jpg"
(binary data)
------Boundary-
- multipart/related
- 루트 데이터와 연관 데이터로 나뉨
- 루트 데이터는 연관 데이터들을 가져와서 완성된 루트 데이터를 형성함
- 예시를 보면, 루트 데이터인 HTML은 연관 데이터인 image 데이터를 가져와서 완성된 HTML을 만듦
- 예시
Content-Type: multipart/related; boundary="boundary123"; start="<main.html>"; type="text/html"
--boundary123
Content-Type: text/html
Content-ID: <main.html>
<!DOCTYPE html>
<html>
<body>
<h1>Hello, World!</h1>
<img src="cid:image1.jpg">
</body>
</html>
--boundary123
Content-Type: image/jpeg
Content-ID: <image1.jpg>
(binary image data)
--boundary123--
- multipart/form-data
- 위 글을 통해 HTTP Form Request에서는 multipart/form-data가 적절한 전송 방식인 것을 확인 할 수 있다.
- form-data로 Request 요청을 내리기 위한 예시는 아래와 같다.
- 예시 - FormData 객체 사용
var formData = new FormData();
formData.append("textField", "sample text");
formData.append("fileInput", fileInput);
var xhr = new XMLHttpRequest();
xhr.open("POST", "/upload", true);
//formData를 xhr에 넣으면 자동으로 Content-Type = multipart/form-data
//xhr.setRequestHeader("Content-Type", "multipart/form-data");
xhr.send(formData);
예시 - HTML Form 사용
<form action="/upload" method="POST" enctype="multipart/form-data">
<input type="text" id="username" name="username"><br><br>
<input type="file" id="profile-pic" name="profile_pic"><br><br>
<input type="submit" value="Submit">
</form>
3. BE(Spring)
- 요약
- 이미지 전송 과정을 확인
- 설명
- 프론트로부터 이미지 파일 받기
- MultipartHttpServletRequest
- @ModelAttrubite
- MultiPartFile
- 이미지 파일 저장
- Base64
- Blob
- 프론트로부터 이미지 파일 받기
- MultipartHttpServletRequest
- Multipart/* 요청을 받기위해서 사용되는 타입
- 예시
- multipart/form-data 요청 시, 각 텍스트 및 파일을 받아올 수 있다.
@PostMapping("/item")
public ResponseEntity<String> item(MultipartHttpServletRequest request){
log.info("request : {}", request);
return ResponseEntity.ok("ok");
}
- @ModelAttrubite
- HTML Form 데이터를 보다 쉽게 받을 수 있는 어노테이션
- 사용 예시
@PostMapping("/item")
public ResponseEntity<?> item(@ModelAttribute ItemFormRequestDto itemFormRequestDto){
if(itemService.save(itemFormRequestDto) != null){
return ResponseEntity.ok("ok");
}
return ResponseEntity.badRequest().body("상품 등록에 실패했습니다");
}
- MultipartFile
- 스프링에서 이미지 파일 저장을 위한 객체
- 구조
- 파일의 이름, 타입, 크기 등을 담을 수 있음
public interface MultipartFile extends InputStreamSource {
String getName();
@Nullable
String getOriginalFilename();
@Nullable
String getContentType();
boolean isEmpty();
long getSize();
byte[] getBytes() throws IOException;
InputStream getInputStream() throws IOException;
default Resource getResource() {
return new MultipartFileResource(this);
}
void transferTo(File dest) throws IOException, IllegalStateException;
default void transferTo(Path dest) throws IOException, IllegalStateException {
FileCopyUtils.copy(this.getInputStream(), Files.newOutputStream(dest));
}
}
- 예시
- FE
- 이미지 전송
- FE
<form>
<label>상품 이미지 파일</label>
<input type="file" name="imageFile"/>
<button type="submit" className={styles.submitButton}>상품 등록</button>
</form>
- BE
- 이미지 받기
import org.springframework.web.multipart.MultipartFile;
public class ItemFormRequestDto {
...
private MultipartFile imageFile;
}
- Base64
- 각 0, 1의 조합을 글자로 해석하는 것
- 방법
- 주어진 바이너리 데이터를 6비트씩 분리
- 각 6비트의 10진수 값과 매칭되는 문자열을 도출함
- 인코딩 : 바이너리 데이터 → 문자
- 디코딩 : 문자 → 바이너리 데이터
- 예시
- 000000 = A
- Base64 ↔ HTML
- HTML에서는 Base64로 인코딩된 값을 디코딩하여 이미지 출력이 가능함
- 예시
- 해당 src 값은 Base64로 인코딩된 값
- HTML에서는 해당 값을 디코딩하여 이미지 출력
<img src="">
- 장점
- 이미지를 온라인으로 불러올 필요없이 바로 출력
- 단점
- HTML에서 데이터 크기가 33% 증가
- 이유
- 가정
- HTML에서 문자 1개는 8비트
- 바이너리 데이터 24bit가 주어짐
- HTML에서 바이너리 데이터를 그대로 옮기면 3개의 문자임
- 8bit 문자 3개
- 하지만 Base64 인코딩 하면 4개의 문자임
- Base64는 6개의 비트를 사용함
- 따라서, 24bit면 4개의 문자를 만들 수 있음
- HTML에서 4개의 문자로 표현되니 32bit로 표현되버림
- ⇒ Base64 인코딩 과정을 거치면서 데이터 뻥튀기
- 가정
- Blob (Binary Larage object)
- 요약
- 바이너리 데이터를 다루기 위한 객체
- 설명
- 사용자 정의 바이너리 데이터 저장 객체
- 각 플랫폼에 따라 바이너리 데이터를 어떻게 다루느냐에 따라 Blob의 기능이 다를 수 있음
- Blob으로 데이터 저장
- 바이너리 데이터를 그대로 저장함
- 따라서, Base64와 같이 데이터 크기가 증가하는 일이 없음
- 예시
- GCS
- Blob
- 바이너리 데이터를 꺼냄
- Blob
- GCS
- 요약
package com.google.cloud.storage;
public class Blob extends BlobInfo {
...
private final StorageOptions options;
private transient Storage storage;
public byte[] getContent(BlobSourceOption... options) {
return this.storage.readAllBytes(this.getBlobId(), Blob.BlobSourceOption.toSourceOptions(this, options));
}
...
}
- BlobInfo
- Blob의 데이터에 대한 정보 표기
package com.google.cloud.storage;
public class BlobInfo implements Serializable {
...
private final BlobId blobId;
private final String generatedId;
private final String selfLink;
private final String cacheControl;
private final List<Acl> acl;
private final Acl.Entity owner;
private final Long size;
...
- 이미지 전송 및 저장
- MultipartFile (이미지)의 바이너리 데이터를 BlobInfo 및 BlobId를 통해 타입 표기
- 해당 이미지 바이너리 데이터를 GCS에 저장
public String uploadImage(MultipartFile image){
if(storage == null){
log.info("Storage 생성 실패");
return null;
}
String uuid = UUID.randomUUID().toString(); // Gcs 에 저장될 파일 이름
String ext = image.getContentType(); // 파일의 형식 ex) JPG
// Gcs 이미지 업로드
BlobId blobId = BlobId.of(bucketName, uuid);
BlobInfo blobInfo = BlobInfo.newBuilder(blobId)
.setContentType(ext)
.build();
try (WriteChannel writer = storage.writer(blobInfo)) {
byte[] imageData = image.getBytes();
writer.write(ByteBuffer.wrap(imageData));
} catch (Exception e) {
log.error("이미지 전송 실패");
return null;
}
return createURL(uuid);
}
QnA)
1) 이미지 파일을 JSON 방식으로 보낼 수 있나?
- 의문
- FE → BE으로 이미지 파일 전송 시, multipart/form-data의 content-type으로 데이터를 종합적으로 전송한다.
- 그렇다면, 이미지 파일을 JSON 타입으로 보낼 수 없을까?
- 결론
- 이미지를 바이너리 데이터 형식으로 JSON 전송을 할 수 없다
- 이미지를 Base64 인코딩된 문자열 형식으로 JSON 전송을 할 수 있다
- 이미지를 바이너리 데이터 형식으로 JSON 전송을 할 수 없다
- 요약
- JSON은 텍스트 기반이므로, 바이너리 데이터 형식으로 데이터 전송 불가
- 설명
- JSON 데이터를 읽는 데는 4가지 타입을 지원
- 문자열, 숫자, bool, 객체
- 바이너리 데이터 형식대로 데이터를 보내도, 읽을 수 있는 타입이 없음
- 따라서 바이너리 데이터 형식으로 데이터 전송 불가
- JSON 데이터를 읽는 데는 4가지 타입을 지원
- 요약
- 이미지를 Base64 인코딩된 문자열 형식으로 JSON 전송을 할 수 있다
- 요약
- Base64로 인코딩된 값은 문자이므로, JSON으로 전송 가능
- 예시
- FE
- image → Base64 인코딩
- Json 형태로 인코딩된 image 값 전달
- FE
- 요약
import React, {useState} from 'react';
import styles from '../css/login.module.css'
import Cookies from 'js-cookie';
import {Link} from "react-router-dom";
const Image= () => {
return (
<div>
<ImageForm/>
</div>
);
};
const ImageForm= () => {
const BASE_URL = process.env.REACT_APP_API_URL || 'http://localhost:8080';
const TEST_ENDPOINT = '/test';
const [base64Image, setBase64Image] = useState("");
// image -> Base64 인코딩
const handleImageChange = (event) => {
const file = event.target.files[0];
if (file) {
const reader = new FileReader();
reader.onload = (e) => {
setBase64Image(e.target.result);
};
reader.readAsDataURL(file);
}
};
const handleSubmit = async (e) => {
e.preventDefault(); // 기본 폼 제출 방지
try {
console.log("image : ", base64Image )
// POST 요청 보내기
const response = await fetch(`${BASE_URL}${TEST_ENDPOINT}`, {
method: 'POST',
headers: {
'Content-Type': 'application/json',
},
body: JSON.stringify({ image: base64Image }),
});
if (response.ok) {
alert('조회 성공' + response.text());
} else {
alert('조회 실패');
}
};
return (
<>
<form onSubmit={handleSubmit}>
<div className={styles.heading}>로그인</div>
<input type="file" id="imageInput" accept="image/*" onChange={handleImageChange}/>
<img id="preview" alt="Preview Image"/>
<input type="submit" value="제출" className={styles['submit-button']}/>
</form>
</>
);
};
export default Image;
//image 예시
//image : 
- BE
- String으로 인코딩된 image값 전달 받기
@Slf4j
@Controller
public class TestController {
@PostMapping("/test")
public ResponseEntity<?> test(@RequestBody Dto dto){
log.info("dto : {}", dto);
return ResponseEntity.ok("ok");
}
@Data
static class Dto{
String image;
}
}
//dto 출력 예시
//dto : TestController.Dto(image=)