/images/profile.png

OpenAPI ์™€ Swagger-ui ์ ์šฉํ•˜๊ธฐ

ใ€€๏ปฟAPI๋ฅผ ๊ฐœ๋ฐœํ•˜๊ณ  ์‚ฌ์šฉ๋ฐฉ๋ฒ•์— ๋Œ€ํ•œ ๋ช…์„ธ๋ฅผ ์ž‘์„ฑํ•˜๋Š” ๋ฐฉ๋ฒ•์€ ์—ฌ๋Ÿฌ ๊ฐ€์ง€๊ฐ€ ์žˆ๋‹ค. ๊ฐ€์žฅ ์‹ฌํ”Œํ•˜๊ฒŒ ๊ฐœ๋ฐœ ์ฝ”๋“œ์™€๋Š” ๋ณ„๋„๋กœ ์ง์ ‘ ์ˆ˜๊ธฐ๋กœ ์ž‘์„ฑํ•˜์—ฌ ํŒŒ์ผ ํ˜น์€ ๋ฌธ์„œ ๋งํฌ๋ฅผ ์ „๋‹ฌํ•˜๋Š” ๋ฐฉ๋ฒ•์ด ์žˆ๋‹ค. ํ•˜์ง€๋งŒ ๊ฐœ๋ฐœ ์ฝ”๋“œ์™€ ๋ณ„๋„๋กœ ์ง์ ‘ ์ž‘์„ฑ์„ ํ•œ๋‹ค๋Š” ์ ์—์„œ ์˜คํƒ€/์‹ค์ˆ˜๊ฐ€ ๋ฐœ์ƒํ•  ์ˆ˜ ์žˆ๊ณ  ์ตœ์‹ ํ™”๊ฐ€ ์•ˆ๋˜๋Š” ์—ฌ๋Ÿฌ ๊ฐ€์ง€ ๋ฌธ์ œ๊ฐ€ ๋ฐœ์ƒํ•œ๋‹ค. ๊ทธ์— ๋“ฑ์žฅํ•œ API ๋ฌธ์„œํ™” ์ž๋™ํ™” ํˆด์˜ ์–‘๋Œ€ ์‚ฐ๋งฅ์ธ SpringRestDocs ์™€ Swagger.

ใ€€๏ปฟ๊ณผ๊ฑฐ SpringRestDocs ์— ๋Œ€ํ•œ ํฌ์ŠคํŒ…์„ ํ–ˆ๊ธฐ์— ์ด๋ฒˆ์—” Swagger์— ๋Œ€ํ•œ ์‚ฌ์šฉ๋ฐฉ๋ฒ•์— ๋Œ€ํ•ด ์ •๋ฆฌํ•ด๋ณด๊ณ ์ž ํ•œ๋‹ค. ์ด ๋‘˜์˜ ์žฅ๋‹จ์ ์€ ๋„ˆ๋ฌด ๋šœ๋ ทํ•˜๊ธฐ์— API๋ฌธ์„œ๋ฅผ ์ œ๊ณตํ•˜๋Š” ์ƒํ™ฉ์— ๋”ฐ๋ผ ์ ์ ˆํ•˜๊ฒŒ ์„ ํƒํ•˜์—ฌ ์‚ฌ์šฉํ•  ์ˆ˜ ์žˆ์—ˆ์œผ๋ฉด ์ข‹๊ฒ ๋‹ค.

๏ปฟSpringBoot์— Swagger ์ ์šฉ

ใ€€๊ธฐ๋ณธ SpringBoot ๊ฐ€ ์…‹ํŒ…๋˜์–ด ์žˆ๋‹ค๋Š” ๊ฐ€์ •ํ•˜์— Swagger ๊ด€๋ จ dependency๋ฅผ ์ถ”๊ฐ€ํ•ด์ฃผ์ž. ์•„์ฐธ, ์ด์ œ๋ถ€ํ„ฐ์˜ ํ”„๋กœ์ ํŠธ ์…‹ํŒ…์€ Gradle๋กœ ํ•˜๋ คํ•œ๋‹ค. (๋ฌผ๋ก  Maven์œผ๋กœ ํ•ด๋„ ๋ฌด๋ฐฉํ•˜์ง€๋งŒ…)

dependencies {
    implementation "io.springfox:springfox-boot-starter:3.0.0"
}

ใ€€๏ปฟ์ดํ›„ JavaConfig ์„ ์•„๋ž˜์™€ ๊ฐ™์ด ์„ค์ •ํ•˜๋Š”๋ฐ ์•„๋ž˜ ๋‚ด์šฉ์€ ์•„์ฃผ ๊ธฐ๋ณธ ์„ธํŒ…์ด๋‹ˆ ์ž์„ธํ•œ ๋‚ด์šฉ์€ ๊ณต์‹ ๋„ํ๋ฌธ์„œ๋ฅผ ์ฐธ๊ณ ํ•ด ๋ณด๋ฉด ์ข‹์„ ๊ฒƒ ๊ฐ™๋‹ค. (๋ฌผ๋ก  ์ƒ˜ํ”Œ ํ”„๋กœ์ ํŠธ๋ฅผ ๋งŒ๋“ค๋ฉฐ ํ•„์š”ํ•  ๊ฒƒ ๊ฐ™์€ ๋‚ด์šฉ์€ ์•„๋ž˜์—์„œ ์„ค๋ช…ํ•˜๊ฒ ๋‹ค.)๏ปฟ

@EnableSwagger2
@Configuration
public class SwaggerConfig {

	@Bean
	public Docket api() {
		return new Docket(DocumentationType.SWAGGER_2)
			.select()
			.apis(RequestHandlerSelectors.any())
			.paths(PathSelectors.any())
			.build();
	}
}

ใ€€๏ปฟํ…Œ์ŠคํŠธํ•  ์ปจํŠธ๋กค๋Ÿฌ๋ฅผ ์•„๋ž˜์ฒ˜๋Ÿผ ์‹ฌํ”Œํ•˜๊ฒŒ ์ž‘์„ฑํ•˜๊ณ (์‚ฌ์น™์—ฐ์‚ฐ…) ์‹คํ–‰์„ ์‹œํ‚จ ํ›„ /swagger-ui/์— ์ ‘์†์„ ํ•ด๋ณด๋ฉด swagger ๊ด€๋ จ javaConfig ํ•˜๋‚˜๋งŒ ์ถ”๊ฐ€ํ–ˆ๋Š”๋ฐ ๋ฌธ์„œ๊ฐ€ ๋งŒ๋“ค์–ด์ง„ ๊ฒƒ์„ ํ™•์ธํ•  ์ˆ˜ ์žˆ๋‹ค. (http method๋Š” ํŽธ์˜์ƒ ๋‹ค์–‘ํ•˜๊ฒŒ ์ž‘์„ฑํ–ˆ์œผ๋‹ˆ ์™œ DELETE ์ธ๊ฐ€๋ผ๋Š” ์˜๋ฌธ์€ ์ ‘์–ด๋‘์ž.)

@RestController
public class SampleController {

	@GetMapping(value = "/addition")
	public Integer addition(Integer num1, Integer num2) {
		return num1 + num2;
	}

	@PostMapping(value = "/subtraction")
	public Integer subtraction(Integer num1, Integer num2) {
		return num1 - num2;
	}

	@PutMapping(value = "/multiplication")
	public Integer multiplication(Integer num1, Integer num2) {
		return num1 * num2;
	}

	@DeleteMapping(value = "/division")
	public Integer division(Integer num1, Integer num2) {
		return num1 / num2;
	}
}
/images/openapi-and-swagger-ui-in-spring-boot/default-config.png
๊ธฐ๋ณธ ์…‹ํŒ…๋งŒ ํ–ˆ๋Š”๋ฐ ์ด๋Ÿฐ ํ™”๋ฉด์ด ๋‚˜ํƒ€๋‚ฌ๋‹ค.

ใ€€๏ปฟ์œ„์—์„œ ํ–ˆ๋˜ ์„ค์ •๋“ค ์ค‘ ๋ช‡ ๊ฐ€์ง€๋งŒ ์ข€ ๋” ์ž์„ธํžˆ ์‚ดํŽด๋ณด์ž.

์„ค์ •์„ค๋ช…
Docket๏ปฟSpringfox ํ”„๋ ˆ์ž„ ์›Œํฌ์˜ ๊ธฐ๋ณธ ์ธํ„ฐํŽ˜์ด์Šค๊ฐ€ ๋  ๋นŒ๋”๋กœ ๊ตฌ์„ฑ์„ ์œ„ํ•œ ์—ฌ๋Ÿฌ ๊ฐ€์ง€ ๊ธฐ๋ณธ๊ฐ’๊ณผ ํŽธ๋ฆฌํ•œ ๋ฐฉ๋ฒ•์„ ์ œ๊ณตํ•˜๊ณ  ์žˆ๋‹ค. ์ดํ›„ select()๋กœ ApiSelectorBuilder๋ฅผ ๋ฐ˜ํ™˜๋ฐ›์•„ ์‚ฌ์šฉํ•  ์ˆ˜ ์žˆ๋„๋ก ํ•ด์ค€๋‹ค.
apis๏ปฟ์–ด๋–ค ์œ„์น˜์— ์žˆ๋Š” API๋“ค์„ ๊ฐ€์ ธ์˜ฌ ๊ฒƒ์ธ๊ฐ€์— ๋Œ€ํ•œ ์ •์˜. RequestHandlerSelectors.any()์ด๋ผ๊ณ  ํ–ˆ์œผ๋‹ˆ SpringBoot์—์„œ ๊ธฐ๋ณธ์œผ๋กœ ์ œ๊ณตํ•˜๋Š” basic-error-controller ๋„ API ๋ฌธ์„œ๋กœ ๋งŒ๋“ค์–ด์ง„ ๊ฒƒ์„ ํ™•์ธํ•  ์ˆ˜ ์žˆ๋‹ค. ํŠน์ • ํŒจํ‚ค์ง€๋งŒ ์ ์šฉํ•˜๊ธฐ ์œ„ํ•ด์„œ๋Š” RequestHandlerSelectors.basePackage("com.taetaetae.swagger.api") ์™€ ๊ฐ™์€ ํ˜•์‹์œผ๋กœ ์ง€์ •ํ•˜๋ฉด ํ•ด๋‹น ํŒจํ‚ค์ง€ ํ•˜์œ„์— ์žˆ๋Š” Controller๋ฅผ ๊ธฐ์ค€์œผ๋กœ ๋ฌธ์„œ๋ฅผ ๋งŒ๋“ค์–ด ์ค€๋‹ค๏ปฟ.
paths๏ปฟ์ด๋ฆ„์—์„œ๋„ ๋ˆˆ์น˜๋ฅผ ์ฑŒ ์ˆ˜ ์žˆ๋“ฏ์ด ํŠน์ • path๋งŒ ํ•„ํ„ฐ๋งํ•ด์„œ ๋ฌธ์„œ๋ฅผ ๋งŒ๋“ค์–ด ์ค€๋‹ค.
useDefaultResponseMessages๏ปฟ๊ธฐ๋ณธ http ์‘๋‹ต ์ฝ”๋“œ๋ฅผ ์‚ฌ์šฉํ•ด์•ผ ํ•˜๋Š”์ง€๋ฅผ ๋‚˜ํƒ€๋‚ด๋Š” ํ”Œ๋ž˜๊ทธ

๏ปฟ์ด์™ธ์—๋„ security ๋‚˜ ๊ณตํ†ต์œผ๋กœ ์‚ฌ์šฉ๋˜๋Š” ํŒŒ๋ผ๋ฏธํ„ฐ ๋“ฑ ๋‹ค์–‘ํ•œ ์˜ต์…˜์„ ์„ค์ •ํ•  ์ˆ˜ ์žˆ์œผ๋‹ˆ ๊ฐ€๋Šฅํ•˜๋ฉด ์ƒํ™ฉ์— ๋งž๊ฒŒ ์„ค์ •์„ ๋ณ€๊ฒฝํ•ด ๋ณด๋Š” ๊ฒƒ๋„ ์ข‹์„ ๊ฒƒ ๊ฐ™๋‹ค. ๋‹ค๋ฅธ ์„ค์ •๋“ค์„ ์ถ”๊ฐ€์‹œ์ผœ์„œ ์ข€ ๋” ์นœ์ ˆํ•˜๊ฒŒ ๋งŒ๋“ค์–ด ๋ณด๋ฉด ์•„๋ž˜์ฒ˜๋Ÿผ ๋งŒ๋“ค ์ˆ˜ ์žˆ๊ณ  ํ•ด๋‹น ์ฝ”๋“œ๋Š” Github์—์„œ ํ™•์ธ ๊ฐ€๋Šฅํ•˜๋‹ค.

/images/openapi-and-swagger-ui-in-spring-boot/change-config.png
API ๋ฌธ์„œํ™”๋Š” ์ตœ๋Œ€ํ•œ ์นœ์ ˆํ•˜๊ฒŒ!!

OpenAPI

