Batch Schedular

Crontab

  • history ๊ด€๋ฆฌ ๋ถˆ๊ฐ€๋Šฅ

  • ์‹ค์ˆ˜๋กœ ํ•ด๋‹น ์Šค์ผ€์ฅด๋ง ์ „์ฒด๋ฅผ ๋‚ ๋ฆด ์ˆ˜ ์žˆ์Œ

  • ๋ฐฐ์น˜ ์ˆ˜ํ–‰ ์‹คํŒจ์‹œ ๋ฐฐ์น˜ ๋ชจ๋‹ˆํ„ฐ๋ง ๊ด€๋ฆฌ ๋Œ€์ƒ๋‚˜ ์˜ค๋ฅ˜ ๋ชจ๋‹ˆํ„ฐ๋ง ๋กœ์ง์„ ๋ณ„๋„๋กœ ์ถ”๊ฐ€ํ•˜์ง€ ์•Š์œผ๋ฉด ์•Œ๊ธฐ ์–ด๋ ค์›€

  • ์šด์˜๋ฐฐ์น˜๋ฅผ ์˜ˆ๋กœ ๋“ค๋ฉด, crontab์— ์Šค์ผ€์ฅด๋ง ๊ด€๋ฆฌ๊ฐ€ ์ œ๋Œ€๋กœ ๋˜์ง€ ์•Š์Œ(์ฃผ์„ ์ฒ˜๋ฆฌ๋˜์–ด์žˆ๋Š” job๋“ค ๋‹ค์ˆ˜)

  • crontab์˜ ๊ฒฝ์šฐ ๊ฐ ์„œ๋ฒ„๋งˆ๋‹ค ๋”ฐ๋กœ ์Šค์ผ€์ค„๋ง์„ ๊ด€๋ฆฌํ•ด์•ผ ํ•˜๋ฉฐ ๋ฌด์—‡๋ณด๋‹ค ํด๋Ÿฌ์Šคํ„ฐ๋ง ๊ธฐ๋Šฅ์ด ์ œ๊ณต๋˜์ง€ ์•Š์•„ ์ถ”์ฒœํ•˜์ง€ ์•Š๋Š”๋‹ค.

Quartz

Quartz๋Š” ์˜คํ”ˆ์†Œ์Šค Job Scheduling ๋ผ์ด๋ธŒ๋Ÿฌ๋ฆฌ๋กœ, ์™„์ „ํžˆ ์ž๋ฐ”๋กœ ๊ฐœ๋ฐœ๋˜์–ด ์ž๋ฐ” ํ™˜๊ฒฝ, ๊ทœ๋ชจ์™€ ์ƒ๊ด€์—†์ด ์‚ฌ์šฉํ•  ์ˆ˜ ์žˆ๋‹ค. Quartz๋Š” ์ˆ˜์ฒœ ๊ฐœ์˜ ์ž‘์—…๋„ ์‹คํ–‰ ๊ฐ€๋Šฅํ•˜๋ฉฐ, ๊ฐ„๋‹จํ•œ interval ํ˜•์‹์ด๋‚˜ Cron ํ‘œํ˜„์‹์œผ๋กœ ๋ณต์žกํ•œ ์Šค์ผ€์ฅด๋ง๋„ ์ง€์›ํ•œ๋‹ค.

์žฅ์ 

  • DB ๊ธฐ๋ฐ˜์œผ๋กœ ์Šค์ผ€์ค„๋Ÿฌ ๊ฐ„์˜ Clustering ๊ธฐ๋Šฅ์„ ์ œ๊ณต

  • ์‹œ์Šคํ…œ Fail-over์™€ Random ๋ฐฉ์‹์˜ ๋กœ๋“œ ๋ถ„์‚ฐ์ฒ˜๋ฆฌ๋ฅผ ์ง€์›ํ•œ๋‹ค

  • In-memory Job Scheduler๋„ ์ œ๊ณต

  • ์—ฌ๋Ÿฌ ๊ธฐ๋ณธ Plug-in์„ ์ œ๊ณต

  • ShutdownHookPlugin - JVM ์ข…๋ฃŒ ์ด๋ฒคํŠธ๋ฅผ ์บ์น˜ํ•ด์„œ ์Šค์ผ€์ค„๋Ÿฌ์—๊ฒŒ ์ข…๋ฃŒ๋ฅผ ์•Œ๋ ค์คŒ

    • LoggingJobHistoryPlugin - Job ์‹คํ–‰์— ๋Œ€ํ•œ ๋กœ๊ทธ๋ฅผ ๋‚จ๊ฒจ ๋””๋ฒ„๊น…ํ•  ๋•Œ ์‚ฌ์šฉ

  • ๋ฐฐ์น˜ ์ˆ˜ํ–‰์ค‘ ์ด์Šˆ๊ฐ€ ์ƒ๊ฒผ์„ ๋•Œ ์‰ฝ๊ฒŒ ํŒŒ์•…ํ•  ์ˆ˜ ์žˆ๋Š” ๊ตฌ์กฐ๋กœ ์„ค๊ณ„๋˜์–ด์žˆ์Œ(๋กœ๊ทธ ๊ด€๋ฆฌ/๋ฐฉ์–ด๋กœ์ง์— ์ข‹์Œ)

