๋ฐ์ด๋๋ฆฌ ํ์ ์ ํ์ผ ์๋ต ๋ชจํนํ๊ธฐ
์ง๊ธ ์ฌ๋ด ํ๋ก์ ํธ๋ ๋ฐฑ์๋ api์ ํ๋ฐํธ ๊ฐ๋ฐ์ ๋์์ ๋ณํ์ ์ผ๋ก ์งํํ๊ณ ์๋ค. ๋ฐ๋ผ์ ์๋ฒ๋จ API๊ฐ ์์ฑ๋๊ณ ๋ฐฐํฌ๋๊ธฐ ์ด์ ์๋ msw๋ผ๋ ์๋ฒ API ๋ชจํน ๋ผ์ด๋ธ๋ฌ๋ฆฌ๋ฅผ ์ฌ์ฉํ๊ณ ์๋๋ฐ, ์ค์ ๋ฐฑ์๋ api ์์ด๋ ํ๋ฐํธ๋จ์์ mock ์๋ฒ๋ฅผ ๊ตฌํํด์ ๊ฐ์ง๋ก ๋คํธ์ํฌ ํต์ ์ ํ ์ ์๋ค.
json ํ์์ผ๋ก ํต์ ์ ์ฃผ๊ณ ๋ฐ๋ ๊ฑด ๊ณต์๋ฌธ์์๋ ์ ๋์์์ด์ Mocking์ ํ์๋๋ฐ, ์ด๋ฒ์ ๋ฐ์ด๋๋ฆฌ ํ์ ์ ์ด๋ฏธ์ง ํ์ผ์ get ํด์ค๋ ๋ถ๋ถ์์ ๋ชฉ์๋ฒ๋ฅผ ๊ตฌํํ๋๋ฐ ํฐ ๋๊ด์ด ์์๋ค.
์๋ต ์์
ํ๋ฐํธ์์ ์ด๋ฏธ์ง ํ์ผ์ get ํด์ฌ๋ ์๋ฒ์์ ์ฃผ๋ ์๋ต์ ์๋์ ๊ฐ๋ค.
๋ฐ์ด๋๋ฆฌ ๋ฐ์ดํฐ
์ฆ, ์๋ฒ์์๋ ์ด ๋ฉํฐ๋ฏธ๋์ด ๋ฐ์ดํฐ๋ฅผ 0๊ณผ 1 ๊ฐ๋ง ์กด์ฌํ๋ ์ด์ง ๋ฐ์ดํฐ๋ก ์๋ตํด์ฃผ๋ ๊ฒ์ด๋ค.
HTTP/1.1 200 OK
Content-Type: image/jpeg;charset=UTF-8
Accept-Ranges: bytes
X-Content-Type-Options: nosniff
X-XSS-Protection: 1; mode=block
Cache-Control: no-cache, no-store, max-age=0, must-revalidate
Pragma: no-cache
Expires: 0
Strict-Transport-Security: max-age=31536000 ; includeSubDomains
X-Frame-Options: DENY
์๋
์ผ๋ฐ์ ์ธ json ํ์์ ๋ฐ์ดํฐ๋ ์ง์ mock data๋ฅผ ๋ฃ์ด๋๊ณ , ์ด ๋ชฉ๋ฐ์ดํฐ๋ฅผ ์๋ต์ผ๋ก ์ฃผ๋๊น..
const todoList = [
{ id: 'bzte_tfG3k2mh3o-Vf-zX', content: 'book a hotel'},
{ id: 'rAFHp_GrutBJzt_5r1Gej', content: 'issue a travel card'},
...
]
์ด์ ๋ง์ฐฌ๊ฐ์ง๋ก ์ด๋ฏธ์งํ์ผ๋ ์๋ฒ์ ์๋๊ฒ ์ฒ๋ผ ๊ฐ์ง๋ก ํํํ๋ ค๋ฉด...?
๋ด ๋ก์ปฌ ์ปดํจํฐ์ ์๋ ์ด๋ฏธ์ง ํ์ผ์ src > assets ํด๋ ์์ ๋ฃ์ด๋๊ณ , ๋ชจํน ์ฝ๋๋ฅผ ์ฐ๋ handler ํ์ผ์์ ์ด๋ฏธ์ง ํ์ผ์ import ํด์์ ๋ฐ์ด๋๋ฆฌ ํ์ ์ผ๋ก ์๋ต์ ํด์ค๋ค.
๋ผ๋ ํ๋ก์ธ์ค ์ด๊ฒ ์ง?๋ผ๊ณ ์๊ฐํ๊ณ ์ค์ ์ด๋ฏธ์ง ํ์ผ์ ๋ฐ์ด๋๋ฆฌ ํ์ ์ผ๋ก ๋ณํํด๋ณด๋ ค๊ณ ํ์ผ๋.. ์ฝ๊ฒ ๋์ง ์์๋ค.
๋ฐ๋ผ์ ์๋์ ํ๋ก์ธ์ค๋ก mock ์ด๋ฏธ์ง ํ์ผ ๋ฐ์ดํฐ๋ฅผ ๋ง๋ค์๊ณ , ๋ฐ์ด๋๋ฆฌ ํ์ ์ผ๋ก ์๋ต์ ์ฃผ๋๋ก ๋ชจํนํ์๋ค.
- ๋ก์ปฌ pc์ ์ ์ฅ๋์ด์๋ ์ด๋ฏธ์ง๊ฐ ์๋ base64 ํ์์ ํ์ผ์ ๊ฐ์ ธ์ค๊ธฐ
- base64 ํ์ผ ๊ฐ์ ธ์ค๋ ๋ฒ
- ๊ตฌ๊ธ๋ง์ผ๋ก ์ฐพ์ ์๋ฌด ์ด๋ฏธ์ง ๋งํฌ๋ฅผ base64 ์ธ์ฝ๋ฉ ์ฌ์ดํธ๋ฅผ ์ด์ฉํด์ base64๋ก ๋ณํํด์ค๋ค.
- base64๋?
- 0๊ณผ 1๋ก ์ด๋ค์ง ๋ฐ์ด๋๋ฆฌ ํ์์ผ๋ก ์ธ์ฝ๋ฉ ๋์ด์๋ ๋ฐ์ดํฐ๋ฅผ ํ ์คํธ ํ์์ผ๋ก ๋ณํํ ๊ฒ
- base64 ํ์ผ ๊ฐ์ ธ์ค๋ ๋ฒ
- ํ
์คํธํ์์ base64 ๋ฐ์ดํฐ๋ฅผ Binary ๋ฐ์ดํฐ๋ก ๋ณํํ๊ธฐ
- js์ ๋ด์ฅ ํจ์์ธ atob()๋ฅผ ์ฌ์ฉํด์ ์ด์ง๋ฐ์ดํฐ๋ก ๋ณํ์ํจ๋ค.
- atob(encodedData) ํจ์๋ Base64 (en-US) ์ธ์ฝ๋ฉ๋ ๋ฌธ์์ด ๋ฐ์ดํฐ๋ฅผ ๋์ฝ๋ฉํ๋ค.
- ์๋ฐ์คํฌ๋ฆฝํธ์์ Binary ๋ฐ์ดํฐ๋ฅผ ์ง์ ๋ค๋ฃฐ ์ ์๋๋ก ArrayBuffer ๊ฐ์ฒด ๋ง๋ค์ด์ฃผ๊ธฐ
- ArrayBuffer๋?
- ๋ฐ์ดํธ๋ก ๊ตฌ์ฑ๋ ๋ฐฐ์ด
- Js์์ ์์๋ฐ์ดํฐ ๋ฐ์ด๋๋ฆฌ ๋ฐ์ดํฐ๋ฅผ ์ ์ฅํ ์ ์๋ ๋ฉ๋ชจ๋ฆฌ ๊ณต๊ฐ
- new ArrayBuffer(16) //16๋ฐ์ดํธ ํฌ๊ธฐ์ ๋ฒํผ๋ฅผ ์์ฑ
- ์ฐ์์ผ๋ก ๊ณ ์ ๋ ํฌ๊ธฐ์ ๋ฉ๋ชจ๋ฆฌ ๊ณต๊ฐ(16๋ฐ์ดํธ ๋ฉ๋ชจ๋ฆฌ ๊ณต๊ฐ)์ ๋ฐ์ด๋๋ฆฌ ๋ฐ์ดํฐ๋ฅผ ์ ์ฅ
- ์ด๊ธฐ๊ฐ์ 0์ผ๋ก ์ฑ์์ ธ์์
- ArrayBuffer๋?
- ArrayBuffer์ ์ ์ฅ๋ ๋ฐ์ด๋๋ฆฌ ๋ฐ์ดํฐ์ ์ง์ ์ ๊ทผํ ์ ์๋ ๊ฐ์ฒด์ธ Unit8Array๋ฅผ ์์ฑ
- ArrayBuffer๋ ๋ฐ์ดํธ๋ก๋ง ๊ตฌ์ฑ๋ ๋ฐฐ์ด๋ก "๋ฐ์ดํธ ๋ฐฐ์ด"์ด๋ผ๊ณ ๋ ๋ถ๋ฆ
- ArrayBuffer์ ๋ด๊ธด ์ ๋ณด๋ฅผ ์ง์ ์์ ํ๋ ๊ฒ์ ๋ถ๊ฐ๋ฅ
- ๋ฐ๋ผ์ ArrayBufferView ๊ฐ์ฒด๋ฅผ ํตํด ๋ฒํผ๋ฅผ ํน์ ํ์์ผ๋ก ๋ํ๋์ผ๋ก์จ ๋ฒํผ์ ๋ด์ฉ์ ์ฝ๊ณ ์ฐ๊ธฐ๊ฐ ๊ฐ๋ฅ
- ArrayBufferView
- TypedArray, DataView 2์ข ๋ฅ์ ๊ฐ์ฒด๊ฐ ์์
- TypedArray (ํ์ํ ๋ฐฐ์ด)
- Unit8Array, Uint16Array, Float32Array ๋ฑ์ด ์์
- Unit8Array
- 8๋นํธ = 1๋ฐ์ดํธ๋ก ๊ฐ ๋ฐ์ดํธ ๋ณ๋ก ์ ๊ทผ ๊ฐ๋ฅํ view ๊ฐ์ฒด
- ArrayBufferView
- ArrayBuffer -> Blob(Binary Large Object) ํ์์ผ๋ก ๋ฐ๊พธ๊ธฐ
- Blob?
- ์ด๋ฏธ์ง, ์ค๋์ค, ๋น๋์ค ๊ฐ์ ๋ฉํฐ๋ฏธ๋์ด ํ์ผ ๋ฐ์ด๋๋ฆฌ ๋ฐ์ดํฐ๋ฅผ ๊ฐ์ฒด ํํ๋ก ์ ์ฅํ ๊ฒ
- base64๊ฐ ๋ฐ์ด๋๋ฆฌ ๋ฐ์ดํฐ๋ฅผ ํ ์คํธ๋ก ์ ์ฅํ๋ค๋ฉด Blob์ ๊ฐ์ฒด ํ์์ผ๋ก ์ ์ฅํ๋ค.
- ๋ง๋ ArrayBuffer๋ฅผ ์ธ์๋ก ๋ฃ์ด์ฃผ์ด new Blob(ArrayBuffer) ๊ฐ์ฒด๋ฅผ ๋ง๋ฌ
- Blob?
msw handler ์ ์ฒด ์ฝ๋
const base64mockImg =
'';
const mockImg = atob(base64mockImg.split(',')[1]); // data:image/jpeg;base64 ๋ผ์ด๋ด์ ๋ฐ์ด๋๋ฆฌ ๋ฐ์ดํฐ๋ก ๋์ฝ๋ฉ
export const siteHandler = [
//์ฌ์ง ํ์ผ ๋ค์ด๋ก๋
rest.get(`photos/:photoId/download`, (req, res, ctx) => {
const arrayBuffer = new ArrayBuffer(mockImg.length);
const view = new Uint8Array(arrayBuffer);
for (let i = 0; i < mockImg.length; i++) {
view[i] = mockImg.charCodeAt(i) & 0xff;
}
return res(
ctx.status(200),
ctx.set('Content-Type', 'image/jpeg'),
ctx.body(new Blob([arrayBuffer])),
);
}),
];
mock ์ด๋ฏธ์ง๋ฅผ ์๋ต์ผ๋ก ์ฃผ๊ธฐ์ํด ์ ํ๋ก์ธ์ค๋๋ก ์ฐจ๋ก๋๋ก ์ฝ๋๋ฅผ ์์ฑํ๊ณ , ์๋ต ๋ฐ๋์ arrayBuffer๋ฅผ ๋ด์ Blob ๊ฐ์ฒด๋ฅผ ๋ฆฌํดํ๋ฉด, ์ฑ๊ณต์ ์ผ๋ก response๊ฐ ๋ค์ด์จ๋ค.
response body์ ๋ด๊ฒจ์ค๋ ๋ชจ์ต ์ฝ์๋ก ํ์ธ
์ปดํฌ๋ํธ ์ฝ๋
export const downloadSiteImageFile = async (photoId: string) => {
try {
const { data }: { data: Blob } = await axios.get(
`http://localhost:5173/photos/${photoId}/download`,
{ responseType: 'blob' },
);
return data;
} catch (err) {
console.log(err);
}
};
export const PhotoCard = () => {
const [imgUrlFromBlob, setImgUrlFromBlob] = useState<string>();
useEffect(() => {
if (photoData.id) {
downloadSiteImageFile(photoData.id)
.then((res) => {
console.log(res, 'axios res');
if (res) {
const urlFromBlob = URL.createObjectURL(res);
setImgUrlFromBlob(urlFromBlob);
}
})
.catch((err) => console.log(err));
}
return () => {
if (imgUrlFromBlob) {
URL.revokeObjectURL(imgUrlFromBlob);
} //๋ธ๋ผ์ฐ์ ์ ๋ฉ๋ชจ๋ฆฌ๊ด๋ฆฌ๋ฅผ ์ํด URL ๊ฐ์ฒด ํด์
};
}, []);
return (
<CardMedia
sx={{
width: 160,
height: 160,
border: 'solid 1px',
}}
image={imgUrlFromBlob}>
</CardMedia>
);
}
์ค์ Blob ๊ฐ์ฒด๋ฅผ ํ๋ฐํธ ํ๋ฉด์ ๋ํ๋ด๊ธฐ ์ํด์ ์ด๋ฏธ์ง tag์ URL.createObjectURL ์ ์ฌ์ฉํด์ url ์ ์์ฑํ๋ค.
์ปดํฌ๋ํธ ์ฝ๋์์ ์ฌ์ง ํ์ผ ๋ค์ด๋ก๋ํด์จ ์๋ต response๋ฅผ ์ฐ์ด๋ณด๋ฉด Blob ๊ฐ์ฒด๊ฐ ์ ๋ค์ด์ค๋ ๊ฒ์ ํ์ธํ ์ ์๋ค.
Reference
https://www.daleseo.com/mock-service-worker/
https://mswjs.io/docs/recipes/binary-response-type