ใ€€๏ปฟSwagger ๊ณต์‹ ํ™ˆํŽ˜์ด์ง€๋ฅผ ์ด๋ฆฌ์ €๋ฆฌ ๋‘˜๋Ÿฌ๋ณด๋ฉด OpenAPI๋ผ๋Š” ๋‚ด์šฉ์ด ๋งŽ์ด ๋‚˜์˜จ๋‹ค. ๊ทธ๋ ‡๋‹ค๋ฉด OpenAPI๋Š” ๋ฌด์—‡์ผ๊นŒ? ๋ฌธ์„œ์— ๋‚˜์™€์žˆ๋Š” ๋‚ด์šฉ์„ ์ง์—ญํ•ด๋ณด๋ฉด Swagger ์‚ฌ์–‘์œผ๋กœ ์•Œ๋ ค์ ธ ์žˆ์œผ๋ฉฐ RESTful ์›น ์„œ๋น„์Šค๋ฅผ ์„ค๋ช…, ์ƒ์„ฑ, ์†Œ๋น„ ๋ฐ ์‹œ๊ฐํ™”ํ•˜๊ธฐ ์œ„ํ•œ ๊ธฐ๊ณ„ ํŒ๋… ๊ฐ€๋Šฅ ์ธํ„ฐํŽ˜์ด์Šค ํŒŒ์ผ์— ๋Œ€ํ•œ ์‚ฌ์–‘์ด๋ผ๊ณ  ํ•œ๋‹ค. ์ฆ‰, API ์ž์ฒด๋ฅผ ์„ค๋ช…ํ•˜๋Š” ์ธํ„ฐํŽ˜์ด์Šค ์ŠคํŽ™์ด๋ผ๊ณ  ์ดํ•ด๋ฅผ ํ•ด๋ณผ ์ˆ˜ ์žˆ๋‹ค. ์œ„์—์„œ ๋งŒ๋“ค์–ด์กŒ๋˜ Swagger๋ฅผ ๋ณด๋ฉด http://localhost:8080/v2/api-docs?group=Test API ๋ผ๊ณ  ๋‚˜์™€์žˆ๋Š”๋ฐ ์ด๋ฅผ ํด๋ฆญํ•ด๋ณด๋ฉด ์•„๋ž˜์™€ ๊ฐ™์ด json ํ˜•ํƒœ๋กœ ๋ณด์ธ๋‹ค. ๋‹ค์‹œ ๋ณด๋ฉด ์šฐ๋ฆฌ๊ฐ€ ์ด์ œ๊นŒ์ง€ Swagger์œผ๋กœ ๋งŒ๋“  API ๋ฌธ์„œ๋ฅผ ์„ค๋ช…ํ•˜๋Š” ์ธํ„ฐํŽ˜์ด์Šค ์ŠคํŽ™์œผ๋กœ ์ดํ•ดํ•  ์ˆ˜ ์žˆ๋‹ค. (๊ฝค ๊ธธ์–ด์„œ ๋ง์…ˆ API๋ฅผ ์ œ์™ธํ•˜๊ณ  ์ง€์› ๋‹ค.) ๊ทธ๋ ‡๋‹ค๋ฉด ์ด JSON ํŒŒ์ผ์„ ์–ด๋–ป๊ฒŒ ํ™œ์šฉํ•  ์ˆ˜ ์žˆ์„๊นŒ?๏ปฟ

Jenkins Job์„ ๋ณ‘๋ ฌ๋กœ ์‹คํ–‰ํ•ด์„œ ์†๋„๋ฅผ ๊ฐœ์„ ํ•ด๋ณด์ž. (by. Pipeline)

๏ปฟใ€€๊ด€๋ฆฌํ•˜๋Š” URL์ด 200์‘๋‹ต์„ ์ฃผ๊ณ  ์žˆ๋Š”์ง€ ๋ชจ๋‹ˆํ„ฐ๋ง์„ ํ•œ๋‹ค๊ณ  ๊ฐ€์ •ํ•ด๋ณด์ž. ๋‹ค์–‘ํ•œ ๋ฐฉ๋ฒ•์ด ์ƒ๊ฐ๋‚˜๊ฒ ์ง€๋งŒ ๊ฐ€์žฅ ์ฒ˜์Œ์œผ๋กœ ๋– ์˜ค๋ฅธ ๊ฑด ๋‹จ์—ฐ Jenkins. ๊ฐ„๋‹จํ•˜๊ฒŒ ์‚ฌ์šฉํ•  ์–ธ์–ด์— ๋งž์ถฐ Execute Script๋ฅผ ์ž‘์„ฑํ•˜๊ณ  ์Šค์ผ€์ค„๋ง์„ ๊ฑธ์–ด ๋†“์œผ๋ฉด ํฐ ์ˆ˜๊ณ  ์—†์ด ๋ชจ๋‹ˆํ„ฐ๋ง์„ ๊ตฌ์„ฑํ•  ์ˆ˜ ์žˆ๊ฒŒ ๋œ๋‹ค. ์•„๋ž˜๋Š” python script๋กœ ์ž‘์„ฑํ•ด ๋ณด์•˜๋‹ค.

import requests
url="http://๋ชจ๋‹ˆํ„ฐ๋งurl"
status_code = requests.get(url).status_code
if status_code != 200:
	print(f'์‘๋‹ต ์‹คํŒจ :{url}, status : {status_code}')
	exit(1)

๏ปฟใ€€ํ•˜์ง€๋งŒ ๋ชจ๋‹ˆํ„ฐ๋ง์„ ํ•ด์•ผ ํ•˜๋Š” URL์ด 1๊ฐœ์—์„œ ์—ฌ๋Ÿฌ ๊ฐœ๋กœ ๋Š˜์–ด๋‚œ๋‹ค๋ฉด ์–ด๋–ป๊ฒŒ ๋ ๊นŒ? ๋‹จ์ˆœํ•˜๊ฒŒ ์ž‘์„ฑํ•œ Script๋ฅผ ์•„๋ž˜์ฒ˜๋Ÿผ ์•ฝ๊ฐ„ ์ˆ˜์ •ํ•˜๋ฉด ๋˜๊ธด ํ•˜์ง€๋งŒ URL๋งˆ๋‹ค ์‘๋‹ต์†๋„๊ฐ€ ๋‹ค๋ฅผ ๊ฒฝ์šฐ ์ˆœ์ฐจ์ ์œผ๋กœ ์‹คํ–‰ํ•˜๋‹ค ๋ณด๋‹ˆ ์‹คํ–‰ ์†๋„๋Š” ๋А๋ฆด ์ˆ˜๋ฐ–์— ์—†๋‹ค.

import requests
urls = [
	"http://๋ชจ๋‹ˆํ„ฐ๋งurl-1",
	"http://๋ชจ๋‹ˆํ„ฐ๋งurl-2",
	"http://๋ชจ๋‹ˆํ„ฐ๋งurl-3"
]

for url in urls:
	status_code = requests.get(url).status_code
	if status_code != 200:
		print(f'์‘๋‹ต ์‹คํŒจ :{url}, status : {status_code}')
		exit(1)

๏ปฟ ใ€€์ด๋Ÿฌํ•œ ๊ฒฝ์šฐ, ๋น ๋ฅธ ์†๋„๋ฅผ ๋ณด์žฅํ•˜๊ธฐ ์œ„ํ•ด์„œ๋Š” ๋ณ‘๋ ฌ๋กœ ์‹คํ–‰์„ ํ•ด์•ผ ํ•œ๋‹ค๋Š” ๊ฑด ๋ˆ„๊ตฌ๋‚˜ ๋‹ค ์•Œ์ง€๋งŒ ๊ทธ๋ ‡๋‹ค๊ณ  Thread๋ฅผ ์‚ฌ์šฉํ•˜๊ธฐ์—” ๋ฒŒ์จ๋ถ€ํ„ฐ ๋œ์ปฅ ๋ถ€๋‹ด์ด ๋œ๋‹ค. ๊ทธ๋ ‡๋‹ค๊ณ  Job์„ URL ๊ฐœ์ˆ˜๋งŒํผ ๋Š˜๋ฆฌ๊ธฐ์—๋Š” ๋ฐฐ๋ณด๋‹ค ๋ฐฐ๊ผฝ์ด ๋” ์ปค๋ฒ„๋ฆฌ๊ณ … ๊ทธ๋Ÿฌ๋‹ค ๋ฐœ๊ฒฌํ•œ ๊ธฐ๋Šฅ์ด ๋ฐ”๋กœ Jenkins Pipeline!

ใ€€์ด๋ฒˆ ํฌ์ŠคํŒ…์—์„œ๋Š” Jenkins Job์„ ๋™์‹œ์— ์—ฌ๋Ÿฌ ๋ฒˆ ์‚ฌ์šฉํ•ด์•ผ ํ•˜๋Š” ๊ฒฝ์šฐ๋ฅผ Pipeline์„ ํ†ตํ•ด์„œ ๊ฐœ์„ ํ•œ ๋‚ด์šฉ์— ๋Œ€ํ•˜์—ฌ ๊ณต์œ ํ•ด๋ณด๋ ค ํ•œ๋‹ค. Jenkins Pipeline์— ๋Œ€ํ•ด ๋“ค์–ด๋งŒ ๋ดค๋Š”๋ฐ ์ด๋ฒˆ์— ์‹ค์ œ๋กœ ์‚ฌ์šฉํ•ด๋ณด๋‹ˆ ์ƒ๊ฐ๋ณด๋‹ค ์‰ฝ๊ฒŒ ๊ฐœ์„ ํ•  ์ˆ˜ ์žˆ์—ˆ๊ณ  ์˜ต์…˜๋“ค์„ ์ƒํ™ฉ์— ๋งž๊ฒŒ ์กฐํ•ฉ์„ ์ž˜ ํ•œ๋‹ค๋ฉด ์ƒ๋‹นํžˆ ํ™œ์šฉ์„ฑ์ด ๋†’์•„ ๋ณด์ด๋Š” ๊ธฐ๋Šฅ์ธ ๊ฒƒ ๊ฐ™๋‹ค.

๊ธฐ์กด์ƒํ™ฉ

ใ€€ํ…Œ๏ปฟ์ŠคํŠธ๋ฅผ ์œ„ํ•ด ์ž„์˜๋กœ ๋А๋ฆฐ ์‘๋‹ต์„ ์ƒ์„ฑํ•˜๋„๋ก URL์„ ๊ตฌ์„ฑํ•˜๊ณ  ์œ„์—์„œ ์ด์•ผ๊ธฐํ–ˆ๋˜ ๊ฒƒ์ฒ˜๋Ÿผ Job ํ•˜๋‚˜์— ์•„์ฃผ ์‹ฌํ”Œํ•˜๊ฒŒ Python script๋ฅผ ์ž‘์„ฑํ•˜๊ณ  ์‹คํ–‰ํ•ด๋ณด๋„๋ก ํ•˜์ž. ์ž„์˜๋กœ ๋А๋ฆฐ ์‘๋‹ต์€ http://slowwly.robertomurray.co.uk/ ์—์„œ ์ œ๊ณตํ•˜๋Š” ๊ธฐ๋Šฅ์„ ํ™œ์šฉํ•˜์˜€๋‹ค.๏ปฟ

import requests
urls = [
	"http://slowwly.robertomurray.co.uk/delay/0/url/https://www.naver.com/",
	"http://slowwly.robertomurray.co.uk/delay/100/url/https://www.naver.com/",
    "http://slowwly.robertomurray.co.uk/delay/200/url/https://www.naver.com/",
    "http://slowwly.robertomurray.co.uk/delay/500/url/https://www.naver.com/",
    "http://slowwly.robertomurray.co.uk/delay/1000/url/https://www.naver.com/",
    "http://slowwly.robertomurray.co.uk/delay/2000/url/https://www.naver.com/",
    "http://slowwly.robertomurray.co.uk/delay/5000/url/https://www.naver.com/",
    "http://slowwly.robertomurray.co.uk/delay/10000/url/https://www.naver.com/",
    "http://slowwly.robertomurray.co.uk/delay/20000/url/https://www.naver.com/"
]

for url in urls:
	status_code = requests.get(url).status_code
	if status_code != 200:
		print(f'์‘๋‹ต ์‹คํŒจ :{url}, status : {status_code}')
		exit(1)
	print(f'์‘๋‹ต์„ฑ๊ณต : {url}')

๊ทธ๋ž˜์„œ ์‹คํ–‰ํ•ด๋ณด๋ฉด 50์ดˆ๊ฐ€ ์†Œ์š”๋˜์—ˆ๋‹ค. ์ž, ์ด์ œ ๊ฐœ์„ ์„ ํ•ด๋ณด์ž!

๊ฐœ์„ ์„ ํ•ด๋ณด์ž