๋‹จ์ 

  • Clustering ๊ธฐ๋Šฅ์„ ์ œ๊ณตํ•˜์ง€๋งŒ, ๋‹จ์ˆœํ•œ random ๋ฐฉ์‹์ด๋ผ์„œ ์™„๋ฒฝํ•œ Cluster ๊ฐ„์˜ ๋กœ๋“œ ๋ถ„์‚ฐ์€ ์•ˆ๋จ.

  • ์–ด๋“œ๋ฏผ UI์„ ์ œ๊ณตํ•˜์ง€ ์•Š์Œ.

  • ์Šค์ผ€์ค„๋ง ์‹คํ–‰์— ๋Œ€ํ•œ History๋Š” ๋ณด๊ด€ํ•˜์ง€ ์•Š์Œ.

  • Fixed Delay ํƒ€์ž…์„ ๋ณด์žฅํ•˜์ง€ ์•Š์œผ๋ฏ€๋กœ ์ถ”๊ฐ€ ์ž‘์—…์ด ํ•„์š”

  • ์Šค์ผ€์ฅด๋ง์„ ๋ณ€๊ฒฝํ•˜๋ ค๋ฉด ์จ๋“œํŒŒํ‹ฐ๋ฅผ ์‚ฌ์šฉํ•˜๊ฑฐ๋‚˜, ๋ฐฐํฌ๊ฐ€ ํ•„์š”ํ•จ

  • ๋ชจ๋“  ๊ฒƒ์„ ์ปค์Šคํ…€ํ•ด์•ผํ•ด์„œ ์‹œ๊ฐ„ ์†Œ์š”๊ฐ€ ๋งŽ์ด๋“ ๋‹ค.

