기타

이미지 전송

infobox503 2025. 2. 7. 13:55

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을 편리하게 사용할 수 있도록 구성함으로써, 동적 업데이트를 쉽게 구현 가능
  • 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
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을 통해 각 데이터를 가져오는게 가능함
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
      • 이미지 전송
  <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
          • 바이너리 데이터를 꺼냄
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, 객체
      • 바이너리 데이터 형식대로 데이터를 보내도, 읽을 수 있는 타입이 없음
        • 따라서 바이너리 데이터 형식으로 데이터 전송 불가
  • 이미지를 Base64 인코딩된 문자열 형식으로 JSON 전송을 할 수 있다
    • 요약
      • Base64로 인코딩된 값은 문자이므로, JSON으로 전송 가능
    • 예시
      • FE
        • image → Base64 인코딩
        • Json 형태로 인코딩된 image 값 전달
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=)