ใ€€๏ปฟ์ „์ฒด์ ์ธ ๊ฐœ์„ ์˜ ํ๋ฆ„์€ ํ•˜๋‚˜์˜ Job์— ๋ชจ๋‹ˆํ„ฐ๋งํ•˜๊ณ ์ž ํ•˜๋Š” url์„ ํŒŒ๋ผ๋ฏธํ„ฐ๋กœ ๋ฐ›์•„์„œ ์ฒ˜๋ฆฌํ•  ์ˆ˜ ์žˆ๋„๋ก ์„ค์ •ํ•˜๊ณ , ์ด๋ฅผ Jenkins Pipeline ์„ ํ†ตํ•ด ์—ฌ๋Ÿฌ URL์„ ๋™์‹œ์— ๋ชจ๋‹ˆํ„ฐ๋งํ•˜๊ฒŒ ๊ตฌ์„ฑํ•˜๋Š” ๊ฒƒ์ด๋‹ค. ๊ทธ๋Ÿฌ๋ฉด ๋‘ ๊ฐœ์˜ Job(ํŒŒ๋ผ๋ฏธํ„ฐ๋กœ ๋ฐ›์•„ ๋ชจ๋‹ˆํ„ฐ๋งํ•˜๋Š” Job, Jenkins Pipeline Job) ๋งŒ์œผ๋กœ ๋ณด๋‹ค ๋น ๋ฅด๊ณ  ํšจ์œจ์ ์ธ ๊ตฌ์„ฑ์„ ํ•  ์ˆ˜ ์žˆ์„ ๊ฒƒ์œผ๋กœ ์ƒ์ƒ์„ ํ•˜๊ณ .

Job์„ ๋ฒ”์šฉ์ ์œผ๋กœ (Jenkins paramters ํ™œ์šฉ)

ใ€€๏ปฟ์œ„์—์„œ ์ƒ˜ํ”Œ๋กœ ์ž‘์„ฑํ•˜์˜€๋˜ Python script๋Š” url ์ด ๋Š˜์–ด๋‚ ์ˆ˜๋ก Job ์•ˆ์— script๋ฅผ ์ˆ˜์ •ํ•ด์•ผ ํ•œ๋‹ค. ๊ทธ๋ ‡๊ฒŒ ํ•ด๋„ ๋ฌด๋ฐฉํ•˜์ง€๋งŒ ์ด๋ฒˆ ๊ฐœ์„ ์˜ ๋ชฉํ‘œ๋Š” ํ•˜๋‚˜์˜ Job์„ Pipeline ์ด ๋ณ‘๋ ฌ๋กœ ์ปจํŠธ๋กคํ•˜๋„๋ก ์„ค์ •ํ•ด์•ผ ํ–ˆ๊ธฐ ๋•Œ๋ฌธ์— Jenkins Job์— ํŒŒ๋ผ๋ฏธํ„ฐ๋ฅผ ๋ฐ›์„ ์ˆ˜ ์žˆ๋„๋ก ์•„๋ž˜์ฒ˜๋Ÿผ Jenkins Job ์„ค์ •์— ํŒŒ๋ผ๋ฏธํ„ฐ๋ฅผ ์„ค์ •ํ•˜๊ณ  Python script ๋˜ํ•œ ์ˆ˜์ •ํ•ด ์ฃผ์ž.

/images/jenkins-job-parallel-processing-by-pipeline/jenkins-parameter-option.jpg
๏ปฟJob > ๊ตฌ์„ฑ > ์ด ๋นŒ๋“œ๋Š” ๋งค๊ฐœ๋ณ€์ˆ˜๊ฐ€ ์žˆ์Šต๋‹ˆ๋‹ค
import requests, os
url = os.environ['url']
status_code = requests.get(url).status_code

if status_code != 200:
  print(f'์‘๋‹ต ์‹คํŒจ :{url}, status : {status_code}')
  exit(1)
  
print(f'์‘๋‹ต์„ฑ๊ณต : {url}')

๋ณ‘๋ ฌ ์‹คํ–‰์„ ์œ„ํ•œ Jenkins ์„ค์ •

ใ€€๏ปฟJenkins Job ์„ ์ƒ์„ฑํ•˜๋ฉด ๊ธฐ๋ณธ์ ์œผ๋กœ Job๋งˆ๋‹ค์˜ ๋Œ€๊ธฐ์—ด(Queue)์ด ์žˆ์–ด Job์ด ์‹คํ–‰ ์ค‘์ด๋ผ๋ฉด ์‹œ์ž‘๋œ ์‹œ๊ฐ„ ์ˆœ์„œ๋Œ€๋กœ ๊ธฐ๋‹ค๋ ธ๋‹ค๊ฐ€ ์•ž์„  Job์ด ์ข…๋ฃŒ๊ฐ€ ๋˜๋ฉด ์ด์–ด์„œ ์‹คํ–‰๋˜๋Š” ๊ตฌ์กฐ์ด๋‹ค. ํ•˜์ง€๋งŒ ์šฐ๋ฆฌ๋Š” Job์„ ๋ณ‘๋ ฌ๋กœ ์‹คํ–‰ํ•ด์•ผ ํ–ˆ๊ธฐ์— Job ์„ค์ • ์ค‘ ํ•„์š”ํ•œ ๊ฒฝ์šฐ concurrent ๋นŒ๋“œ ์‹คํ–‰ ์˜ต์…˜์„ ์ผœ์ค˜์„œ ๊ธฐ๋‹ค๋ฆฌ์ง€ ์•Š๊ณ  ๋ณ‘๋ ฌ๋กœ ์‹คํ–‰๋  ์ˆ˜ ์žˆ๋„๋ก ํ•ด์ค€๋‹ค.

/images/jenkins-job-parallel-processing-by-pipeline/jenkins-concurrent-build-option.jpg
๏ปฟJob > ๊ตฌ์„ฑ > ํ•„์š”ํ•œ ๊ฒฝ์šฐ concurrent ๋นŒ๋“œ ์‹คํ–‰

ใ€€๏ปฟ๋˜ํ•œ Jenkins Job ์ž์ฒด๋Š” ๋ณ‘๋ ฌ๋กœ ์‹คํ–‰๋˜๋„๋ก ์„ค์ •๋˜์—ˆ๋‹ค ํ•ด๋„ ๊ธฐ๋ณธ์ ์œผ๋กœ Jenkins ์ž์ฒด์˜ ๋Œ€๊ธฐ์—ด์€ ํ•œ์ •๋˜์–ด ์žˆ๊ธฐ ๋•Œ๋ฌธ์— ์ ๋‹นํžˆ ๋Š˜๋ ค์ค˜์„œ ์—ฌ๋Ÿฌ ๊ฐœ์˜ Job์ด ๋Œ€๊ธฐ ์—ด ์—†์ด ๋™์‹œ์— ์‹คํ–‰๋  ์ˆ˜ ์žˆ๋„๋ก ํ•ด์ค€๋‹ค.

/images/jenkins-job-parallel-processing-by-pipeline/jenkins-executors.jpg
๏ปฟJenkins > Jenkins ๊ด€๋ฆฌ > ์‹œ์Šคํ…œ ์„ค์ • > of executors

Jenkins Pipeline

ใ€€๏ปฟJob ์„ Pipeline์œผ๋กœ ๋งŒ๋“ค๊ณ  Pipeline scirpt๋ฅผ ์ž‘์„ฑํ•˜๋Š”๋ฐ ๋ˆˆ์—ฌ๊ฒจ๋ด์•ผ ํ•  ์˜ต์…˜๋“ค์„ ์งš๊ณ  ๋„˜์–ด๊ฐ€๊ณ ์ž ํ•œ๋‹ค.

๊ธฐ์ˆ ๋ธ”๋กœ๊ทธ ๊ฐœํŽธ๊ธฐ (by HUGO)

ใ€€์›น์„œ๋น„์Šค ๊ฐœ๋ฐœ์ž๋ผ๋ฉด ๋‚˜๋งŒ์˜ ๋ธ”๋กœ๊ทธ์ฏค์€ ์žˆ์–ด์•ผ์ง€ ํ•˜๋ฉฐ ๊ธฐ์ˆ  ๋ธ”๋กœ๊ทธ๋ฅผ ์‹œ์ž‘ํ•œ ์ง€๋„ ์–ด๋А๋ง 4๋…„์ด ๋˜์—ˆ๋‹ค. ์ฒ˜์Œ์—” ๊ทธ์ € ์ƒˆ๋กœ ์•Œ๊ฒŒ ๋œ ๊ธฐ์ˆ ์ด๋‚˜ ์‚ฝ์งˆํ•˜๋ฉฐ ๊ฒฝํ—˜ํ•œ ๊ฒƒ๋“ค ์ค‘์— ํ•ต์‹ฌ๋งŒ์„ ์ ์–ด๋†“๋Š” ์ˆ˜์ค€์ด์—ˆ๋‹ค. (์ง€๊ธˆ ๋‹ค์‹œ ๋ณด๋ฉด ๋ญ”๊ฐ€ ์˜ค๊ธ€๊ฑฐ๋ฆฌ๋Š” ๊ฑด ๊ธฐ๋ถ„ ํƒ“์ด๊ฒ ์ง€…) ๊ทธ๋ ‡๊ฒŒ ๊ณ„์† ๊ธ€์„ ์จ์˜ค๋ฉด์„œ ๊ธ€์“ฐ๊ธฐ๋ผ๋Š” ๊ฒƒ์— ๊ด€์‹ฌ์„ ๊ฐ–๊ฒŒ ๋˜๊ณ  ๋‚ด ๊ธ€์ด ๋ˆ„๊ตฐ๊ฐ€์—๊ฒŒ ๋„์›€์ด ๋  ๊ฑฐ๋ผ๋Š” ๊ธฐ๋Œ€์— ์กฐ๊ธˆ์ด๋ผ๋„ ๊ธ€์„ ์ž˜ ์จ๋ณด๊ณ ์ž ๋‹จ์ˆœ ๊ธฐ๋ก ์šฉ์ด ์•„๋‹Œ ํ•˜๋‚˜์˜ ‘๊ธ€’์„ ์“ฐ๋ ค๊ณ  ๋…ธ๋ ฅํ•ด ์˜จ ๊ฒƒ ๊ฐ™๋‹ค.

ใ€€์ผ์ฃผ์ผ์— ํ•œ ๊ฐœ๋Š” ์จ์•ผ์ง€. ํ•œ ๋‹ฌ์— ํ•œ ๊ฐœ๋Š” ์จ์•ผ์ง€. ํ•˜๋ฉฐ ์ž๊พธ ๋‚˜ ์ž์‹ ๊ณผ์˜ ํƒ€ํ˜‘์„ ํ•˜๋‹ค๊ฐ€ ์ตœ๊ทผ์—๋Š” ํšŒ์‚ฌ์—์„œ ์šด์˜ํ•˜๋Š” ์„œ๋น„์Šค ๊ฐœํŽธ ๋•Œ๋ฌธ์— ์ •์‹ ์—†์ด ๋ฐ”์˜๋‹ค๋Š” ํ•‘๊ณ„๋กœ ‘๋ธ”๋กœ๊ทธ’์— ‘ใ…‚’์ž๋„ ์ƒ๊ฐํ•˜์ง€ ๋ชปํ•˜๊ฒŒ ๋œ๋‹ค. ๋ฌด์—‡์ด ๋ฌธ์ œ์ผ๊นŒ?๋ผ๋Š” ์ƒ๊ฐ์€ ๊ฒฐ๊ตญ ๋‚ด ๊ธฐ์ˆ  ๋ธ”๋กœ๊ทธ๋„ ํšŒ์‚ฌ ์„œ๋น„์Šค์ฒ˜๋Ÿผ ‘๊ฐœํŽธ’์„ ํ•ด๋ณด์ž๋Š” ์ƒ๊ฐ์œผ๋กœ ๋„๋‹ฌํ•˜๊ฒŒ ๋˜์—ˆ๊ณ  ๊ฐ„๋‹จํ•  ๊ฒƒ๋งŒ ๊ฐ™์•˜๋˜ ๊ธฐ์ˆ  ๋ธ”๋กœ๊ทธ ๊ฐœํŽธ ์ž‘์—…์€ ๊ฝค ์˜ค๋žซ๋™์•ˆ + ๋‹ค์–‘ํ•œ ์‚ฝ์งˆ๋“ค๋กœ ์ž‘์—…์„ ํ•˜๊ฒŒ ๋œ๋‹ค.

ใ€€์ด๋ฒˆ ํฌ์ŠคํŒ…์—์„œ๋Š” ๊ธฐ์ˆ  ๋ธ”๋กœ๊ทธ๋ฅผ ๊ฐœํŽธํ•˜๋ฉฐ ๊ฒช์—ˆ๋˜ ๋‚ด์šฉ๋“ค์— ๋Œ€ํ•ด ์ •๋ฆฌํ•ด๋ณด๊ณ ์ž ํ•œ๋‹ค. ๊ธฐ์กด์— ๊ธฐ์ˆ  ๋ธ”๋กœ๊ทธ๋ฅผ ์šด์˜ํ•˜์‹œ๋Š” ๋ถ„๋“ค์ด๋‚˜ ์ด๋ฒˆ์— ์ƒˆ๋กญ๊ฒŒ ์‹œ์ž‘ํ•˜์‹œ๋Š” ๋ถ„๋“ค๊ป˜ ๋„์›€์ด ๋  ๊ฑฐ๋ผ ๊ธฐ๋Œ€ํ•œ๋‹ค. ๋”๋ถˆ์–ด ์„œ๋น„์Šค ‘์ถœ์‹œ’ ๊ฐ€ ์•„๋‹Œ ๊ฐœํŽธ’์ด๋ผ๋Š” ๊ณผ์ • ์†์—์„œ ๋А๋ผ๊ฒŒ ๋˜์—ˆ๋˜ ์ธ์‚ฌ์ดํŠธ๋„ ๊ฐ„๋žตํ•˜๊ฒŒ ์ž‘์„ฑํ•ด๋ณผ๊นŒ ํ•œ๋‹ค.