Jenkins

  • groovy ๊ธฐ๋ฐ˜์œผ๋กœ ๊ตฌ์„ฑ๋˜์–ด์žˆ์Œ.

  • ๋ฌด๋ฃŒ CIํˆด๋กœ, ๋ฐฐ์น˜ ์Šค์ผ€์ฅด๋ง์œผ๋กœ ๋งŽ์ด ์‚ฌ์šฉ

  • ํŒŒ์ดํ”„๋ผ์ธ ์„ค์ •๋„ ๊ฐ€๋Šฅ

  • ๋ฐฐ์น˜ ๋นŒ๋“œ ์‹คํŒจ์‹œ ์Šฌ๋ž™ ๋“ฑ์œผ๋กœ ๋ฐ”๋กœ ์•Œ๋ฆผ ์ƒ์„ฑ ๊ฐ€๋Šฅ

  • ํ•ด๋‹น ๋ฐฐ์น˜ ๋กœ๊ทธ๋„ ํ™•์ธ ๊ฐ€๋Šฅ

  • ์ˆ˜์ • ํžˆ์Šคํ† ๋ฆฌ ๊ด€๋ฆฌ ๊ฐ€๋Šฅ

  • jenkins ์ž์ฒด์—์„œ ๋ฐฐ์น˜ ์ˆ˜ํ–‰ on/off ๊ฐ€๋Šฅ

  • ํด๋Ÿฌ์Šคํ„ฐ๋ง ๊ฐ€๋Šฅ

  • ํŒŒ๋ผ๋ฏธํ„ฐํ™”, ์Šค์ผ€์ฅด๋ง, ssh ํŠธ๋ฆฌ๊ฑฐ ๋“ฑ ์‰ฝ๊ฒŒ ๊ฐ€๋Šฅ

  • ๋ชจ๋“  ์ด๋ ฅ, ์„ค์ •์ •๋ณด๋“ค์ด ์ „๋ถ€ ํŒŒ์ผ๋กœ ๊ด€๋ฆฌ๋จ

    • ์„ค์ • ์ •๋ณด/์‹คํ–‰ ์ด๋ ฅ/ํ˜„์žฌ Job ์ •๋ณด๋“ฑ์ด ๊ถ๊ธˆํ•˜๋ฉด Jenkins ๊ฐ€ ์ œ๊ณตํ•˜๋Š” API ํ˜น์€ ์„œ๋ฒ„๋‚ด์— ์กด์žฌํ•˜๋Š” XMLํŒŒ์ผ๋กœ๋งŒ ํ™•์ธํ•  ์ˆ˜ ์žˆ์Œ

  • ๋ฐฑ์—…&์ด์ค‘ํ™”๊ฐ€ ์–ด๋ ค์›€

  • ์‹ ๋ขฐํ•  ์ˆ˜ ์—†๋Š” ํ”Œ๋Ÿฌ๊ทธ์ธ

    • ๋Œ€๋ถ€๋ถ„์˜ ํ”Œ๋Ÿฌ๊ทธ์ธ์— ๋Œ€ํ•ด Jenkins๊ฐ€ ํ™•์‹คํ•˜๊ฒŒ ๋ณด์žฅํ•˜์ง€ ์•Š์Œ.

    • ์  ํ‚จ์Šค ๋ฒ„์ „์—…์‹œ ๋Œ€๋ถ€๋ถ„ ์‚ฌ์šฉ ๋ถˆ๊ฐ€๋Šฅ

