๐ค
readFile vs readFileSync (feat: Artillery)
blog
2025-04-04
blog
2025-04-04
MDX ํ์ผ์์ ์ด๋ฏธ์ง๋ฅผ ๋ฐ์์ค๋ ๋ถ๋ถ์ ๋ค์๊ณผ ๊ฐ์ด ๊ตฌํ๋์ด ์์๋ค.
1import fs from "fs";2import sizeOf from "image-size";3import Image from "next/image";4import path from "path";5import { useMemo } from "react";67interface MDXImageProps {8 src: string;9 alt: string;10}1112interface ImageDimensions {13 width: number;14 height: number;15}1617export default function SyncImage({ src, alt }: MDXImageProps) {18 const dimensions: ImageDimensions = useMemo(() => {19 // ํด๋ผ์ด์ธํธ ์ธก์์๋ ์คํํ์ง ์์20 if (typeof window !== "undefined") {21 return { width: 700, height: 475 }; // ๊ธฐ๋ณธ๊ฐ ์ค์ 22 }2324 try {25 // public ํด๋ ๋ด์ ์ด๋ฏธ์ง ๊ฒฝ๋ก ๊ตฌ์ฑ26 const imagePath = path.join(process.cwd(), "public", src);2728 // ํ์ผ์ด ์กด์ฌํ๋์ง ํ์ธ29 if (!fs.existsSync(imagePath)) {30 console.warn(`Image not found: ${imagePath}`);31 return { width: 700, height: 475 }; // ์ด๋ฏธ์ง๋ฅผ ์ฐพ์ง ๋ชปํ ๊ฒฝ์ฐ ๊ธฐ๋ณธ๊ฐ ๋ฐํ32 }3334 // ํ์ผ์ ๋ฐ์ด๋๋ฆฌ๋ก ์ฝ๊ธฐ35 const buffer = fs.readFileSync(imagePath);3637 // image-size๋ฅผ ์ฌ์ฉํ์ฌ ์ด๋ฏธ์ง ํฌ๊ธฐ ๊ณ์ฐ38 const dimensions = sizeOf(buffer);3940 return {41 width: dimensions.width || 700,42 height: dimensions.height || 475,43 };44 } catch (error) {45 console.error("Error getting image dimensions:", error);46 return { width: 700, height: 475 }; // ์ค๋ฅ ๋ฐ์ ์ ๊ธฐ๋ณธ๊ฐ ๋ฐํ47 }48 }, [src]);4950 return (51 <Image52 src={src}53 alt={alt || ""}54 width={dimensions.width}55 height={dimensions.height}56 style={{57 maxWidth: "100%",58 height: "auto",59 }}60 className="rounded-[10px]"61 />62 );63}
๊ฐ์ฅ ํต์ฌ์ ์ธ ๋ถ๋ถ์ ๋ค์๊ณผ ๊ฐ๋ค.
1// ํ์ผ์ ๋ฐ์ด๋๋ฆฌ๋ก ์ฝ๊ธฐ2const buffer = fs.readFileSync(imagePath);34// image-size๋ฅผ ์ฌ์ฉํ์ฌ ์ด๋ฏธ์ง ํฌ๊ธฐ ๊ณ์ฐ5const dimensions = sizeOf(buffer);67return {8 width: dimensions.width || 700,9 height: dimensions.height || 475,10};
readFileSync๋ฅผ ํตํด ์ด๋ฏธ์ง๋ฅผ ๋ถ๋ฌ์ค๊ณ , sizeOf๋ฅผ ์ด์ฉํด ๋ถ๋ฌ์จ ์ด๋ฏธ์ง์ ์ฌ์ด์ฆ๋ฅผ ๊ตฌํด์ return ํ๋ค.
์ด๋ ๊ฒ๋ง ๋ณด๋ฉด, ์๋ฌด๋ฐ ์๊ฐ์ด ๋ค์ง ์๋๋ค.
๊ด์ฐฎ์ ์ฝ๋ ์๋๊ฐ?
๊ทธ๋ฌ๋ ๊ณต์ ๋ฌธ์์์๋ ๋ค์๊ณผ ๊ฐ์ด ๋งํ๊ณ ์๋ค.
1Reading from a file Syncronously (not recommended)2v1.x of this library had a sync API, that internally used sync file reads.3This isn't recommended because this blocks the node.js main thread, which reduces the performance, and prevents this library from being used concurrently.4However if you still need to use this package syncronously, you can read the file syncronously into a buffer, and then pass the buffer to this library.
๋๊ธฐ์ ์ผ๋ก ํ์ผ์ ์ฝ์ด์ค๋ฉด node.js์ ๋ฉ์ธ ์ฐ๋ ๋๋ฅผ ๋ฉ์ถ๊ฒ ํ๊ณ , ์ด๋ ๊ณง ์ฑ๋ฅ ์ ํ๋ก ์ด์ด์ง๋ค๊ณ ํ๋ค.
๋ค์ ๋ต๋ณ์์๋ โ์ ๋โ ์ฐ์ง ๋ง๋ผ๊ณ ํ๊ธฐ๋ ํ๋ค.
ํด๋น ์ฝ๋๊ฐ ๋ฐ๋์ ์ ๋ต์ ์๋๋ค.
1import Image from "next/image";2import { getPlaiceholder } from "plaiceholder";3import fs from "fs/promises";4import path from "path";56interface MDXImageProps {7 src: string;8 alt: string;9}1011interface ImageMetadata {12 width: number;13 height: number;14 blurDataURL?: string;15}1617// This function can be used with getStaticProps or in a Server Component18export async function getImageMetadata(src: string): Promise<ImageMetadata> {19 try {20 const imagePath = path.join(process.cwd(), "public", src);21 const buffer = await fs.readFile(imagePath);2223 // Get image dimensions and optional blur placeholder24 const { metadata, base64 } = await getPlaiceholder(buffer);2526 return {27 width: metadata.width,28 height: metadata.height,29 blurDataURL: base64,30 };31 } catch (error) {32 console.error(`Error processing image ${src}:`, error);33 return { width: 700, height: 475 };34 }35}3637// For use in Server Components (App Router)38export default async function AsyncImage({ src, alt }: MDXImageProps) {39 const imageMetadata = await getImageMetadata(src);4041 return (42 <Image43 src={src}44 alt={alt || ""}45 width={imageMetadata.width}46 height={imageMetadata.height}47 placeholder={imageMetadata.blurDataURL ? "blur" : "empty"}48 blurDataURL={imageMetadata.blurDataURL}49 style={{50 maxWidth: "100%",51 height: "auto",52 }}53 className="rounded-[10px]"54 />55 );56}
๋จ์ํ readFileSync๋ฅผ readFile๋ก ๋ฐ๊ฟ์คฌ๋ค. (์ง์~) ๊ทธ๋ฆฌ๊ณ plaiceholder ๋ผ์ด๋ธ๋ฌ๋ฆฌ๋ฅผ ์ด์ฉํด ์ด๋ฏธ์ง๊ฐ ๋ ๋ก๋ฉ ๋์์ ๋, ๋ธ๋ฌ ์ฒ๋ฆฌ๋ฅผ ํด์ฃผ์๋ค. ๊ณผ์ฐ ์ด ๋์ ์ฑ๋ฅ ์ฐจ์ด๋ ์ผ๋ง๋ ๋ ๊น?
๊ฒฐ๋ก ๋ถํฐ ๋งํ์๋ฉด, ์ฐจ์ด๊ฐ ์์ฃผ ๋ฏธ๋ฏธํ๋ค. ๊ทธ๋์ ๋๋ ์ด๋จ ๋ ์ฐจ์ด๊ฐ ๋ฐ์ํ๋์ง ๊ถ๊ธํด์ Artillery๋ฅผ ์ด์ฉํด ๋ถํ ํ ์คํธ๋ฅผ ์งํํ๋ค.
Artillery์ ์๋๋ฆฌ์ค๋ ๋ค์๊ณผ ๊ฐ๋ค. ๋น๋๊ธฐ, ๋๊ธฐ ๋ ๋ค ๋๊ฐ์ ์๋๋ฆฌ์ค์ ๋ถํ๋ฅผ ๊ฐ๋๋ค. ๋จ์ง url์ด โ/test-asyncโ์ธ์ง โ/test-syncโ์ธ์ง์ ๋ํ ์ฐจ์ด๋ง ์๋ค.
1config:2 target: "http://localhost:3000"3 phases:4 - name: "์ด๊ธฐ ์ค๋น ๋จ๊ณ"5 duration: 106 arrivalRate: 100 # ์ฒซ 10์ด ๋์ ์ด๋น 50๋ช7 - name: "์ค๊ฐ ๋ถํ ๋จ๊ณ"8 duration: 109 arrivalRate: 500 # ๋ค์ 10์ด ๋์ ์ด๋น 100๋ช10 - name: "๊ณ ๋ถํ ๋จ๊ณ"11 duration: 2012 arrivalRate: 1000 # ๋ง์ง๋ง 20์ด ๋์ ์ด๋น 200๋ช13scenarios:14 - name: "ํ ์คํธ ์๋ํฌ์ธํธ"15 flow:16 - get:17 url: "/test-async"
์ผ์ชฝ์ด ๋น๋๊ธฐ, ์ค๋ฅธ์ชฝ์ด ๋๊ธฐ์ด๋ค.
์ฌ์ค ๋์ ์ฐจ์ด๋ ๊ฑฐ์ ์๋ค. (ํํํ) ์ด๋ ๊ฒ ๊ฒฐ๋ก ์ ๋งบ์ผ๋ฉด ์ฌ๋ฏธ๊ฐ ์์ผ๋๊น! ๋ฐ์ดํฐ๋ฅผ ์ข๋ง ๋ ๋ถ์ํด๋ณด์.
์ฐ์ , ํธ๋ํฝ์ด ์ ์ ์ํฉ์์๋ ๋๊ธฐ์ ๋น๋๊ธฐ ๋ฐฉ์ ๋ชจ๋ ํ๊ท ์๋ต ์๊ฐ์ด 0ms ์ ๋๋ก ๋น์ทํ๊ฒ ๋์ํ๋ค. ํ์ง๋ง ๋ถํ๊ฐ ์ฆ๊ฐํ ์๋ก ๋ ๋ฐฉ์์ ํน์ฑ์ด ๋ช ํํ๊ฒ ๋๋ฌ๋ฌ๋ค.
์ ๋ถํ ํ๊ฒฝ(100 RPS)
๋ ๋ฐฉ์ ๋ชจ๋ ํ๊ท ์๋ต ์๊ฐ์ด ์ฝ 0ms๋ก ๋น ๋ฅด๊ฒ ๋์ํ์ผ๋ฉฐ, ์ง์ฐ ์๊ฐ๋ ๋ฏธ๋ฏธํ๋ค. ์ด ๊ตฌ๊ฐ์์๋ ํฐ ์ฐจ์ด๋ฅผ ๋๋ ์ ์์๋ค.
์ค๊ฐ ๋ถํ ํ๊ฒฝ(500 RPS)
๊ณ ๋ถํ ํ๊ฒฝ(์: 1000 RPS)
๊ฒฐ๋ก ์ ์ผ๋ก, ์ ๋ถํ์์ ์ค๋ถํ๊น์ง๋ ๋น๋๊ธฐ ๋ฐฉ์์ด ์๋ต ์ง์ฐ ์์ด ๋ ์์ ์ ์ผ๋ก ๋์ํ์ผ๋, ๊ณ ๋ถํ ํ๊ฒฝ์์๋ ๋ ๋ฐฉ์ ๋ชจ๋ ํ๊ณ์ ๋๋ฌํด ์๋ต ์ง์ฐ๊ณผ ์คํจ์จ์ด ๊ธ๊ฒฉํ ์ฆ๊ฐํ๋ค.
๋๊ธฐ ๋ฐฉ์์ ๊ฐ๋ณ ์์ฒญ์ ์ง์ฐ์ ๋ฎ๊ฒ ์ ์งํ์ง๋ง, ์ ์ฒด ์ฒ๋ฆฌ๋ ์ธก๋ฉด์์๋ ํด๋ผ์ด์ธํธ ์์ ๊ณ ๊ฐ ๋ฌธ์ ๊ฐ ๋ฐ์ํ๋ค. ๋ฐ๋ฉด, ๋น๋๊ธฐ ๋ฐฉ์์ ์ด๊ธฐ์๋ ๋์ ์ฒ๋ฆฌ๋์ ๋ณด์์ผ๋, ์๊ณ์น๋ฅผ ๋์ด์๋ฉด ์๋ฒ ์์ ๋ถ์กฑ๊ณผ ์ฐ๊ฒฐ ๊ฑฐ๋ถ ์ค๋ฅ๋ก ์ธํด ์ฑ๋ฅ์ด ๊ธ๋ฝํ๋ค.
๋ง์ฝ ์ค์ ์๋ฒ์์ ์ด์ ๋์ ๋ถํ๊ฐ ๋ฐ์ํ๋ค๋ฉด, ๋ฌด์กฐ๊ฑด ํฐ์ง๋ค. ๋ฐ๋ผ์ ๋น๋๊ธฐ ๋ฐฉ์์ ์ด์ ์ ์ด๋ฆฌ๋, ์ฌ์ ์ ์ด๋ฏธ์ง ์ฒ๋ฆฌ(๋ฆฌ์ฌ์ด์ง, placeholder ์์ฑ)๋ฅผ ํด๋๊ณ ์บ์๋ฅผ ํ์ฉํ๋ฉฐ, ์๋น์ค ์ธํ๋ผ๋ฅผ ์ฆ์คํ์ฌ ๋จ์ผ ์๋ฒ์ ๊ณผ๋ถํ๊ฐ ๊ฑธ๋ฆฌ์ง ์๋๋ก ์ด์ํ๋ ๊ฒ์ ์๊ฐํด๋ณผ ์ ์๋ค.
์๋๋, ๋น๋ํ์ผ์ ๋์ปค์ ๋์์ ๋์ปค์์ cpu๋ฐ ๋ฉ๋ชจ๋ฆฌ ์์์ ์กฐ์ ํด๊ฐ๋ฉด์ ๋ถํํ ์คํธ๋ฅผ ํด๋ณด๋ ค ํ์๋ค. ๋์ปค๋ฅผ ๊ฑฐ์ ์ฒ์ ์ฐ๋ค๋ณด๋ ์์ฒญ ํค๋งธ์ง๋ง ์ด์ฐ์ ์ฐ ๋์ปค ์ด๋ฏธ์ง๋ฅผ ๋น๋ํ์ฌ ์ปจํ ์ด๋์ ์ฌ๋ ค์ ์คํ๊น์ง ์์ผฐ๋ค.
1docker run --cpus="12" --memory="1024m" -p 3000:3000 test
์ ์ปค๋งจ๋๋ก ์คํ์ ์ํค๊ณ ๋ถํํ ์คํธ๋ฅผ ํด๋ณด์๋๋ฐ, ๋น๋๊ธฐ ํ ์คํธ์ธ๋ฐ๋ ๋ค ์คํจ๋ฅผ ํ๋ ๊ฑฐ์๋ค. ๊ทธ๋์ ๋ถํ๋ฅผ ๊ฑฐ์ ์๋ค์ํผ ์ค์ ํ๊ณ ํ ์คํธ๋ฅผ ํด๋ณด์๋ค. ๊ทธ๋ฐ๋ฐ๋ ๋ชจ๋ ์คํจํ๋ค. . .๊ทธ๋์ ๋์ปค์ ๋์์ ํ์ง๋ง๊ณ ๊ทธ๋ฅ ๋น๋ํ์ผ์ next start๋ก ๋์ด ๋ค์์ ๋ถํ๋ฅผ ๋ง์ด ๊ฑธ์ด๋ณด๋ ๊ฑธ๋ก ๋ฐ๊ฟจ๋ค.
1phases:2 - name: "์ด๊ธฐ ์ค๋น ๋จ๊ณ"3 duration: 104 arrivalRate: 50 # ์ฒซ 10์ด ๋์ ์ด๋น 50๋ช5 - name: "์ค๊ฐ ๋ถํ ๋จ๊ณ"6 duration: 107 arrivalRate: 100 # ๋ค์ 10์ด ๋์ ์ด๋น 100๋ช8 - name: "๊ณ ๋ถํ ๋จ๊ณ"9 duration: 2010 arrivalRate: 200 # ๋ง์ง๋ง 20์ด ๋์ ์ด๋น 200๋ช
1phases:2 - name: "์ด๊ธฐ ์ค๋น ๋จ๊ณ"3 duration: 104 arrivalRate: 100 # ์ฒซ 10์ด ๋์ ์ด๋น 100๋ช5 - name: "์ค๊ฐ ๋ถํ ๋จ๊ณ"6 duration: 107 arrivalRate: 500 # ๋ค์ 10์ด ๋์ ์ด๋น 500๋ช8 - name: "๊ณ ๋ถํ ๋จ๊ณ"9 duration: 2010 arrivalRate: 1000 # ๋ง์ง๋ง 20์ด ๋์ ์ด๋น 1000๋ช
๊ทธ ๊ฒฐ๊ณผ ๋คํํ ๊ฒฐ๊ณผ๊ฐ ๋์๋ค. (๋ชจ๋ ์คํจ๊ฐ ์๋๊ฑฐ์ ๊ฐ์ฌํจ..ใ ใ ใ )
์ ๋์ปค๋ก ๋์ฐ๋ฉด ์๋๋ ๊ฑด์ง ์์ง๋ ์ ํ์ ํ์ง ๋ชปํ๋ค. ๊ทธ๋ฐ๋ฐ ์ปดํจํฐ๋ฅผ ๊ป๋ค ํค๋ฉด ๋ฑ ์ต์ด 1ํ๋ ๊ด์ฐฎ๊ฒ ๋์จ๋ค. ๊ทธ ์ดํ๋ก๋ ์๋ฌด๋ฆฌ ์ฝํ ๋ถํ๋ฅผ ๊ฑธ์ด๋ ๋ค ์คํจ๋ฅผ ํ๋ค. hmm.. ๋์ค์ ๊ผญ ์์ธ์ ๋ฐํ๊ณ ์ผ ๋ง๊ฒ ๋ค!