๋ณธ๋ฌธ ๋ฐ”๋กœ๊ฐ€๊ธฐ

ํ”„๋กœ์ ํŠธ/Airbnb Clone

(13)
Github Actions ์œผ๋กœ CI ํ…Œ์ŠคํŠธ ์ž๋™ํ™” ์ค‘ ๊ฒช์€ ์ด์Šˆ ์ •๋ฆฌ (Embedded Redis, Profile, Spring Rest Docs) 210๊ฐœ์˜ ํ…Œ์ŠคํŠธ ์ฝ”๋“œ์™€ 99% ํ…Œ์ŠคํŠธ ์ปค๋ฒ„๋ฆฌ์ง€์ˆ™์†Œ ์˜ˆ์•ฝ ํ”Œ๋žซํผ์„ ๊ฐœ๋ฐœํ•˜๋ฉฐ ์ด 210์—ฌ ๊ฐœ์˜ ํ…Œ์ŠคํŠธ ์ฝ”๋“œ๋ฅผ ์ž‘์„ฑํ•˜๊ณ , 99% ํ…Œ์ŠคํŠธ ์ปค๋ฒ„๋ฆฌ์ง€๋ฅผ ๋‹ฌ์„ฑํ–ˆ์Šต๋‹ˆ๋‹ค. ํ•˜์ง€๋งŒ ์ฝ”๋“œ์— ๋ณ€๊ฒฝ ์‚ฌํ•ญ์ด ์žˆ์„ ๋•Œ๋งˆ๋‹ค ์ „์ฒด ํ…Œ์ŠคํŠธ๋ฅผ ์ˆ˜๋™์œผ๋กœ ์‹คํ–‰ํ•ด์•ผ ํ–ˆ๊ณ , ์ด๋Š” ์ ์  ๋ถˆํŽธํ•œ ์ž‘์—…์ด ๋˜์—ˆ์Šต๋‹ˆ๋‹ค. ๊ทธ๋ž˜์„œ CI/CD ํˆด์ธ GitHub Actions๋ฅผ ๋„์ž…ํ•˜์—ฌ push ๋˜๋Š” pull_request๊ฐ€ ๋ฐœ์ƒํ•  ๋•Œ ์ž๋™์œผ๋กœ ํ…Œ์ŠคํŠธ๊ฐ€ ์‹คํ–‰๋˜๋„๋ก ์„ค์ •ํ–ˆ์Šต๋‹ˆ๋‹ค. GitHub Actions๋ฅผ ์„ ํƒํ•œ ์ด์œ GitHub Actions๋ฅผ ์„ ํƒํ•œ ์ด์œ ๋Š” ๋‹จ์ˆœํ•ฉ๋‹ˆ๋‹ค.Jenkins์ฒ˜๋Ÿผ ๋ณ„๋„์˜ ์„œ๋ฒ„๋ฅผ ๊ตฌ์ถ•ํ•  ํ•„์š”๊ฐ€ ์—†์œผ๋ฉฐ,๋‹จ์ˆœํžˆ .github/workflows ๋””๋ ‰ํ† ๋ฆฌ์— YAML ํŒŒ์ผ๋งŒ ์ž‘์„ฑํ•˜๋ฉด ๋ฐ”๋กœ ๋™์ž‘ํ•˜๊ธฐ ๋•Œ๋ฌธ์ž…๋‹ˆ๋‹ค.๋˜ํ•œ Groovy์™€ ๊ฐ™์€ ์ƒˆ๋กœ์šด ๋ฌธ๋ฒ•์„..
ํŠธ๋žœ์žญ์…˜ ๋กค๋ฐฑ ์‹œ ๊ฒฐ์ œ ์ทจ์†Œ ์š”์ฒญํ•˜๊ธฐ - @TransactionalEventListener & ApplicationEventPublisher ๋ชจํ‚น ์ด์Šˆ ๊ฒฐ์ œ ํ”„๋กœ์„ธ์Šค๋ฅผ ์„ค๊ณ„ํ•˜๋ฉด์„œ, ๊ธฐ์กด์—๋Š” try-catch๋ฅผ ํ™œ์šฉํ•ด ํŠธ๋žœ์žญ์…˜ ๋‚ด๋ถ€์—์„œ ๋ฐœ์ƒํ•˜๋Š” ๋Ÿฐํƒ€์ž„ ์˜ˆ์™ธ๋ฅผ ์ฒ˜๋ฆฌํ•˜๊ณ , ๊ฒฐ์ œ ์ทจ์†Œ ์š”์ฒญ์„ ๋ณด๋ƒˆ์Šต๋‹ˆ๋‹ค. ๊ทธ๋Ÿฌ๋‚˜, ์ด์— ๋Œ€ํ•œ ํ•œ๊ณ„๋ฅผ ๋Š๊ผˆ๊ณ  ํŠธ๋žœ์žญ์…˜ ๋กค๋ฐฑ ์‹œ ๊ฒฐ์ œ ์ทจ์†Œ ์š”์ฒญ์„ ๋ช…ํ™•ํ•˜๊ณ  ์ง๊ด€์ ์œผ๋กœ ์ฒ˜๋ฆฌํ•˜๊ธฐ ์œ„ํ•ด @TransactionalEventListener๋ฅผ ๋„์ž…ํ–ˆ์Šต๋‹ˆ๋‹ค.   ๊ธฐ์กด try-catch ๋ฐฉ์‹์˜ ํ•œ๊ณ„ try-catch์—์„œ๋Š” ์ฃผ๋กœ DB ์˜ˆ์™ธ(์˜ˆ: DataIntegrityViolationException)๋‚˜ ๋น„์ฆˆ๋‹ˆ์Šค ๋กœ์ง์—์„œ ๋ฐœ์ƒํ•˜๋Š” ์˜ˆ์™ธ๋ฅผ ์ฒ˜๋ฆฌํ•˜๊ณ , ๊ฒฐ์ œ ์ทจ์†Œ ์š”์ฒญ์„ ๋ณด๋ƒˆ์Šต๋‹ˆ๋‹ค. ํ•˜์ง€๋งŒ, try-catch๋ฅผ ์‚ฌ์šฉํ•˜๋ฉด ์˜ˆ์™ธ๋ฅผ ์ •์ƒ์ ์ธ ํ๋ฆ„์œผ๋กœ ์ œ์–ดํ•˜๋Š” ๊ฒƒ์ฒ˜๋Ÿผ ๋ณด์ด๊ฒŒ ๋ฉ๋‹ˆ๋‹ค. ์ด๋Š” "ํŠธ๋žœ์žญ์…˜์ด ๋กค๋ฐฑ๋˜์—ˆ์„ ๋•Œ ๊ฒฐ์ œ ์ทจ์†Œ๋ฅผ ์ˆ˜ํ–‰ํ•œ๋‹ค"๋ผ๋Š” ์˜๋ฏธ๋ฅผ ์ •ํ™•ํžˆ ํ‘œํ˜„ํ•˜์ง€..
[ํŠธ๋Ÿฌ๋ธ”์ŠˆํŒ…] RestTemplate ๋กœ๊น… ์ค‘ 404 ์‘๋‹ต์— ๋Œ€ํ•œ FileNotFoundException ํ•ด๊ฒฐ ๊ฒฐ์ œ ์Šน์ธ POST ์š”์ฒญ ์‹œ RestTemplate ๋กœ๊ทธ์—์„œ ๋ฐœ์ƒํ•œ ์˜ค๋ฅ˜ ๊ฒฐ์ œ ๊ธฐ๋Šฅ์„ ๊ฐœ๋ฐœํ•˜๋ฉด์„œ, ๊ฒฐ์ œ ์Šน์ธ ์š”์ฒญ์„ ๋ณด๋‚ด๋Š” RestTemplate์— ํƒ€์ž„์•„์›ƒ์„ ์ ์šฉํ•˜๊ณ , ์š”์ฒญ๊ณผ ์‘๋‹ต์— ๋Œ€ํ•œ ๋กœ๊น…์„ ์ถ”๊ฐ€ํ•˜๊ณ ์ž ํ–ˆ์Šต๋‹ˆ๋‹ค.  Baeldung์˜ "RestTemplage Logging" ๊ณผ ๋ง๋‚˜๋‹ˆ ๊ฐœ๋ฐœ์ž๋‹˜์˜ ๋กœ๊น… ์„ ์ฐธ๊ณ ํ•ด์„œ RestTemplate ์— ๋กœ๊ทธ๋ฅผ ์ถ”๊ฐ€ ํ–ˆ์ง€๋งŒ ์‘๋‹ต์„ ๋ฐ›์„ ๋•Œ, FileNotFoundException ์ด ๋ฐœ์ƒํ•˜๋ฉฐ ์•„๋ž˜์™€ ๊ฐ™์€ ์Šคํƒ ํŠธ๋ ˆ์ด์Šค๋ฅผ ํ™•์ธํ–ˆ์Šต๋‹ˆ๋‹ค.java.io.FileNotFoundException: https://api.tosspayments.com/v1/payments/confirmjava.base/sun.net.www.protocol.http.HttpURLConnection..
@Async ์ด๋ฉ”์ผ ์ „์†ก ๊ณ ๋„ํ™”: ์žฌ์‹œ๋„, ์˜ˆ์™ธ ์ฒ˜๋ฆฌ, ํ…Œ์ŠคํŠธ ์ธ์ฆ ์ด๋ฉ”์ผ ์ „์†ก ์‹œ, ๊ฐ„ํ˜น ๋„คํŠธ์›Œํฌ ์˜ค๋ฅ˜๋กœ ์ธํ•ด ๋ฉ”์ผ์ด ์ •์ƒ์ ์œผ๋กœ ์ „์†ก๋˜์ง€ ์•Š๋Š” ๊ฒฝ์šฐ๊ฐ€ ๋ฐœ์ƒํ•ฉ๋‹ˆ๋‹ค.์ด๋Ÿฌํ•œ ์ƒํ™ฉ์—์„œ ์‚ฌ์šฉ์ž์—๊ฒŒ ์ฆ‰์‹œ ์˜ค๋ฅ˜๋ฅผ ์•Œ๋ฆฌ๋Š” ๋Œ€์‹ , ์žฌ์‹œ๋„ ๋กœ์ง์„ ํ†ตํ•ด ์ด๋ฉ”์ผ ์ „์†ก ์„ฑ๊ณต ๊ฐ€๋Šฅ์„ฑ์„ ๋†’์ด๋Š” ๊ฒƒ์ด ์‚ฌ์šฉ์ž ๊ฒฝํ—˜์„ ํ–ฅ์ƒํ•˜๋Š” ์ค‘์š”ํ•œ ์š”์†Œ๋ผ๊ณ  ํŒ๋‹จํ–ˆ์Šต๋‹ˆ๋‹ค. ๊ทธ๋ž˜์„œ, ์žฌ์‹œ๋„ ์ถ”๊ฐ€ ์‹œ ๊ณ ๋ คํ•œ ๊ฒƒ๋“ค์„ ๋‹ค์Œ๊ณผ ๊ฐ™์€ ๋ชฉ์ฐจ๋กœ ์†Œ๊ฐœํ•ฉ๋‹ˆ๋‹ค.์žฌ์‹œ๋„(@Retryable) ์กฐ๊ฑด ์„ค์ •์žฌ์‹œ๋„ ์˜ˆ์™ธ์™€ ์“ฐ๋ ˆ๋“œ ํ’€ ์˜ˆ์™ธ๋ฅผ ๊ตฌ๋ถ„ํ•ด์„œ ์ฒ˜๋ฆฌํ•˜๊ธฐ์žฌ์‹œ๋„ ๋ฐ ๋ณต๊ตฌ๊ฐ€ ์ž˜ ๋˜์—ˆ๋Š”์ง€ ํ…Œ์ŠคํŠธํ•˜๊ธฐ* ๋น„๋™๊ธฐ์—์„œ ์žฌ์‹œ๋„ ๋กœ์ง ๊ตฌํ˜„์„ ์œ„ํ•ด Baeldung(async-retry)๋ฅผ ์ฐธ๊ณ ํ–ˆ์Šต๋‹ˆ๋‹ค. ์žฌ์‹œ๋„(@Retryable) ์กฐ๊ฑด ์„ค์ •๋จผ์ €, ์ด๋ฉ”์ผ ์ „์†ก์— ์žฌ์‹œ๋„ ์กฐ๊ฑด์„ ์„ค์ •ํ•˜๊ธฐ ์œ„ํ•ด Spring Mail ๊ด€๋ จ ์˜ˆ์™ธ๋ฅผ ์†Œ๊ฐœํ•ฉ๋‹ˆ๋‹ค.MailAuthentic..
์ด๋ฉ”์ผ ์ „์†ก์„ Async-NonBlocking ์ฒ˜๋ฆฌํ•˜๊ธฐ: ์“ฐ๋ ˆ๋“œ ํ’€ ์„ค์ •๊ณผ API ์ตœ๋Œ€ ์‘๋‹ต ์†๋„ 1.4์ดˆ → 0.01์ดˆ ๊ฐœ์„  @Async ๋ฅผ ๋„์ž…ํ•˜๋ฉฐ ์ปจํ…์ŠคํŠธ ์Šค์œ„์นญ๊ณผ ์“ฐ๋ ˆ๋“œ ๋ฉ”๋ชจ๋ฆฌ ๊ตฌ์กฐ ์ง€์‹์„ ๋ฐ”ํƒ•์œผ๋กœ ์“ฐ๋ ˆ๋“œ ํ’€ ์„ค์ •์„ ํ•™์Šตํ•˜๊ณ  ์ ์šฉํ•˜๋Š” ๊ณผ์ •์„ ์†Œ๊ฐœํ•˜๋ฉฐ Jmeter ๋ฅผ ํ†ตํ•ด API ์‘๋‹ต ์†๋„๊ฐ€ ๊ฐœ์„ ๋˜์—ˆ๋Š”์ง€ ํ™•์ธํ•˜๋Š” ๊ณผ์ •์„ ๋‹ค์Œ๊ณผ ๊ฐ™์€ ๋ชฉ์ฐจ๋กœ ์†Œ๊ฐœํ•ฉ๋‹ˆ๋‹ค.์ด๋ฉ”์ผ ์ „์†ก์„ ๋น„๋™๊ธฐ๋กœ ๊ฒฐ์ •ํ•œ ์ด์œ ์“ฐ๋ ˆ๋“œ ํ’€ ์‚ฌ์šฉ ์—ฌ๋ถ€ ๊ณ ๋ ค์“ฐ๋ ˆ๋“œ ํ’€ ์„ค์ • ๊ณ ๋ ค ์‚ฌํ•ญ์“ฐ๋ ˆ๋“œ ํ’€ ์„ค์ •ThreadPoolTaskExecutor ๋™์ž‘ ๋ฐฉ์‹์“ฐ๋ ˆ๋“œ ํ’€ ์˜ค๋ฅ˜ ์ฒ˜๋ฆฌ ์„ค์ •Jmeter ๋กœ @Async ๋„์ž… ์ ์šฉ ์ „๊ณผ ํ›„์˜ API ์‘๋‹ต ์‹œ๊ฐ„์„ ๋น„๊ต@Async ์‚ฌ์šฉ ์‹œ ์ฃผ์˜์  ์ด๋ฉ”์ผ ์ „์†ก์„ ๋น„๋™๊ธฐ๋กœ ๊ฒฐ์ •ํ•œ ์ด์œ ์—์–ด๋น„์•ค๋น„ ํด๋ก  ํ”„๋กœ์ ํŠธ์—์„œ๋Š” ์ด๋ฉ”์ผ ์ธ์ฆ ๊ธฐ๋ฐ˜์˜ ํšŒ์›๊ฐ€์ž…/๋กœ๊ทธ์ธ์„ ์ง„ํ–‰ํ•˜๊ณ  ์žˆ์Šต๋‹ˆ๋‹ค.  ์ด๋ฉ”์ผ ์ „์†ก์—๋Š” Spring Mail์˜ JavaMailSender๋ฅผ ์‚ฌ์šฉํ•˜๊ณ  ์žˆ์œผ๋ฉฐ..
[๊ฒฐ์ œ ์Šน์ธ ํ”„๋กœ์„ธ์Šค ์„ค๊ณ„] ๊ฒฐ์ œ ์Šน์ธ ์š”์ฒญ๊ณผ DB ํŠธ๋žœ์žญ์…˜์˜ ์ˆœ์„œ๊ฐ„์˜ ์žฅ๋‹จ์ ์„ ๋น„๊ต ๋ถ„์„ ์‚ฌ์ด๋“œ ํ”„๋กœ์ ํŠธ ์ค‘, ์ˆ™์†Œ ์˜ˆ์•ฝ์„ ํ•  ๋•Œ์˜ ๊ฒฐ์ œ ํ”„๋กœ์„ธ์Šค๋ฅผ ๊ตฌํ˜„์ค‘์ด์—ˆ์Šต๋‹ˆ๋‹ค. ์˜ˆ์•ฝ์„ ํ™•์ •ํ•  ๋•Œ, ๊ฒฐ์ œ ํ”„๋กœ์„ธ์Šค๋Š” ๋‹ค์Œ ์‚ฌ์ง„์ฒ˜๋Ÿผ ์—ฌ๋Ÿฌ ํ”„๋กœ์„ธ์Šค๋กœ ๋‚˜๋‰˜์–ด ์ง‘๋‹ˆ๋‹ค.  ์ด ๋•Œ, ์ œ๊ฐ€ ๊ณ ๋ คํ•œ ๊ฒƒ์€ ์ด 5๊ฐ€์ง€์ด๋ฉฐ ์ด์— ๋Œ€ํ•ด ์„ค๋ช…์„ ์ง„ํ–‰ํ•ฉ๋‹ˆ๋‹ค. ๊ฒฐ์ œ ํ”„๋กœ์„ธ์Šค์˜ ํŠธ๋žœ์žญ์…˜ ์„ค์ • ๋ฒ”์œ„'๊ฒฐ์ œ ์Šน์ธ์„ ์œ„ํ•œ PG์‚ฌ API ํ˜ธ์ถœ'๊ณผ ํŠธ๋žœ์žญ์…˜์˜ ์ˆœ์„œ'๊ฒฐ์ œ ์Šน์ธ์„ ์œ„ํ•œ PG์‚ฌ API ํ˜ธ์ถœ'์„ ๋™๊ธฐ/๋น„๋™๊ธฐ ์ค‘ ์„ ํƒ ๋™์‹œ์„ฑ ์ œ์–ด ๋ฐฉ๋ฒ•์ž”์•ก ๋ถ€์กฑ์œผ๋กœ ์ธํ•œ ๊ฒฐ์ œ ์Šน์ธ ์‹คํŒจ๋‚˜ ํŠธ๋žœ์žญ์…˜ ๋กค๋ฐฑ ์‹œ ๋ฐœ์ƒํ•˜๋Š” ๊ฒฝ์šฐ๋ฅผ ์–ด๋–ป๊ฒŒ ์ฒ˜๋ฆฌํ•  ๊ฒƒ์ธ์ง€ 1. ๊ฒฐ์ œ ํ”„๋กœ์„ธ์Šค์˜ ํŠธ๋žœ์žญ์…˜ ์„ค์ • ๋ฒ”์œ„'๊ฒฐ์ œ ์Šน์ธ์„ ์œ„ํ•œ PG์‚ฌ API ํ˜ธ์ถœ' ์€ ๋„คํŠธ์›Œํฌ๋ฅผ ํ†ตํ•ด PG์‚ฌ์—๊ฒŒ POST ์š”์ฒญ์„ ํ†ตํ•ด ๊ฒฐ์ œ๋ฅผ ์Šน์ธ ๋ฐ›๋Š” ๊ณผ์ •์œผ๋กœ ํŠธ๋žœ์žญ์…˜์— ํฌํ•จ๋˜์–ด ํŠธ๋žœ์žญ์…˜์ด ๊ธธ์–ด์ง€๊ฒŒ ๋œ๋‹ค๋ฉด ๋‹ค์Œ๊ณผ ๊ฐ™..
[๋กœ๊ทธ์ธ ๋ฐ ์ธ๊ฐ€] Interceptor์—์„œ ArgumentResolver๋กœ ๋ฆฌํŒฉํ† ๋งํ•˜๊ธฐ [์„œ๋ก ]์—์–ด๋น„์•ค๋น„ ํด๋ก  ํ”„๋กœ์ ํŠธ๋ฅผ ์ง„ํ–‰ํ•˜๋ฉฐ, ๋กœ๊ทธ์ธ ํšŒ์›๋งŒ '์ˆ™์†Œ ๋“ฑ๋ก' ๊ณผ '์ˆ™์†Œ ์˜ˆ์•ฝ' ์„ ํ•  ์ˆ˜ ์žˆ๊ธฐ ์œ„ํ•ด(์ธ๊ฐ€๋ฅผ ์ฃผ๊ธฐ ์œ„ํ•ด) ๋กœ๊ทธ์ธํ•œ ํšŒ์›์„ ๋Œ€์ƒ์œผ๋กœ ์„ธ์…˜์„ ์‚ฌ์šฉํ•ด์„œ stateful ํ•˜๊ฒŒ ๊ตฌํ˜„ ํ–ˆ์Šต๋‹ˆ๋‹ค.@PostMapping("/login") public ApiResponse login(@RequestBody @Valid MemberLoginRequest request, HttpSession httpSession) { memberService.login(request.toServiceRequest()); // ๋กœ๊ทธ์ธ์— ์„ฑ๊ณตํ•˜๋ฉด httpSession.setAttribute(LOGIN_MEMBER, true..
[ํŠธ๋Ÿฌ๋ธ” ์ŠˆํŒ…] ํ…Œ์ŠคํŠธ๋กœ ์•Œ์•„๋ณด๋Š” ์ปจํŠธ๋กค๋Ÿฌ์—์„œ HttpSession ์‚ฌ์šฉ ์‹œ ์ฃผ์˜์  Junit5 ๋ฐ Mock ์— ๋Œ€ํ•ด ์„ค๋ช…๋“œ๋ฆฌ์ง€ ์•Š๋Š” ์  ์–‘ํ•ด ๋ถ€ํƒ๋“œ๋ฆฝ๋‹ˆ๋‹ค.ํ…Œ์ŠคํŠธ ์ฝ”๋“œ๋ณด๋‹ค๋Š” ์ปจํŠธ๋กค๋Ÿฌ์—์„œ HttpSession ์„ ์‚ฌ์šฉํ•  ๋•Œ์˜ ์ฃผ์˜์ ์„ ์•Œ์•„๋ณด๋Š” ์‹œ๊ฐ„์„ ๊ฐ€์ ธ๊ฐ€์…จ์œผ๋ฉด ์ข‹๊ฒ ์Šต๋‹ˆ๋‹ค. ์ปจํŠธ๋กค๋Ÿฌ์—์„œ ํŒŒ๋ผ๋ฏธํ„ฐ๋กœ HttpSession ์„ ์ฃผ์ž…๋ฐ›์•„์„œ ์‚ฌ์šฉํ•  ๋•Œ ์ฃผ์˜ํ•  ์ ์ด ์žˆ์Šต๋‹ˆ๋‹ค. ์™œ ์ฃผ์˜ํ•ด์„œ ์‚ฌ์šฉํ•ด์•ผ ํ•˜๋Š”์ง€ HttpSession ๊ณผ HttpServletRequest.getSession() ์˜ ์ฐจ์ด๋ฅผ ํ…Œ์ŠคํŠธ ์ฝ”๋“œ๋กœ ์•Œ์•„๋ณด๊ฒ ์Šต๋‹ˆ๋‹ค.  โญ HttpSession ๋ฅผ ๋ฉ”์„œ๋“œ์˜ ํŒŒ๋ผ๋ฏธํ„ฐ๋กœ ์ฆ‰์‹œ ์ฃผ์ž… ๋ฐ›๊ธฐ์ฝ”๋“œ ์„ค๋ช…- ํšŒ์›์ด ์ธ์ฆ ์ฝ”๋“œ๋ฅผ ์ž˜๋ชป ์ž…๋ ฅํ•ด์„œ ์ด๋ฉ”์ผ ๊ฒ€์ฆ์— ์‹คํŒจํ•˜๋ฉด if ๋ฌธ์ด ๊ฑฐ์ง“์ด ๋ฉ๋‹ˆ๋‹ค.- Session ์— ๊ฐ’์„ ์„ค์ •ํ•˜์ง€ ์•Š๊ณ , ๋ฉ”์„œ๋“œ๊ฐ€ ์ข…๋ฃŒ๋ฉ๋‹ˆ๋‹ค.@GetMapping("/email/authenticat..