Airflow

  • ์—์–ด๋น„์•ค๋น„์—์„œ ๊ฐœ๋ฐœํ•œ ์›Œํฌํ”Œ๋กœ์šฐ ์Šค์ผ€์ค„๋ง, ๋ชจ๋‹ˆํ„ฐ๋ง ํ”Œ๋žซํผ

    • ๋น…๋ฐ์ดํ„ฐ๋Š” ์ˆ˜์ง‘, ์ •์ œ, ์ ์ œ, ๋ถ„์„ ๊ณผ์ •์„ ๊ฑฐ์น˜๋ฉด์„œ ์—ฌ๋Ÿฌ๊ฐ€์ง€ ๋‹จ๊ณ„๋ฅผ ๊ฑฐ์น˜๊ฒŒ ๋˜๋Š”๋ฐ ์ด ์ž‘์—…๋“ค์„ ๊ด€๋ฆฌํ•˜๊ธฐ ์œ„ํ•œ ๋„๊ตฌ

  • DAG(Directed Acyclic Graph) ๊ฐœ๋…์˜ workflow ๋‹จ์œ„๋กœ ์‹คํ–‰

  • ํŒŒ์ด์ฌ ์ฝ”๋“œ๋กœ ์ž‘์„ฑ

    • ๋™์ผํ•œ task ์ˆ˜ํ–‰์‹œ์—๋„ for๋ฌธ๊ฐ€ if๋ฌธ์œผ๋กœ ํŒŒ์ดํ”„๋ผ์ธ ์žก ์‹คํ–‰ ๊ฐ€๋Šฅ

  • ํŒŒ๋ผ๋ฏธํ„ฐํ™”๊ฐ€ ๋˜์–ด์žˆ์ง€์•Š์•„, ์‹คํ–‰์‹œ๋งˆ๋‹ค dag.py๋ฅผ ๊ณ„์† ์ˆ˜์ •

    ํ•ด์ค˜์•ผํ•œ๋‹ค.

    • ์—์–ดํ”Œ๋กœ์šฐ ํŒŒ์ดํ”„๋ผ์ธ์€ ๊ฐ„๊ฒฐํ•˜๊ณ  ๋ช…์‹œ์ ์ด๋ฉฐ, **์ง„์ž ํ…œํ”Œ๋ฆฟ(jinja template)**์„ ์ด์šฉํ•˜์—ฌ ํŒŒ๋ผ๋ฏธํ„ฐํ™” ๋œ ๋ฐ์ดํ„ฐ๋ฅผ ์ „๋‹ฌํ•˜๊ณ  ์ž๋™์œผ๋กœ ํŒŒ์ดํ”„๋ผ์ธ์„ ์ƒ์„ฑํ•˜๋Š” ๊ฒƒ์ด ๊ฐ€๋Šฅ

  •   operator

    ๋ฅผ ์ด์šฉํ•ด ๋ณต์žกํ•œ workflow ๊ตฌ์„ฑ์ด ์‰ฌ์›€

    • t1 >> [t2,t3] >> t4 >> t1

  • dag๊ฐ„ ์—ฐ๊ฒฐ์„ ์œ„ํ•ด

    externalTaskSensor

    ์‚ฌ์šฉ

  • Task ๋ณ‘๋ ฌ ์ˆ˜ํ–‰์„ ์œ„ํ•ด์„œ๋Š”

    celery executor

    ์„ ์‚ฌ์šฉ ํ•„์š”

    • celery executor ๋ฅผ ์‚ฌ์šฉํ•˜๊ธฐ ์œ„ํ•ด์„œ๋Š” RabbitMQ๋‚˜ Redis๊ฐ€ ํ•„์š”

    • meta store๋กœ mysql or postgresql ์‚ฌ์šฉํ•„์š”.

  • ํ˜น์€

    Kubernetes Executor

    ๋„ ์‚ฌ์šฉ ๊ฐ€๋Šฅ

    • Task๋ฅผ ์Šค์ผ€์ค„๋Ÿฌ๊ฐ€ ์‹คํ–‰๊ฐ€๋Šฅ ์ƒํƒœ๋กœ ๋ณ€๊ฒฝํ•˜๋ฉด ๋ฉ”์‹œ์ง€ ๋ธŒ๋กœ์ปค์— ์ „๋‹ฌํ•˜๋Š”๊ฒŒ ์•„๋‹ˆ๋ผ Kubernetes API๋ฅผ ์‚ฌ์šฉํ•˜์—ฌ Airflow ์›Œ์ปค๋ฅผ pod ํ˜•ํƒœ๋กœ ์‹คํ–‰

    • ๋งค Task๋งˆ๋‹ค pod๊ฐ€ ์ƒ์„ฑ๋˜๋ฏ€๋กœ ๊ฐ€๋ณ๊ณ , Worker์— ๋Œ€ํ•œ ์œ ์ง€ ๋ณด์ˆ˜๊ฐ€ ํ•„์š”์—†๋‹ค๋Š” ์žฅ์ 

    • Kubernetes๋ฅผ ํ™œ์šฉํ•˜์—ฌ ์ง€์†์ ์œผ๋กœ ์ž์›์„ ์ ์œ ํ•˜์ง€ ์•Š๊ธฐ ๋•Œ๋ฌธ์— ํšจ์œจ์ ์œผ๋กœ ์ž์›์„ ์‚ฌ์šฉํ•  ์ˆ˜ ์žˆ์Œ.

    • ์งง์€ Task์—๋„ pod์„ ์ƒ์„ฑํ•˜๋Š” overhead๊ฐ€ ์žˆ์œผ๋ฉฐ, celery executor์— ๋น„ํ•ด ์ž๋ฃŒ๊ฐ€ ์ ๊ณ  ๊ตฌ์„ฑ์ด ๋ณต์žกํ•˜๋‹ค๋Š” ๋‹จ์ 

    • ๋ณ„๋„ ์„œ๋ฒ„ 1๊ฐœ์—์„œ ์šด์˜ํ• ๊ฒƒ์ด๋ฏ€๋กœ ๋งž์ง€ ์•Š์•„ ๋ณด์ž„

https://airflow.apache.org/_images/airflow.gif