๊ธฐ์ˆ ๋ธ”๋กœ๊ทธ ํ”Œ๋žซํผ ์„ ํƒ

ใ€€์ฒ˜์Œ ๋ธ”๋กœ๊ทธ๋ฅผ ์“ฐ๊ธฐ ์‹œ์ž‘ํ–ˆ์„ ๋•Œ ํฌํ„ธ์„œ๋น„์Šค์˜ ๊ธ€์“ฐ๊ธฐ ํ”Œ๋žซํผ์„ ์‚ฌ์šฉํ•˜์ง€ ์•Š์€ ์ด์œ ๋Š” ๋‹จ ํ•˜๋‚˜๋‹ค. ‘๊ธ€์“ฐ๊ธฐ’ ๋ฟ๋งŒ ์•„๋‹ˆ๋ผ ๊ฐœ๋ฐœ์ž์ด๊ธฐ์— ์›น์‚ฌ์ดํŠธ(๋ธ”๋กœ๊ทธ)๋ฅผ ๋‚ด ์ž…๋ง›์— ๋งž๊ฒŒ ์ปค์Šคํ„ฐ๋งˆ์ด์ง• ํ•˜๊ธฐ ์œ„ํ•ด์„œ. ๊ทธ ์ด์œ ๋กœ hexo ๋ผ๋Š” ํ”„๋ ˆ์ž„์›Œํฌ์— github์˜ ํ˜ธ์ŠคํŒ…์„ ์‚ฌ์šฉํ•˜์—ฌ ์šด์˜์„ ํ•ด์™”๋‹ค. ๊ทธ๋ ‡๊ฒŒ ๋ธ”๋กœ๊ทธ๋ฅผ ์šด์˜ํ•ด์˜ค๋ฉด์„œ ๋А๊ผˆ๋˜ ๋ถˆํŽธํ–ˆ๋˜ ๋ถ€๋ถ„๋“ค๊ณผ ๊ฐœํŽธ์„ ํ•˜๋ฉฐ ๊ธฐ๋Œ€ํ•˜๋Š” ๋ถ€๋ถ„๋“ค์„ ์ •๋ฆฌํ•˜๋ฉด ์•„๋ž˜์™€ ๊ฐ™๋‹ค.

  • ํ…Œ๋งˆ(UI)๊ฐ€ ์ด๋ป์•ผ ํ•˜๊ณ  ๊ธฐ๋Šฅ๋“ค์ด ๋งŽ์œผ๋ฉด ์ข‹๊ฒ ๋‹ค.
  • ๊ธฐ์ˆ  ๋ธ”๋กœ๊ทธ์ธ ๋งŒํผ ์ฝ”๋“œ๊ฐ€ ๋งŽ์ด ์‚ฝ์ž…๋˜๋‹ˆ ์ฝ”๋“œ ํ‘œํ˜„ ๋˜ํ•œ ์ด๋ป์•ผ ํ•œ๋‹ค.
  • ํ…Œ๋งˆ ๋˜๋Š” ํ”„๋ ˆ์ž„์›Œํฌ์˜ ์ปค๋ฎค๋‹ˆํ‹ฐ๊ฐ€ ํ™œ๋ฐœํ•ด์•ผ ํ•œ๋‹ค.
  • ํŽ˜์ด์ง€ ์ƒ์„ฑ ๋˜๋Š” ๋งŒ๋“ค์–ด์ง„ ์›นํŽ˜์ด์ง€์˜ ์„ฑ๋Šฅ์ด ์ข‹์•„์•ผ ํ•œ๋‹ค.
  • ๊ธ€์„ ์ž‘์„ฑํ•˜๊ณ  ๋ฐฐํฌํ•˜๋Š” ๊ณผ์ •์ด ์‹ฌํ”Œํ•˜๊ณ  ๊น”๋”ํ•ด์•ผ ํ•œ๋‹ค. ๏ปฟ

ใ€€์œ„์™€ ๊ฐ™์€ ์ด์œ ๋ฅผ ๊ธฐ๋ฐ˜์œผ๋กœ ๊ฒ€์ƒ‰์„ ํ•ด๋ณด๋‹ค SSG(์“ฑ ์‡ผํ•‘๋ชฐ ์•„๋‹˜, Static site generators)๋ฅผ ๊น”๋”ํ•˜๊ฒŒ ์ •๋ฆฌํ•ด ๋†“์€ ์‚ฌ์ดํŠธ๋ฅผ ๋ฐœ๊ฒฌํ•œ๋‹ค. ์ •๋ง ๋‹ค์–‘ํ•œ ํ”Œ๋žซํผ๋“ค์„ ์‚ดํŽด๋ณด๋ฉฐ ํ•„์ž์—๊ฒŒ ๋งž๋Š” ๊ฒŒ ์–ด๋–ค ๊ฑด์ง€ ๊ณ ๋ฏผํ•˜๋‹ค ๊ฒฐ๊ตญ hugo ๋ฅผ ์„ ํƒํ•˜๊ฒŒ ๋œ๋‹ค. hugo๋ฅผ ์„ ํƒํ•œ ์ด์œ ๋Š” go๋ผ๋Š” ์–ธ์–ด๋ฅผ ์‚ฌ์šฉํ•œ๋‹ค๋Š” ๊ฒƒ๊ณผ (๊ฐ„์ ‘์ ์œผ๋กœ๋ผ๋„ ๋‹ค๋ฅธ ์–ธ์–ด๋ฅผ ๊ฒฝํ—˜ํ•ด๋ณด๊ณ  ์‹ถ์–ด์„œ + go ์–ธ์–ด๊ฐ€ ๋น ๋ฅด๋‹ค๋Š” ์†Œ๋ฆฌ๋ฅผ ์–ด๋””์„ ๊ฐ€ ๋“ค์–ด์„œ) ํ…Œ๋งˆ๋“ค์ด ๋„ˆ๋ฌด ๋‹ค์–‘ํ–ˆ๊ธฐ ๋•Œ๋ฌธ์ด๋‹ค.

/images/blog-reorganization-by-hugo/hugo_homepage.jpg
๏ปฟ์•„์ฃผ ๋Œ€๋†“๊ณ  ๋น ๋ฅด๋‹ค๊ณ  ํ•˜๋‹ˆ… ์“ฐ๊ณ  ์‹ถ์–ด์ง„๋‹ค.

ใ€€๊ฒฐ๊ตญ hugo์— hugo-ranking-trend๋ผ๋Š” ์‚ฌ์ดํŠธ์—์„œ ์ƒ์œ„์— ๋žญํฌ๊ฐ€ ๋˜์–ด์žˆ๊ณ  ๊ธฐ์ˆ  ๋ธ”๋กœ๊ทธ ์„ฑ๊ฒฉ์— ์ ํ•ฉํ•  ๊ฒƒ ๊ฐ™์€ LoveIt์ด๋ผ๋Š” ํ…Œ๋งˆ๋ฅผ ์‚ฌ์šฉํ•˜๊ธฐ๋กœ ๊ฒฐ์ •ํ•˜์˜€๋‹ค. ์ž ๊ทธ๋Ÿผ ์‹œ์ž‘ํ•ด๋ณผ๊นŒ?!

hugo ๋Š” ์–ด๋–ป๊ฒŒ ์“ฐ๋Š”๊ฑฐ์•ผ?

๏ปฟใ€€๋Œ€๋ถ€๋ถ„์˜ ์˜คํ”ˆ์†Œ์Šค๋Š” hello world ํ˜น์€ quick start ๊ฐ™์ด ์ฒ˜์Œ ์ ‘ํ•˜๋Š” ์‚ฌ๋žŒ๋“ค์„ ์œ„ํ•œ ๋„ํ๋จผํŠธ๊ฐ€ ์žˆ๊ธฐ ๋งˆ๋ จ. hugo๋„ ๋งˆ์ฐฌ๊ฐ€์ง€๋กœ quick-start๊ฐ€ ์žˆ์—ˆ๊ณ  ์ด๋ฅผ ์ฒœ์ฒœํžˆ ๋”ฐ๋ผ ํ•˜๋ฉด ์ƒ๊ฐ๋ณด๋‹ค ์‰ฝ๊ฒŒ ์ดˆ๊ธฐ ์„ธํŒ…์„ ํ•  ์ˆ˜ ์žˆ์—ˆ… ์„๊บผ๋ผ ๊ธฐ๋Œ€ํ–ˆ์ง€๋งŒ ์•ฝ๊ฐ„ ์ดˆ๊ธฐ ์„ค์ • ๊ณผ์ •์ด ์–ด๋ ค์›Œ์„œ ๋‚จ๊ฒจ ๋‘๊ณ ์ž ํ•œ๋‹ค.

์ฐธ๊ณ ๋กœ ํ•„์ž๋Š” ์œˆ๋„ 10 ํ™˜๊ฒฝ์—์„œ ๊ตฌ์„ฑํ•˜์˜€๋‹ค. mac์ด๋ผ๋ฉด ๋” ์‰ฝ๊ฒŒ ์„ค์ •ํ•  ์ˆ˜ ์žˆ๋Š” ๊ฒƒ ๊ฐ™์€๋ฐ ์ด ๋ถ€๋ถ„์€ OS์˜ ์ฐจ์ด์—์„œ ์ƒ๊ฒจ๋‚˜๋Š” ์–ด์ฉ” ์ˆ˜ ์—†๋Š” ์•ฝ๊ฐ„์˜ ์žฅ๋ฒฝ์ด๋ผ ์ƒ๊ฐํ•œ๋‹ค. ์ด์œ ํ…Œ๋งˆ์™€ ์ƒˆ๋กœ์šด ํ™˜๊ฒฝ์„ ์‚ฌ์šฉํ•  ์ˆ˜ ์žˆ๋‹ค๋Š” ๊ธฐ๋Œ€๊ฐ์œผ๋กœ ๊พน ์ฐธ์•„๋ณธ๋‹ค.

๊ธฐ๋ณธ์„ค์ •

ใ€€๏ปฟgit์ด ์„ค์น˜๋˜์–ด ์žˆ๋‹ค๋Š” ๊ฐ€์ •ํ•˜์— ์šฐ์„  hugo๋Š” go ์–ธ์–ด๊ธฐ๋ฐ˜์œผ๋กœ ๋Œ์•„๊ฐ€๊ธฐ์— ์šฐ์„  go๋ฅผ ์„ค์น˜ํ•ด์•ผ ํ•œ๋‹ค. ๋‹ค์šด๋กœ๋“œํŽ˜์ด์ง€์—์„œ ํ™˜๊ฒฝ์— ๋งž๋Š” ์„ค์น˜ ํŒŒ์ผ์„ ๋‹ค์šด๋กœ๋“œํ•˜๊ณ  ์„ค์น˜๋ฅผ ํ•ด์ค€๋‹ค. ๋‹ค์Œ์œผ๋กœ ํŒจํ‚ค์ง€ ๊ด€๋ฆฌ์ž์ธ chocolatey ๋˜ํ•œ ์„ค์น˜๊ฐ€ ํ•„์š”ํ•˜๋‹ค. ๊ณต์‹ ํ™ˆํŽ˜์ด์ง€ํŽ˜์ด์ง€์—์„œ ๋‚˜์™€์žˆ๋Š” ์ˆœ์„œ๋Œ€๋กœ ์ง„ํ–‰ํ•˜๋ฉด ์„ค์น˜ ์™„๋ฃŒ. ํ•„์ž๋Š” ์—ฌ๊ธฐ์„œ ์ง„ํ–‰์ด ์ž˜ ์•ˆ๋์—ˆ๋Š”๋ฐ, ‘๊ด€๋ฆฌ์ž ๊ถŒํ•œ’์œผ๋กœ PowerShell ์„ ์‹คํ–‰์‹œ์ผœ์•ผ์ง€๋งŒ ์„ฑ๊ณต์„ ํ•  ์ˆ˜ ์žˆ์—ˆ๋‹ค.๏ปฟ

๏ปฟใ€€์œ„ ์„ค์ •์ด ์™„๋ฃŒ๋˜์—ˆ์œผ๋ฉด ๋“œ๋””์–ด hugo๋ฅผ ์„ค์น˜ํ•ด ์ฃผ๊ณ  ์ดˆ๊ธฐํ™”๋ฅผ ํ•ด์ค€ ๋’ค ์ƒ˜ํ”Œ๋กœ ๊ธ€ ํ•˜๋‚˜๋ฅผ ๋งŒ๋“ค๊ณ  ์„œ๋ฒ„๋ฅผ ๋„์šฐ๋ฉด ๋.