Teamcity

  • JetBrains์˜ CI ๋„๊ตฌ

  • ํŒŒ์ดํ”„๋ผ์ธ / ์Šค์ผ€์ค„๋ง ๋“ฑ Jenkins๊ฐ€ ์ง€์›ํ•˜๋Š” ๋Œ€๋ถ€๋ถ„์˜ ๊ธฐ๋Šฅ์„ ๋™์ผํ•˜๊ฒŒ ์ง€์›

  • Jetbrains ์ œํ’ˆ๊ตฐ (IntelliJ, DataGrip, Upsource ๋“ฑ) ๊ณผ ํ†ตํ•ฉ ์ง€์›์ด ์ข‹์Œ

    • Job ์•Œ๋žŒ์— ๋Œ€ํ•ด IntelliJ์—์„œ ์•Œ๋žŒ์ด ๋ณด์—ฌ์ง€๋Š” ๋“ฑ

  • ์„ค์ • ์ •๋ณด๋“ค์ด DB๋กœ ๊ด€๋ฆฌ

    • ๋ฐฑ์—…/์ด์ค‘ํ™”๋Š” DB์— ๋ชจ๋‘๋จ

    • Teamcity๋Š” ์–ด๋””๋“  ์„ค์น˜๋งŒํ•ด์„œ ๋ฐ”๋กœ DB ์—ฐ๊ฒฐ๋งŒ ํ•˜๋ฉด ๋˜‘๊ฐ™์€ Teamcity ํ™˜๊ฒฝ์ด ๊ตฌ์„ฑ

    • ์—ฌ๋Ÿฌ๋Œ€์˜ ์„œ๋ฒ„๋ฅผ ์šด์˜ํ•œ๋‹ค๊ณ  ํ•ด๋„ ์Šค์ผ€์ค„๋ง/์„ค์ • ๋“ฑ์— ๋Œ€ํ•œ ๊ด€๋ฆฌ ์š”์†Œ๊ฐ€ ์ „ํ˜€ ์—†์Œ.

    • DB์—์„œ ๋‹ค ๊ด€๋ฆฌ์ค‘์ด๋‹ˆ, ๊ฐ TeamCity ์„œ๋ฒ„์— ๋Œ€ํ•œ ๋™๊ธฐํ™” ๊ฑฑ์ •์ด ํ•„์š” ์—†์Œ.

    • ๋‹น์—ฐํžˆ ๋ณ„๋„์˜ ๊ธฐ๋Šฅ์ด ํ•„์š”ํ•œ ๊ฒฝ์šฐ API๋ฅผ ์‚ฌ์šฉํ•ด๋„ ๋˜๊ณ , ์ง์ ‘ Teamcity ์„ค์ • ์ •๋ณด๋ฅผ ๋‹ด๊ณ  ์žˆ๋Š” DB์— Query๋ฅผ ๋‚ ๋ ค์„œ ์‚ฌ์šฉ ๊ฐ€๋Šฅ

  • ์ผ์ • ๊ทœ๋ชจ์ด์ƒ์—์„œ๋Š” ์œ ๋ฃŒ ํ”Œ๋žœ์ด ํ•„์š”

  • ๊ฒฐ๊ตญ์€ CI/CD ๋„๊ตฌ์ด๋‹ค๋ณด๋‹ˆ ๋ฐฐ์น˜์ชฝ์œผ๋กœ ๋ฐœ์ „ ๋ฐฉํ–ฅ์ด ํ–ฅํ•˜๊ณ  ์žˆ์ง„ ์•Š์Œ.

  • ํ”Œ๋Ÿฌ๊ทธ์ธ ์ƒํƒœ๊ณ„๊ฐ€ Jenkins์— ๋น„ํ•ด ์•ฝํ•จ.

    • ์ „์ฒด ํ”Œ๋Ÿฌ๊ทธ์ธ ์ˆ˜๊ฐ€ 10๋ฐฐ ์ด์ƒ ์ฐจ์ด๋‚จ.

    • ๊ทธ๋Ÿผ์—๋„ Github ๋กœ๊ทธ์ธ / ์Šฌ๋ž™ ์—ฐ๋™ ๋“ฑ ๋Œ€๋ถ€๋ถ„์˜ ํ”Œ๋Ÿฌ๊ทธ์ธ๋“ค์€ ์กด์žฌ