๊ทธ๋Ÿฐ ๊ฐœ๋ฐœ์ž๋กœ ๊ดœ์ฐฎ์€๊ฐ€ - '๋กœ๊ทธ & ๋ชจ๋‹ˆํ„ฐ๋ง' ํŽธ

ใ€€์บ๋ฆญํ„ฐ๋ฅผ ์œก์„ฑํ•˜๋ฉฐ ๊ฒŒ์ž„ํ•˜๋Š” ๊ฒฝ์šฐ๋ฅผ ์ƒ๊ฐํ•ด ๋ณด์ž. ๋” ์ข‹์€ ์•„์ดํ…œ์„ ์–ป๊ฑฐ๋‚˜ ํ€˜์ŠคํŠธ๋ฅผ ๋‹ฌ์„ฑํ•˜๊ธฐ ์œ„ํ•ด ๋‹น์‹ ์€ ๋‹ค์–‘ํ•œ ๋ฐฉ๋ฒ•์„ ํ†ตํ•ด ์บ๋ฆญํ„ฐ๋ฅผ ์„ฑ์žฅ์‹œํ‚จ๋‹ค. ์‚ฌ๋ƒฅ์„ ํ•˜๋‹ค ์ฒด๋ ฅ์ด ๋–จ์–ด์ง€๊ฒŒ ๋˜๋ฉด ๋ฌผ์•ฝ์„ ๋จน๊ณ , ์บ๋ฆญํ„ฐ์˜ ๋Šฅ๋ ฅ ์ค‘ ๋ถ€์กฑํ•œ ๋ถ€๋ถ„์ด ์žˆ์œผ๋ฉด ํ›ˆ๋ จ์„ ๋” ํ•˜๊ฑฐ๋‚˜ ๊ทธ์— ๋งž๋Š” ์•„์ดํ…œ์„ ์žฅ์ฐฉํ•˜๊ฒŒ ๋œ๋‹ค. ์ด๋ ‡๊ฒŒ ์บ๋ฆญํ„ฐ์˜ ‘์ƒํƒœ’๋ฅผ ์ ์ ˆํ•œ UI๋ฅผ ํ†ตํ•ด ์‚ฌ์šฉ์ž์—๊ฒŒ ์•Œ๋ ค์ฃผ๊ธฐ ๋•Œ๋ฌธ์— ‘ํ™•์ธ’์ด ๊ฐ€๋Šฅํ•˜๊ณ  ‘๋Œ€์‘’์ด ๊ฐ€๋Šฅํ•˜๊ฒŒ ๋œ๋‹ค.

ใ€€์šฐ๋ฆฌ๊ฐ€ ๋งŒ๋“œ๋Š” ์• ํ”Œ๋ฆฌ์ผ€์ด์…˜ ๋˜ํ•œ ์œ„์—์„œ ์ด์•ผ๊ธฐ ํ•œ ๊ฒŒ์ž„์ƒ์˜ ์บ๋ฆญํ„ฐ๊ฐ€ ์•„๋‹๊นŒ ์‹ถ๋‹ค. ๋ณต์žกํ•œ ์ŠคํŽ™์„ ๋‹ค์–‘ํ•œ ํ…Œ์ŠคํŠธ ์ผ€์ด์Šค๋ฅผ ๋งŒ๋“ค๋ฉฐ ๋กœ์ง ๋™์ž‘์—๋Š” ์ด์ƒ์ด ์—†์Œ์„ ํ™•์ธํ–ˆ๋‹ค๋ฉด ๊ทธ๊ฑธ๋กœ ๋งŒ์กฑํ•  ์ˆ˜ ์žˆ์„๊นŒ? ๊ฐœ๋ฐœ์ž์˜ ‘๋ ˆ๋ฒจ’์€ ์ด ๋ถ€๋ถ„์—์„œ ์ฐจ์ด๊ฐ€ ๋‚œ๋‹ค๊ณ  ์ƒ๊ฐํ•œ๋‹ค. ์šด์˜ํ™˜๊ฒฝ์— ์ถœ์‹œํ•œ ์• ํ”Œ๋ฆฌ์ผ€์ด์…˜์— ์—๋Ÿฌ๊ฐ€ ๋‚˜๋Š”์ง€, ํŠธ๋ž˜ํ”ฝ์ด ์–ผ๋งˆ๋‚˜ ๋“ค์–ด์˜ค๊ณ  ์žˆ๊ณ  ํŠธ๋ž˜ํ”ฝ์˜ ์œ ํ˜•์€ ๋˜ ์–ด๋– ํ•œ์ง€, ์š”์ฒญ์— ๋Œ€ํ•œ ์‘๋‹ต์†๋„๋Š” ์–ด๋–ป๊ณ  ์„œ๋ฒ„์˜ ์‹œ์Šคํ…œ ์ง€ํ‘œ์—๋Š” ๋ฌธ์ œ๊ฐ€ ์—†๋Š”์ง€ ๋“ฑ๋“ฑ. ์• ํ”Œ๋ฆฌ์ผ€์ด์…˜์˜ ์œ ํ˜•์— ๋”ฐ๋ผ ๋‹ค์–‘ํ•˜๊ฒ ์ง€๋งŒ ์ ์ ˆํ•œ ๋กœ๊ทธ๋ฅผ ์ด์šฉํ•˜์—ฌ ์• ํ”Œ๋ฆฌ์ผ€์ด์…˜์˜ ‘์ƒํƒœ’๋ฅผ ํ™•์ธํ•˜๊ณ  ๋ฌธ์ œ๊ฐ€ ์žˆ๋‹ค๋ฉด ‘๋Œ€์‘’ํ•˜๋Š” ๊ฒŒ ๊ผญ ํ•„์š”ํ•˜๋‹ค๊ณ  ์ƒ๊ฐํ•œ๋‹ค.

ใ€€์ด๋ฒˆ ํฌ์ŠคํŒ…์—์„œ๋Š” ํฌ๊ฒŒ ๋กœ๊น…๊ณผ ๋ชจ๋‹ˆํ„ฐ๋ง์— ๋Œ€ํ•ด ์•Œ์•„๋ณด๊ณ ์ž ํ•œ๋‹ค. ์ด๋ฅผ ํ†ตํ•ด ์• ํ”Œ๋ฆฌ์ผ€์ด์…˜์˜ ‘๊ฐœ๋ฐœ’์—๋งŒ ์ง‘์ค‘ํ•˜๊ณ  ์žˆ๋˜ ๊ด€์ ์„ ๋ณด๋‹ค ๋” ๋†’์€ ๊ณณ์—์„œ ๋ฐ”๋ผ๋ณด๋ฉฐ ์• ํ”Œ๋ฆฌ์ผ€์ด์…˜์˜ ‘์šด์˜’ ์ธก๋ฉด์—์„œ๋„ ๊ณ ๋ฏผํ•ด ๋ณด๋Š” ๊ธฐํšŒ๊ฐ€ ๋˜์—ˆ์œผ๋ฉด ํ•œ๋‹ค.

ํ•„์ž๋Š” ์„œ๋ฒ„ ๊ฐœ๋ฐœ์ž์ด๋‹ค ๋ณด๋‹ˆ ๊ธ€์˜ ๋‚ด์šฉ์ด ๋‹ค์†Œ ์„œ๋ฒ„ ๊ฐœ๋ฐœ์ž์˜ ์‹œ์„ ์—์„œ ์ž‘์„ฑํ•˜๊ฒŒ ๋˜์—ˆ๋‹ค. ํ•˜์ง€๋งŒ ‘๊ฐœ๋ฐœ์ž’๋ผ๋ฉด ์œ ํ˜•๋งŒ ๋‹ค๋ฅด์ง€ ๋Œ€๋ถ€๋ถ„ ๋น„์Šทํ•˜๊ธฐ ๋•Œ๋ฌธ์— ํฌ๊ฒŒ ๋‹ค๋ฅด์ง€ ์•Š๋‹ค๊ณ  ์ƒ๊ฐํ•œ๋‹ค.

๋กœ๊ทธ๋Š” ์–ด๋–ค๊ฑธ, ์–ด๋–ป๊ฒŒ ๋‚จ๊ฒจ์•ผ ํ• ๊นŒ?

ใ€€๏ปฟ๋กœ๊ทธ๊ฐ€ ์™œ ํ•„์š”ํ•œ์ง€์— ๋Œ€ํ•œ ๋‚ด์šฉ์€ ๋‹ค๋ฃจ์ง€ ์•Š๊ฒ ๋‹ค. (๊ตณ์ด ๋งํ•˜์ง€ ์•Š์•„๋„ ๊ทธ๋งŒํผ ์ค‘์š”ํ•˜๋‹ค๋Š” ํ‘œํ˜„์ด ๋” ์–ด์šธ๋ฆด ์ˆ˜๋„ ์žˆ๊ฒ ๋‹ค.) ๊ทธ๋ ‡๋‹ค๋ฉด ์šฐ์„  ์–ด๋–ค ๋กœ๊ทธ๋ฅผ ๋‚จ๊ฒจ์•ผ ํ• ๊นŒ?

/images/a-good-developer-in-terms-of-Log-and-Monitoring/talk.jpg
ํ•„์ž๊ฐ€ ๊ฟˆ๋‚˜๋ฌด ์‹œ์ ˆ๋•Œ ๋‚˜๋ˆ„์—ˆ๋˜ ์กฐ์ง์žฅ๋‹˜๊ณผ์˜ ๋Œ€ํ™” ๋‚ด์šฉ

ใ€€๏ปฟ์•„์ง๊นŒ์ง€๋„ ๊ธฐ์–ต์— ๋‚จ์•„์žˆ๋Š” ์˜ˆ์ „ ์กฐ์ง ์žฅ๋‹˜๊ณผ์˜ ๋Œ€ํ™”. ์ผ๋‹จ ๋กœ๊ทธ๋Š” ์ตœ๋Œ€ํ•œ ๋งŽ์ด (๊ณผํ•˜๊ฒŒ) ๋‚จ๊ฒจ์•ผ ํ•œ๋‹ค๊ณ  ์ƒ๊ฐํ•œ๋‹ค. ๊ทธ๋‹ค์Œ ๋ถˆํ•„์š”ํ•œ ๋กœ๊ทธ๋“ค์€ ์ œ๊ฑฐํ•˜๊ฑฐ๋‚˜ ๋ ˆ๋ฒจ์„ ๋‚ฎ์ถ”๋Š” ๋“ฑ ์ƒํ™ฉ์— ๋งž๋„๋ก ์ปค์Šคํ„ฐ๋งˆ์ด์ง•์ด ํ•„์š”ํ•˜๋‹ค. ๊ฒฝํ—˜์„ ํ•ด๋ณด๋ฉด ์•Œ๊ฒ ์ง€๋งŒ ์šด์˜ํ™˜๊ฒฝ์— ์• ํ”Œ๋ฆฌ์ผ€์ด์…˜์„ ๋ฐฐํฌํ•˜๊ณ  ์„œ๋น„์Šค๋ฅผ ์šด์˜ํ•˜๋‹ค ๋ณด๋ฉด ๊ฐœ๋ฐœ ํ™˜๊ฒฝ์—์„œ ๋งŒ๋‚˜๊ธฐ ์–ด๋ ต๊ฑฐ๋‚˜ ๊ฒฝํ—˜ํ•ด๋ณด์ง€ ๋ชปํ•œ ์ƒํ™ฉ์ด ๋ฐœ์ƒํ•˜๊ณค ํ•œ๋‹ค. ์ด๋Ÿด ๋•Œ ์ƒํ™ฉ์— ๋งž๋Š” ๋กœ๊ทธ๋“ค์ด ์žˆ๋‹ค๋ฉด ๋ฏธ๋ฆฌ ๋‚จ๊ฒจ๋‘” ๋กœ๊ทธ๋ฅผ ํ†ตํ•ด ๋” ํšจ๊ณผ์ ์œผ๋กœ ์ƒํ™ฉ์„ ํŒŒ์•…ํ•  ์ˆ˜ ์žˆ๋‹ค. ํŠธ๋ž˜ํ”ฝ์˜ ์ •๋ณด(request url, parameter, UA, remote ip ๋“ฑ)๋ฅผ ๋‚จ๊ฒจ์„œ ์™ธ๋ถ€์—์„œ ํ˜ธ์ถœํ•˜๋Š” ํ˜•ํƒœ๋ฅผ ๋ถ„์„ํ•˜๋Š”๋ฐ ํ™œ์šฉํ•  ์ˆ˜๋„ ์žˆ๊ณ , ์• ํ”Œ๋ฆฌ์ผ€์ด์…˜์—์„œ ์™ธ๋ถ€๋กœ ํ˜ธ์ถœ์„ ํ•˜๊ณ  ๋‚œ ๋’ค์— ๋ฐ›๋Š” ์‘๋‹ต์— ๋Œ€ํ•ด์„œ ๋กœ๊ทธ๋ฅผ ๋‚จ๊ฒจ๋‘๋ฉด ์™ธ๋ถ€ ํ†ต์‹ ์˜ ์˜ค๋ฅ˜๋ฅผ ํŒŒ์•…ํ•˜๋Š” ๋ฐ ๋„์›€์ด ๋  ์ˆ˜ ์žˆ๋‹ค. ์–ด๋–ค ๋กœ๊ทธ๋ฅผ ๋‚จ๊ฒจ์•ผ ํ•˜๋Š”๊ฐ€์— ๋Œ€ํ•œ ๊ณ ๋ฏผ์€ ์šด์˜ํ•˜๋Š” ์• ํ”Œ๋ฆฌ์ผ€์ด์…˜์ด ์–ด๋–ค ํ–‰๋™์„ ํ•˜๋Š”๊ฐ€์— ๊ด€์ ์„ ๋‘๊ณ  ๊ณ ๋ฏผํ•ด๋ณด๋ฉด ์ข€ ๋” ์‰ฝ๊ฒŒ ์ฐพ์„ ์ˆ˜ ์žˆ์„ ๊ฒƒ์ด๋ผ ์ƒ๊ฐํ•œ๋‹ค.

ใ€€๋กœ๊ทธ๋ฅผ ๋‚จ๊ธฐ๋Š” ๋ฐฉ๋ฒ• ๋˜ํ•œ ๋‹ค์–‘ํ•˜๋‹ค. ์‹œ์Šคํ…œ ๋กœ์ปฌ์— ํŒŒ์ผ๋กœ ๋‚จ๊ธฐ๊ฑฐ๋‚˜ ํŠน์ • ๋กœ๊ทธ ์„œ๋ฒ„๋ฅผ ์„ค์ •ํ•˜์—ฌ ์—ฌ๋Ÿฌ ๋Œ€์˜ ์„œ๋ฒ„ ๋กœ๊ทธ๋ฅผ ํ•œ๊ณณ์—์„œ ๋ณผ ์ˆ˜๋„ ์žˆ๋‹ค. ๋‹ค๋งŒ ๋กœ๊ทธ๋ฅผ ‘๋‚จ๊ธฐ๋Š”’ ๊ฒƒ ๋˜ํ•œ ํ•˜๋‚˜์˜ ๋น„์šฉ์— ํฌํ•จ๋˜๊ธฐ ๋•Œ๋ฌธ์— ์• ํ”Œ๋ฆฌ์ผ€์ด์…˜์˜ ๊ธฐ๋Šฅ์— ์ตœ๋Œ€ํ•œ ์˜ํ–ฅ์ด ๊ฐ€์ง€ ์•Š๋„๋ก ์ตœ๋Œ€ํ•œ ๋น ๋ฅธ ์‹œ๊ฐ„ ๋‚ด์— ์ฒ˜๋ฆฌ๊ฐ€ ๋˜๋„๋ก ํ•ด์•ผ ํ•œ๋‹ค. (ํ˜น์€ ๋น„๋™๊ธฐ๋กœ ๋‚จ๊ธฐ๊ฑฐ๋‚˜ ๋“ฑ)

ใ€€๋กœ๊ทธ๋ฅผ ๋‚จ๊ธฐ๋Š” ์ด์œ  ์ค‘ ๊ฐ€์žฅ ํฐ ์ด์œ ๋Š” ‘๋‚˜์ค‘์— ๋ณด๊ธฐ ์œ„ํ•ด์„œ’์ด๋‹ค. ๊ทธ๋งŒํผ ํ•œ๋ฒˆ ๋กœ๊ทธ๋ฅผ ๋‚จ๊ธธ ๋•Œ์—๋„ ๋ณด๊ธฐ ์ข‹๊ฒŒ ๋‚จ๊ฒจ์•ผ ํ•œ๋‹ค. ์˜ˆ์ปจ๋Œ€, ์•„๋ž˜์— ์ ์–ด๋†“์€ ๋กœ๊ทธ ๋ฐฉ์‹์˜ ๊ฒฝ์šฐ ์ž‘์€ ์ฐจ์ด์ง€๋งŒ ๋‚˜์ค‘์— ๋ณผ ๋•Œ ๊ฝค ํฐ ์ฐจ์ด๋ฅผ ์œ ๋ฐœํ•œ๋‹ค. ๏ปฟ

  • ์•ˆ์ข‹์€ ์˜ˆ
try {
	...
} catch (Exception e){
	log.Error(e); // ์–ด๋–ค ์ƒํ™ฉ์ด์ง€..?
}
  • ๋ณด๋‹ค ์กฐ๊ธˆ ๋” ์ข‹์€ ์˜ˆ
try {
	...
} catch (Exception e){
	log.Error("url : " + url + ", parameter : " + parameter + ", remote ip : " + remoteIp, e); // ๋กœ๊ทธ๋Š” ๊ฐ€๊ธ‰์  ์ž์„ธํ•˜๊ฒŒ !
}

๋กœ๊ทธ๊ฐ€ ๊ฐ€์ ธ๋‹ค ์ฃผ๋Š” ๋˜ ๋‹ค๋ฅธ ์„ธ์ƒ

ใ€€๋กœ๊ทธ๋Š” ๋˜ ๋‹ค๋ฅธ ๋ฐ์ดํ„ฐ๊ฐ€ ๋  ์ˆ˜ ์žˆ๋‹ค. ๊ธ€ ๋ชฉ๋ก์„ ๋ณด์—ฌ์ฃผ๋Š” ์›นํŽ˜์ด์ง€๊ฐ€ ์žˆ๋‹ค๊ณ  ๊ฐ€์ •ํ•ด๋ณด์ž. ์ด๋•Œ ์‚ฌ์šฉ์ž๋“ค์ด ์–ด๋–ค ๊ธ€์„ ๋” ๋งŽ์ด ์ฝ๋Š”์ง€ ‘ํด๋ฆญ ์ง€ํ‘œ’์— ๋Œ€ํ•œ ๋กœ๊ทธ๋ฅผ ๋‚จ๊ฒจ ๋‘”๋‹ค๋ฉด ‘์ธ๊ธฐ๊ธ€’ ๊ฐ™์€ ๋˜ ๋‹ค๋ฅธ ์›น ํŽ˜์ด์ง€๊ฐ€ ๋‚˜์˜ฌ ์ˆ˜ ์žˆ์„ ๊ฒƒ ๊ฐ™๋‹ค. ๋งŒ์•ฝ ๊ทธ ํŽ˜์ด์ง€๊ฐ€ ํšŒ์›๋งŒ ์ฝ์„ ์ˆ˜ ์žˆ๋Š” ํŽ˜์ด์ง€๋ผ๋ฉด ‘20๋Œ€ ๋‚จ์„ฑ์ด ์›”์š”์ผ ์˜คํ›„์— ๋งŽ์ด ์ฝ๋Š” ๊ธ€’ ๊ฐ™์ด ํšŒ์›์˜ ์ •๋ณด๋ฅผ ์กฐํ•ฉํ•˜์—ฌ ์ƒˆ๋กœ์šด ๋ฐ์ดํ„ฐ๋ฅผ ๋งŒ๋“ค ์ˆ˜ ์žˆ๊ฒŒ ๋œ๋‹ค. ์ด๋Ÿฌํ•œ ๋ฐ์ดํ„ฐ๋“ค์€ ๋ณด๋‹ค ๋” ์ข‹์€ ์„œ๋น„์Šค๋ฅผ ํ•  ์ˆ˜ ์žˆ๊ฒŒ ๋„์™€์ค„ ์ˆ˜ ์žˆ๋Š” ๋ฐ‘๊ฑฐ๋ฆ„์ด ๋˜๊ณ  ๊ทธ ๋ฐ”ํƒ•์€ ๋กœ๊ทธ๋ผ๋Š” ๊ฑธ ๋ช…์‹ฌํ•˜์ž.

๋นŒ๋“œ/ํ…Œ์ŠคํŠธ๋Š” ๋‚ด๊ฐ€ ํ•ด์ค„๊ฒŒ. ๋„ˆ๋Š” ์ฝ”๋”ฉ์— ์ง‘์ค‘ํ•ด (by GitHub Pull Request Builder)

ใ€€git ์€ ๋ถ„์‚ฐ ๋ฒ„์ „ ๊ด€๋ฆฌ ์‹œ์Šคํ…œ ์ค‘ ๊ฐ€์žฅ ์ž˜ ์•Œ๋ ค์ ธ ์žˆ๋‹ค๊ณ  ํ•ด๋„ ๊ณผ์–ธ์ด ์•„๋‹ ์ •๋„๋กœ ๋Œ€๋ถ€๋ถ„์˜ ์‹œ์Šคํ…œ์—์„œ ์‚ฌ์šฉ๋˜๊ณ  ์žˆ๋Š” ๊ฒƒ ๊ฐ™๋‹ค. ์ด๋ฅผ ์›น์„œ๋น„์Šค์—์„œ ๋ณด๋‹ค ํŽธํ•˜๊ฒŒ ์‚ฌ์šฉํ•  ์ˆ˜ ์žˆ๋„๋ก ํ•œ ์‹œ์Šคํ…œ์ด Github. Github ์„ ์‚ฌ์šฉํ•˜๋Š” ์ด์œ  ์ค‘์— ๊ฐ€์žฅ ํฐ ์ด์œ ๋ฅผ ํ•˜๋‚˜๋งŒ ์ด์•ผ๊ธฐํ•ด๋ณด์ž๋ฉด ๋ฐ”๋กœ ์˜จ๋ผ์ธ์ƒ์—์„œ ์ฝ”๋“œ ๋ฆฌ๋ทฐ๋ฅผ ํ•  ์ˆ˜ ์žˆ๋Š” pullRequest๋ผ๋Š” ๊ธฐ๋Šฅ ๋•Œ๋ฌธ์ด ์•„๋‹๊นŒ ์กฐ์‹ฌ์Šค๋Ÿฝ๊ฒŒ ์ƒ๊ฐ์„ ํ•ด๋ณธ๋‹ค.

ใ€€pullRequest๋Š” work branch์—์„œ ์ž‘์—…ํ•œ ๋‚ด์šฉ์„ base branch๋กœ merge ์ „ ๊ผญ ์ฝ”๋“œ ๋ฆฌ๋ทฐ๊ฐ€ ์•„๋‹ˆ๋”๋ผ๋„ ์ž‘์—…ํ•œ ๋‚ด์šฉ์— ๋Œ€ํ•ด์„œ ๋‹ค์–‘ํ•œ ๊ฒ€์‚ฌ๋ฅผ ์ž๋™ํ™”ํ•  ์ˆ˜ ์žˆ๋Š” ๊ฐ•๋ ฅํ•œ ๊ธฐ๋Šฅ๋“ค์ด ๋งŽ๋‹ค. ์ด๋Ÿฌํ•œ ์ž๋™ํ™”๋Š” CI(์ง€์†์  ํ†ตํ•ฉ) ๊ด€์ ์—์„œ ๋งค์šฐ ์ค‘์š”ํ•œ๋ฐ ์ฝ”๋“œ์— ๋Œ€ํ•ด ์ฒดํฌํ•ด์•ผ ํ•  ๋ถ€๋ถ„๋“ค(๋นŒ๋“œ, ํ…Œ์ŠคํŠธ, ์ •์  ๋ถ„์„ ๋“ฑ)์„ “์•Œ์•„์„œ” ํ•ด์ค€๋‹ค๋ฉด ์ž‘์—…์ž๋Š” ์˜ค๋กฏ์ด ๋น„์ฆˆ๋‹ˆ์Šค ๋กœ์ง ๊ฐœ๋ฐœ์— ๋Œ€ํ•ด์„œ๋งŒ ์‹ ๊ฒฝ ์“ธ ์ˆ˜ ์žˆ์œผ๋‹ˆ ์ƒ์‚ฐ์„ฑ ์ ˆ์•ฝ ์ธก๋ฉด์—์„œ ์—„์ฒญ๋‚œ ํšจ๊ณผ๋ฅผ ๋ณผ ์ˆ˜ ์žˆ๋‹ค.

/images/github-pullrequest-build/car.gif
๋‚ด๊ฐ€ ํ•˜๋Š”์ผ์—๋งŒ ์ง‘์ค‘ํ•  ์ˆ˜ ์žˆ๊ฒŒ!
์ถœ์ฒ˜ : https://www.clien.net/service/board/park/10453442

์ด๋ฒˆ ํฌ์ŠคํŒ…์—์„œ๋Š” ๊ทธ์ค‘์—์„œ๋„ ์•„์ฃผ ๊ฐ„๋‹จํ•œ ์„ค์ •๋งŒ์œผ๋กœ work branch์˜ ๋นŒ๋“œ ์ƒํƒœ๋ฅผ ๊ฒ€์‚ฌํ•ด ๋ณผ ์ˆ˜ ์žˆ๋Š” Jenkins์˜ Github Pull Request Builder๋ฅผ ์„ค์น˜ ๋ฐ ํ™œ์šฉํ•ด ๋ณด๊ณ ์ž ํ•œ๋‹ค.