Spring Cloud Data Flow

  • Spring์—์„œ ๊ณต์‹์ ์œผ๋กœ ๋ฐ€๊ณ  ์žˆ๋Š” Batch/Data Stream ๋งค๋‹ˆ์ €

  • Spring์—์„œ ๋Œ€๋†“๊ณ  Batch/Data Stream ๋งค๋‹ˆ์ €๋กœ ๋‚˜์˜จ ๋„๊ตฌ๋ผ์„œ ๋ฐœ์ „ ๋ฐฉํ–ฅ์ด ๊ทธ์ชฝ์œผ๋กœ ๋ช…ํ™•

  • Teamcity์™€ ๋งˆ์ฐฌ๊ฐ€์ง€๋กœ DB์— ์—ฌ๋Ÿฌ ์„ค์ • ์ •๋ณด๋“ค์„ ๊ด€๋ฆฌ

    • ๋ณ„๋„๋กœ ์ง€์ •ํ•˜์ง€ ์•Š์œผ๋ฉด ์ธ๋ฉ”๋ชจ๋ฆฌ DB (H2)๋ฅผ ์‚ฌ์šฉ

  • ์˜คํ”ˆ์†Œ์Šค

  • CloudFoundry ํ˜น์€ Kubernates ํ™˜๊ฒฝ์ด ์•„๋‹ˆ๋ฉด ์ œ๋Œ€๋กœ ํ™œ์šฉํ•˜๊ธฐ๊ฐ€ ์–ด๋ ค์›€.

    • ์œ„ ํ™˜๊ฒฝ์—์„œ๋งŒ ์Šค์ผ€์ค„๋ง ๊ธฐ๋Šฅ์„ ์‚ฌ์šฉ ํ•  ์ˆ˜ ์žˆ์Œ.

    • ์ฆ‰, ๋‹จ์ผ ์„œ๋ฒ„์—์„œ๋Š” ์Šค์ผ€์ค„๋ง ๊ธฐ๋Šฅ์„ ์‚ฌ์šฉ ๋ชปํ•จ.

  • ์ปจ์…‰ ์ž์ฒด๊ฐ€ ๋ฐฐ์น˜๊ฐ€ ์‹คํ–‰๋ ๋•Œ๋งŒ ์ปจํ…Œ์ด๋„ˆ๋ฅผ ๋ณ„๋„๋กœ ์ƒ์„ฑํ•ด์„œ ์‹คํ–‰ํ•˜๊ณ  ์ข…๋ฃŒํ•˜๊ธฐ ์œ„ํ•จ์ด๋ผ ์ปจํ…Œ์ด๋„ˆ ์˜ค์ผ€์ŠคํŠธ๋ ˆ์ด์…˜ ์—†์ด ์‚ฌ์šฉํ•˜๋ ค๋ฉด ๊ต‰์žฅํžˆ ์ œํ•œ์ 

    • ๋‹น์—ฐํžˆ ์œ„ ๋‹จ์ ์œผ๋กœ ์ธํ•ด ํ—ˆ๋“ค์ด ๋‹ค๋ฅธ ์–ด๋–ค ๋„๊ตฌ๋“ค ๋ณด๋‹ค ๋†’์Œ.

    • Cloud Native Batch Application ์„ ์œ„ํ•ด ๋‚˜์˜จ ์„œ๋น„์Šค๋ผ ํด๋ผ์šฐ๋“œ๋ฅผ ๊ต‰์žฅํžˆ ๋‹จ์ˆœํ•˜๊ฒŒ๋งŒ ์‚ฌ์šฉํ•˜๋Š” ๊ทธ๋ฃน์—์„œ ์‚ฌ์šฉํ•˜๊ธฐ์—” ์ดˆ๊ธฐ ํ—ˆ๋“ค์ด ๋†’์Œ.

    • Spring Batch์— ๋Œ€ํ•œ ๊ณต๋ถ€๋ณด๋‹ค Kubernates ์™€ Docker ๊ณต๋ถ€๊ฐ€ ์šฐ์„  ๋˜์–ด์•ผํ•  ์ˆ˜๋„ ์žˆ์Œ.

  • ๋‹จ์ผ ์„œ๋ฒ„์— Kubernates ์„ค์น˜ํ•˜๊ณ  ๊ทธ ์„œ๋ฒ„์•ˆ์—์„œ Docker ์ƒ์„ฑ&์‚ญ์ œ๋ฅผ ํ•˜๋„๋ก ํ•˜๋Š”๊ฑด ๊ฑฐ์˜ ๋ฌด์˜๋ฏธ(๋‹จ์ผ์„œ๋ฒ„์—์„œ ์‚ฌ์šฉํ•˜๋Š”๊ฒƒ ๊ถŒ์žฅํ•˜์ง€ ์•Š์Œ.)

์ฐธ๊ณ 

Last updated

Was this helpful?