์‚ฌ์‹ค ์ตœ๊ทผ ํŒ€์—์„œ CI ์„œ๋ฒ„๋ฅผ ์ด์ „ํ•ด์•ผ ํ–ˆ์—ˆ๋‹ค. ๋จธ๋ฆฟ์†์—์„œ๋Š” ์–ด๋–ป๊ฒŒ ํ•˜๋ฉด ๋˜๊ฒ ์ง€ ์‹ถ์—ˆ์ง€๋งŒ ๋ง‰์ƒ ํ•ด๋ณด๋ ค๋‹ˆ Jenkins ๋ฒ„์ „์—…๋„ ๋˜์—ˆ๊ณ  ๋ญ๋ถ€ํ„ฐ ํ•ด์•ผ ํ• ์ง€ ํ—ˆ๋‘ฅ๋Œ€๋Š” ํ•„์ž๊ฐ€ ๋ถ€๋„๋Ÿฌ์› ๋‹ค. ์ด์ฐธ์— ์ •๋ฆฌ๋ฅผ ํ•ด๋ณด๋ฉฐ ๋‹ค์‹œ ํ•œ๋ฒˆ ๋ฆฌ๋งˆ์ธ๋“œ ํ•˜๋Š” ์‹œ๊ฐ„์„ ๊ฐ€์ ธ๋ณด๊ณ ์ž ํ•œ๋‹ค. (์ด๋ž˜์„œ ๊ธฐ์–ต๋ณด๋‹ค ๊ธฐ๋ก์ด ์ค‘์š”ํ•˜๋‹ค.)

์ค€๋น„๋ฌผ

ใ€€์ „์ฒด์ ์ธ ํ๋ฆ„์€ ์•„๋ž˜ ๊ทธ๋ฆผ์ฒ˜๋Ÿผ ํ˜๋Ÿฌ๊ฐ€๊ธฐ ๋•Œ๋ฌธ์— ๋‹น์—ฐํžˆ ์„œ๋ฒ„์— Jenkins ๊ฐ€ ์„ค์น˜๋˜์–ด ์žˆ์–ด์•ผ ํ•œ๋‹ค. Jenkins ์„ค์น˜๋Š” ํ•„์ž์˜ ํฌ์ŠคํŒ…(Jenkins ์„ค์น˜ ์น˜ํŠธํ‚ค)๋ฅผ ์ฐธ๊ณ ํ•ด ๋ณด๋Š” ๊ฒƒ๋„ ์ข‹์„ ๊ฒƒ ๊ฐ™๋‹ค.

/images/github-pullrequest-build/programmer-github-jenkins.jpg
์ „์ฒด์ ์ธ ํ๋ฆ„

ใ€€์ฐธ๊ณ ๋กœ ํ•„์ž๋Š” GitHub Enterprise ๋ฒ„์ „์—์„œ ์‚ฌ์šฉํ–ˆ๋Š”๋ฐ ์ผ๋ฐ˜ Github์—์„œ๋„ ๋™์ผํ•œ ๋ฐฉ๋ฒ•์œผ๋กœ ์‚ฌ์šฉ ๊ฐ€๋Šฅํ•˜๋‹ค.

Github๊ณผ Jenkins์˜ ์—ฐ๋™์„ ์œ„ํ•œ 2๊ฐ€์ง€ ์„ค์ •

ใ€€Github ๊ณผ Jenkins ๊ฐ€ ํ†ต์‹ ์ด ๋˜๋„๋ก ์„ค์ •ํ•ด ์ค˜์•ผ ํ•œ๋‹ค. ๊ทธ๋ž˜์•ผ Github์˜ ์ฝ”๋“œ๋ฅผ ๋ฐ›์•„์„œ Jenkins ๊ฐ€ ๋นŒ๋“œ๋ฅผ ํ•˜๊ณ  ๊ทธ ๋นŒ๋“œ ๊ฒฐ๊ณผ๋ฅผ ๋‹ค์‹œ Github์— ๋ฆฌํฌํŠธ๊ฐ€ ๊ฐ€๋Šฅํ•ด์ง€๊ธฐ ๋•Œ๋ฌธ์ด๋‹ค. ๋จผ์ € ์ฒซ ๋ฒˆ์งธ๋กœ ssh ์„ค์ •์œผ๋กœ Github์˜ ์ฝ”๋“œ๋ฅผ ๊ฐ€์ ธ์˜ค๋„๋ก ssh ์„ค์ •์„ ํ•ด๋‘์ž. ssh ์„ค์ •ํ•˜๋Š” ๋ฐฉ๋ฒ•์€ ํ•„์ž์˜ ํฌ์ŠคํŒ…(Github๊ณผ Jenkins ์—ฐ๋™ํ•˜๊ธฐ)ํŽธ์„ ํ™•์ธํ•ด๋ณด๋ฉด ๋  ๊ฒƒ ๊ฐ™๋‹ค.

ใ€€๊ทธ๋‹ค์Œ์œผ๋กœ ์•„๋ž˜์—์„œ ์ด์•ผ๊ธฐํ•  GitHub Pull Request Builder๋ผ๋Š” Jenkins plugin ์ด ๋นŒ๋“œ๊ฐ€ ๋๋‚œ ๋’ค์— ๊ฒฐ๊ณผ๋ฅผ ๋ฆฌํฌํŒ… ํ•ด์ค„ ์ˆ˜ ์žˆ๋Š” ์ธ์ฆ ํ† ํฐ์„ ๋ฐœ๊ธ‰๋ฐ›์•„๋‘์ž. Github > Settings > Developer settings > Personal access tokens ํ™”๋ฉด์—์„œ ํ‚ค๋ฅผ ์ƒ์„ฑํ•˜๊ณ  ๋งŒ๋“ค์–ด์ง„ ํ‚ค๋ฅผ ์ €์žฅํ•ด ๋‘”๋‹ค. (์ด ํ‚ค๋Š” ๋ณด์•ˆ์— ์œ ์˜ํ•ด์•ผ ํ•˜๊ณ , ํ™”๋ฉด ๊ฒฝ๊ณ (?)์—์„œ๋„ ๋ณผ ์ˆ˜ ์žˆ๋“ฏ์ด ํ‚ค๋Š” ์ƒ์„ฑ ์‹œ ํ•œ ๋ฒˆ๋ฐ–์— ๋ณผ ์ˆ˜ ์—†๊ธฐ ๋•Œ๋ฌธ์— ๋ฏธ๋ฆฌ ์ €์žฅํ•ด ๋‘ฌ์•ผ ํ•œ๋‹ค.)

/images/github-pullrequest-build/github-access-token.jpg
์ธ์ฆํ† ํฐ์„ ๋ฏธ๋ฆฌ ๋ฐ›์•„๋‘์ž.

Jenkins ์„ค์ •

ใ€€Jenkins > ๊ด€๋ฆฌ > pluginManager์— ๋“ค์–ด๊ฐ€ GitHub Pull Request Builder๋ฅผ ๊ฒ€์ƒ‰ ํ›„ ์„ค์น˜ํ•ด ์ค€๋‹ค. ๊ทธ๋Ÿฌ๊ณ  ๋‚˜์„œ Jenkins > ๊ด€๋ฆฌ > ํ™˜๊ฒฝ์„ค์ •์— ๋“ค์–ด๊ฐ€ ๋ณด๋ฉด ์•„๋ž˜์™€ ๊ฐ™์ด GitHub Pull Request Builder ํ•ญ๋ชฉ์ด ์ƒ๊ธด ๊ฒƒ์„ ํ™•์ธํ•  ์ˆ˜ ์žˆ๊ณ  ์œ„์—์„œ ์„ค์ •ํ•œ ์ธ์ฆํ† ํฐ์„ ์•„๋ž˜์ฒ˜๋Ÿผ ๋“ฑ๋ก ํ›„ ์ €์žฅ์„ ํ•œ๋‹ค.

/images/github-pullrequest-build/add-github-access-token.jpg
credentials ์„ ์œ„์—์„œ ๋ฐœ๊ธ‰๋ฐ›์€ ์ธ์ฆํ† ํฐ์œผ๋กœ ๋“ฑ๋กํ•ด์ค€๋‹ค.

ใ€€Jenkins job์„ ํ•˜๋‚˜ ๋งŒ๋“ค๊ณ  pullRequest ๊ฐ€ ๋ฐœ์ƒํ–ˆ์„ ๋•Œ ์ž๋™์œผ๋กœ ์‹คํ–‰๋  ์ˆ˜ ์žˆ๋„๋ก ์„ค์ •์„ ํ•ด์ค€๋‹ค. ๋จผ์ € General ํƒญ์— Github project์— Github url ์„ ์ ์–ด์ฃผ๊ณ 

/images/github-pullrequest-build/jenkins-general.jpg

ใ€€์†Œ์Šค ์ฝ”๋“œ ๊ด€๋ฆฌ ํƒญ์—์„œ ssh ์ฃผ์†Œ๋ฅผ ์ ๊ณ  ์œ„์—์„œ ๋ฏธ๋ฆฌ ์„ค์ •ํ•œ ssh ํ‚ค๋กœ credentials ๊ฐ’์„ ๋„ฃ์–ด์ค€๋‹ค. ์ „์—๋„ ์ด์•ผ๊ธฐํ–ˆ์ง€๋งŒ ์ด ๋ถ€๋ถ„์—์„œ ์˜ค๋ฅ˜๊ฐ€ ๋ฐœ์ƒํ•˜๋ฉด ๋นจ๊ฐ„์ƒ‰ ๊ธ€์”จ๋กœ ์˜ค๋ฅ˜ ๋‚ด์šฉ์ด ๋‚˜์˜ค๊ณ  ์•„๋ž˜ ํ™”๋ฉด์ฒ˜๋Ÿผ ์˜ค๋ฅ˜๊ฐ€ ์—†๋‹ค๋ฉด ์•„๋ฌด๊ฒƒ๋„ ์•ˆ ๋‚˜์˜จ๋‹ค. Refspec ์— +refs/pull/*:refs/remotes/origin/pr/* ๋ผ๊ณ  ์ ์–ด์ฃผ๊ณ  ๋ธŒ๋žœ์น˜ ์„ค์ •์€ ํŒŒ๋ผ๋ฏธํ„ฐ๋กœ ๋ฐ›์•„์™€์„œ pullRequest๋ฅผ ๋ฐœ์ƒ์‹œํ‚จ ๋ธŒ๋žœ์น˜๋ฅผ ๋นŒ๋“œ ํ•  ์ˆ˜ ์žˆ๋„๋ก ${sha1} ๋ผ๊ณ  ์ ์–ด์ฃผ์ž.

๋ฒŒ์จ 2๋…„ (feat. ํ† ์ดํ”„๋กœ์ ํŠธ ํšŒ๊ณ ,๊ฐ€์น˜,์ˆ˜์ž…)

ใ€€์ •ํ™•ํžˆ 2018๋…„ 07์›” 12์ผ ํ•„์ž์˜ ์ฒซ ํ† ์ด ํ”„๋กœ์ ํŠธ์ธ โ€˜๊ธฐ์ˆ  ๋ธ”๋กœ๊ทธ ๊ตฌ๋… ์„œ๋น„์Šคโ€™๋ฅผ ์˜คํ”ˆํ•˜๊ฒŒ ๋œ๋‹ค. ์–ผ๋งˆ๋‚˜ ๋งŽ์ด ๊ตฌ๋…(๊ฐ€์ž…) ํ•˜๊ฒ ์–ด ํ•˜๋Š” ์ƒ๊ฐ์ด ๋ถ€๋„๋Ÿฌ์šธ ๋งŒํผ 6๊ฐœ์›”์ด ์ง€๋‚˜ ๊ตฌ๋…์ž ์ˆ˜๋Š” 1,000๋ช…์„ ๋„˜๊ธฐ๊ณ  1๋…„์ด ์ง€๋‚˜ 2,000๋ช…. ์–ด๋А๋ง ๋‹ฌ๋ ฅ์„ ๋ณด๋‹ˆ ์˜ค๋Š˜์ด ์ •ํ™•ํ•˜๊ฒŒ ํ† ์ด ํ”„๋กœ์ ํŠธ๋ฅผ ์„œ๋น„์Šคํ•œ์ง€ ๋ฒŒ์จ 2๋…„์ด ๋˜๋Š” ๋‚ . ๊ตฌ๋…์ž ์ˆ˜๋Š” ์–ด๋А๋ง 3,000๋ช…์„ ๋„˜์–ด์„ ๋‹ค. ๋ญ”๊ฐ€ ๋ฟŒ๋“ฏํ•˜๋ฉด์„œ๋„ ์„œ๋น„์Šค๋ฅผ ์ข€ ๋” ๋””๋ฒจ๋กญ ํ•˜์ง€ ๋ชปํ•œ ํ•„์ž ์ž์‹ ์„ ๋Œ์•„๋ณด๋‹ˆ ๊ดœํžˆ ๋งˆ์Œ์ด ๋ฌด๊ฑฐ์›Œ์ง€๊ณ .

/images/toy-projects-second-year-review/dog.jpg
๋ญ”๊ฐ€ ํ•ด์•ผํ•˜๋Š”๋ฐ… ๊ดœํžˆ ๋ˆˆ์น˜๋งŒ ๋ณด์ด๋„ค…
์ถœ์ฒ˜ : http://egloos.zum.com/nievess/v/657827

ใ€€์ง€๋‚œ 2๋…„ ๋™์•ˆ์„ ๋Œ์ด์ผœ๋ณด๋ฉฐ ์„œ๋น„์Šค๋ฅผ ์–ด๋–ป๊ฒŒ ์šด์˜ํ•ด ์™”๋Š”์ง€, ๊ทธ๋ฆฌ๊ณ  ํ† ์ด ํ”„๋กœ์ ํŠธ๊ฐ€ ํ•„์ž์—๊ฒŒ ์–ด๋–ค ์˜ํ–ฅ์„ ์ฃผ์—ˆ๋Š”์ง€ ๋˜๋Œ์•„๋ณด๋ฉฐ ์…€ํ”„ ๋ฆฌ๋ทฐ๋ฅผ ํ•ด ๋ณด๊ณ ์ž ํ•œ๋‹ค.

์„œ๋น„์Šค ์ž์ฒด ํ‰๊ฐ€

์‹ฌํ”Œํ•œ ๊ธฐ๋Šฅ

ใ€€๋ง ๊ทธ๋Œ€๋กœ ํ† ์ด ํ”„๋กœ์ ํŠธ์ด๊ธฐ ๋•Œ๋ฌธ์— ๊ธฐ๋Šฅ ๋˜ํ•œ ์•„์ฃผ ๊ฐ„๋‹จํ•˜๋‹ค. awesome-devblog์—์„œ ์ œ๊ณตํ•˜๋Š” ๊ฐœ์ธ/๋‹จ์ฒด ๋ธ”๋กœ๊ทธ๋“ค์˜ ํฌ์ŠคํŒ…์„ ์กฐํšŒํ•˜์—ฌ ์–ด์ œ ์ž‘์„ฑ๋œ ๊ธ€๋“ค๋งŒ ๋ชจ์•„ ๋ฐœ์†กํ•œ๋‹ค. ๊ฑฐ๊ธฐ์— ์ฃผ๊ฐ„ ๋งŽ์ด ํด๋ฆญ๋œ ํฌ์ŠคํŒ…์„ ๋ชจ์•„์„œ ํ•œ ๋ฒˆ ๋” ๋ฐœ์†กํ•˜๋Š” ๊ธฐ๋Šฅ๊นŒ์ง€. ์ถ”๊ฐ€์ ์ธ ๊ธฐ๋Šฅ์„ ๋” ๋””๋ฒจ๋กญ ํ•ด์•ผ ํ•˜๋Š”๋ฐ ์•„์ด๋””์–ด๊ฐ€ ์—†์–ด์„œ ์ธ์ง€ ๋””๋ฒจ๋กญ ํ•  ํž˜์ด ์•ˆ ๋‚˜์„œ ์ธ์ง€ ์œ ์ง€๋งŒ ํ•˜๊ณ  ์žˆ๋Š” ์ƒํƒœ๋‹ค.

์„œ๋น„์Šค์— ์—†์–ด์„œ๋Š” ์•ˆ๋  ‘๋กœ๊น…(Logging)’

ใ€€ํ˜•์‹์„ ๋ง‰๋ก ํ•˜๊ณ  ์ปดํ“จํ„ฐ๋กœ ๋Œ์•„๊ฐ€๋Š” ๋ชจ๋“  ‘ํ”„๋กœ๊ทธ๋žจ’์€ ์ƒํ™ฉ์— ๋”ฐ๋ผ ๋ฏธ๋ฆฌ ๋งŒ๋“ค์–ด ๋†“์€ ๋กœ์ง์— ๋”ฐ๋ผ ์›€์ง์ด๋Š” ๋กœ๋ด‡์— ๋ถˆ๊ณผํ•˜๋‹ค. ๋ฌผ๋ก  ์š”์ฆ˜์—๋Š” ๋จธ์‹ ๋Ÿฌ๋‹์ด๋‚˜ AI ๊ฐ™์€ ๊ธฐ์ˆ ๋“ค๋กœ ์ปดํ“จํ„ฐ๊ฐ€ ์Šค์Šค๋กœ ํ•™์Šตํ•˜๋Š” ๊ฒฝ์šฐ๋„ ์žˆ์ง€๋งŒ ๊ทธ ๋˜ํ•œ ๋ฏธ๋ฆฌ ์ฝ”๋”ฉ์„ ํ†ตํ•ด ๋งŒ๋“ค์–ด์ง„ ๋ถ€๋ถ„๋“ค. ๊ทธ๋ ‡๊ธฐ ๋•Œ๋ฌธ์— 2๋…„์ด ์ง€๋‚œ ์ง€๊ธˆ ์ด์ œ๊นŒ์ง€ ์„œ๋น„์Šค๊ฐ€ ์–ด๋–ป๊ฒŒ ๋Œ์•„๊ฐ”๋Š”์ง€๋ฅผ ํ™•์ธํ•˜๊ธฐ ์œ„ํ•ด์„œ๋Š” ์‚ฌ์ „์— ์ค€๋น„ํ•ด์•ผ ํ•  ๊ฒƒ์ด ์žˆ๋‹ค. ๊ทธ๊ฒƒ์€ ๋ฐ”๋กœ ‘๋กœ๊น…’. ์„œ๋น„์Šค ํˆฌ์ž… ์ „๋ถ€ํ„ฐ ํ”„๋ก ํŠธ๋ถ€ํ„ฐ ๋ฐฑ์—”๋“œ๊นŒ์ง€ ๋‹ค์–‘ํ•œ ๋กœ๊น…์„ ํ•ด์„œ์ธ์ง€ 2๋…„์ด ์ง€๋‚œ ์ง€๊ธˆ, ๊ธฐ๋ก๋œ ๋กœ๊ทธ๋กœ ๋‹ค์–‘ํ•œ ์„œ๋น„์Šค ์ง€ํ‘œ๋ฅผ ํ™•์ธํ•ด ๋ณผ ์ˆ˜ ์žˆ์Œ์— ๋‹คํ–‰์ด๋ผ ์ƒ๊ฐํ•œ๋‹ค.

๊ฐ์ข… ์ง€ํ‘œ

ใ€€๋จผ์ € ๋ด์•ผ ํ•  ์ง€ํ‘œ๋Š” ๋‹น์—ฐํžˆ ๊ฐ€์ž…/ํ•ด์ง€ ์ถ”์ด. ๋“œ๋ผ๋งˆํ‹ฑ ํ•œ ์„ ํ˜• ๊ทธ๋ž˜ํ”„๋Š” ์•„๋‹ˆ์ง€๋งŒ ๋‹น์—ฐํžˆ(?) ํ•ด์ง€ ๋ณด๋‹ค ๊ฐ€์ž…์ด ๋” ๋งŽ๊ณ  ์‹œ๊ฐ„์ด ์ง€๋‚ ์ˆ˜๋ก ์–ด๋А ์ •๋„ ๊พธ์ค€ํ•˜๊ฒŒ ๊ฐ€์ž…์ž๊ฐ€ ๋“ค์–ด์˜ค๋Š” ๊ฒƒ์„ ๋ณด๋ฉด ์–ด๋–ป๊ฒŒ ์•Œ๊ณ  ๊ฐ€์ž…์„ ํ•˜๋Ÿฌ ์˜ค๋Š”์ง€ ์‹ ๊ธฐํ•  ๋”ฐ๋ฆ„์ด๋‹ค. ํ•˜์ง€๋งŒ ๋งˆ๋ƒฅ ์‹ ๊ธฐํ•ดํ•˜์ง€๋งŒ ๋ง๊ณ  ํ•ด์ง€ํ•˜๋Š” ์›์ธ์„ ๋ถ„์„ํ•ด์•ผ ํ•  ํ•„์š”๊ฐ€ ์žˆ์–ด ๋ณด์ธ๋‹ค. ์•„๋งˆ๋„ ์ˆ˜์ง‘ํ•˜๋Š” ๋ธ”๋กœ๊ทธ๋“ค ์ค‘ ๊ฐ„ํ˜น ๊ฐœ๋ฐœ๊ณผ ๊ด€๋ จ๋˜์ง€ ์•Š๋Š” ๊ธ€๋“ค์ด ์ข…์ข… ์ˆ˜์ง‘๋˜์–ด์„œ ๊ทธ๋Ÿฐ ๊ฒƒ ๊ฐ™๊ธฐ๋„ ํ•˜๋‹ค.

/images/toy-projects-second-year-review/regist.jpg
๊ฐ€์ž…/ํ•ด์ง€ ํŠธ๋žœ๋“œ

ใ€€๋‹ค์Œ์œผ๋กœ๋Š” ํด๋ฆญ์ˆ˜. ๋ˆˆ์น˜๊ฐ€ ๋น ๋ฅธ ๋ถ„๋“ค์€ ์ด๋ฏธ ์•Œ๊ณ  ์žˆ๊ฒ ์ง€๋งŒ ์ด๋ฉ”์ผ์—์„œ ํด๋ฆญ ์‹œ ์„œ๋ฒ„์—์„œ ๊ฐ์ข… ๋กœ๊น…์„ ํ•˜๊ณ  ๋„˜์–ด๊ฐ€๊ฒŒ ๋œ๋‹ค. ๊ทธ๋Ÿฌ๋‹ค ๋ณด๋‹ˆ ํด๋ฆญ ์„ฑํ–ฅ(?)์— ๋Œ€ํ•ด ์ง‘๊ณ„๋„ ๊ฐ€๋Šฅํ•œ๋ฐ ์•„๋ž˜ ์ง€ํ‘œ๋ฅผ ๋ณด๋ฉด ์˜ค์ „ ์ผ๊ณผ๋ฅผ ์‹œ์ž‘ํ•˜๋ฉด์„œ ๋ฉ”์ผ๋กœ ์ข…ํ•ฉ๋œ ๊ธฐ์ˆ  ๋ธ”๋กœ๊ทธ ๋“ค์„ ์ฝ๊ธฐ ์‹œ์ž‘ํ•˜๊ณ  ๊ทธ์ค‘์—์„œ ํŠนํžˆ ์›”์š”์ผ - 10์‹œ๊ฐ€ ๊ฐ€์žฅ ๋งŽ์€ ํด๋ฆญ์ˆ˜๊ฐ€ ์ง‘๊ณ„๋˜์—ˆ๋‹ค.

/images/toy-projects-second-year-review/click.jpg
ํด๋ฆญ์ˆ˜ ํŠธ๋žœ๋“œ | ์‹œ๊ฐ„+์š”์ผ ๋ณ„ ํด๋ฆญ์ˆ˜ ํŠธ๋žœ๋“œ | ์‹œ๊ฐ„+์š”์ผ ๋ณ„ ํด๋ฆญ์ˆ˜ ํžˆํŠธ๋งต

ใ€€์ด ํฌ์ŠคํŒ…์„ ์ž‘์„ฑํ•˜๊ณ  ์žˆ๋Š” ์ง€๊ธˆ๊นŒ์ง€ ์•ฝ 19,000์—ฌ ๊ฐœ์˜ ํฌ์ŠคํŒ…์„ ์ˆ˜์ง‘ํ•˜๊ณ  ๋ฐœํ–‰ํ•˜์˜€๋Š”๋ฐ ๊ทธ์ค‘์—์„œ ๊ฐ€์žฅ ์ธ๊ธฐ ์žˆ์—ˆ๋˜ ํฌ์ŠคํŒ… TOP 30 ์€ ๋‹ค์Œ๊ณผ ๊ฐ™๋‹ค. ์•„๋ฌด๋ž˜๋„ ๋‹จ์ฒด ๋ธ”๋กœ๊ทธ์˜ ํฌ์ŠคํŒ…์„ ๋ฉ”์ผ ์ƒ๋‹จ์— ์œ„์น˜ํ•˜๊ณ  ๋…ธ๋ž€์ƒ‰์œผ๋กœ ํ…Œ๋‘๋ฆฌ๋ฅผ ํ‘œ์‹œํ•ด์„œ์ธ์ง€ ๋Œ€๋ถ€๋ถ„์˜ ๊ธ€๋“ค์ด ๋‹จ์ฒด ๋ธ”๋กœ๊ทธ์˜ ํฌ์ŠคํŒ…์ธ ๊ฒƒ์„ ์•Œ ์ˆ˜ ์žˆ๋‹ค.

๊ทธ๋ ‡๋‹ค๋ฉด ์ˆ˜์ต์€ ์–ผ๋งˆ๋‚˜ ์žˆ์—ˆ์„๊นŒ? ์Œ?! ์ด ์„œ๋น„์Šค์˜ ์ˆ˜์ต์ด ์žˆ๋‹ค๊ณ ?? ์šฐ์„  ์•„๋ž˜ ์ฐจํŠธ๋ฅผ ๋ณด์ž.