create-express-esm 1.1.4 โ†’ 1.1.8

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -1,171 +1,171 @@
1
- # ๐Ÿš€ Create Express ESM (CLI)
2
-
3
- > **Modern Express Generator**
4
- >
5
- > "1์ดˆ ๋งŒ์— ์™„์„ฑํ•˜๋Š” Modern Express(ESM) ํ™˜๊ฒฝ"
6
- >
7
- > CommonJS(require)๊ฐ€ ์•„๋‹Œ, ์ตœ์‹  ES Modules(import/export) ๋ฌธ๋ฒ•์„ ๊ธฐ๋ฐ˜์œผ๋กœ ํ•˜๋Š” Express ํ”„๋กœ์ ํŠธ ๊ตฌ์กฐ๋ฅผ ์ž๋™์œผ๋กœ ์ƒ์„ฑํ•ด์ฃผ๋Š” CLI ๋„๊ตฌ์ž…๋‹ˆ๋‹ค.
8
-
9
- [![npm version](https://img.shields.io/npm/v/create-express-esm.svg)](https://www.npmjs.com/package/create-express-esm)
10
- [![License: MIT](https://img.shields.io/badge/License-MIT-yellow.svg)](https://opensource.org/licenses/MIT)
11
-
12
- ## โœจ Demo
13
-
14
- ![Image](https://github.com/user-attachments/assets/10a7c289-ba5b-42a0-bfb8-6319610fef78)
15
-
16
- ## โœจ Key Features (ํ•ต์‹ฌ ๊ธฐ๋Šฅ)
17
-
18
- ๊ธฐ์กด `express-generator`์˜ ํ•œ๊ณ„๋ฅผ ๋ถ„์„ํ•˜๊ณ  ๊ฐœ์„ ํ•˜์—ฌ ๊ฐœ๋ฐœํ–ˆ์Šต๋‹ˆ๋‹ค.
19
-
20
- - **โšก๏ธ 100% ES Modules**: ๊ตฌ์‹ CommonJS(`require`)๋ฅผ ๋ฒ„๋ฆฌ๊ณ  ์ตœ์‹  `import/export` ๋ฌธ๋ฒ•์„ ๊ธฐ๋ณธ์œผ๋กœ ์ฑ„ํƒํ–ˆ์Šต๋‹ˆ๋‹ค.
21
- - **๐Ÿ— Layered Architecture**: ์‹ค๋ฌด ํ‘œ์ค€์ธ `Controller` - `Service` - `Model` ๊ตฌ์กฐ๋ฅผ ์ž๋™์œผ๋กœ ์žก์•„์ค๋‹ˆ๋‹ค.
22
- - **๐Ÿ“ฆ Auto Installation**: ํ”„๋กœ์ ํŠธ ์ƒ์„ฑ ํ›„ ๊ท€์ฐฎ์€ `npm install` ๊ณผ์ •์„ ์ž๋™์œผ๋กœ ์ˆ˜ํ–‰ํ•ฉ๋‹ˆ๋‹ค.
23
- - **๐Ÿ›  Ready-to-Use**: `dotenv`, `cors`, `morgan`, `nodemon` ๋“ฑ ํ•„์ˆ˜ ๊ฐœ๋ฐœ ํ™˜๊ฒฝ์ด ์„ธํŒ…๋˜์–ด ์žˆ์Šต๋‹ˆ๋‹ค.
24
-
25
- ## ๐Ÿš€ Quick Start (์‚ฌ์šฉ๋ฒ•)
26
-
27
- ๋ณ„๋„์˜ ์„ค์น˜ ์—†์ด `npx` ๋ช…๋ น์–ด๋กœ ์ฆ‰์‹œ ์‹คํ–‰ํ•  ์ˆ˜ ์žˆ์Šต๋‹ˆ๋‹ค.
28
-
29
- ```bash
30
- npx create-express-esm
31
- npm create express-esm
32
- ```
33
-
34
- ๋˜๋Š” ์ „์—ญ์œผ๋กœ ์„ค์น˜ํ•˜์—ฌ ์‚ฌ์šฉํ•  ์ˆ˜๋„ ์žˆ์Šต๋‹ˆ๋‹ค
35
-
36
- ```
37
- npm install -g create-express-esm
38
- create-express-esm
39
- ```
40
-
41
- ## ๐Ÿ“‚ Project Structure (ํด๋” ๊ตฌ์กฐ)
42
-
43
- ์ด ๋„๊ตฌ๋Š” **Layered Architecture (๊ณ„์ธตํ˜• ์•„ํ‚คํ…์ฒ˜)**๋ฅผ ๊ธฐ๋ฐ˜์œผ๋กœ ํ”„๋กœ์ ํŠธ๋ฅผ ์ƒ์„ฑํ•ฉ๋‹ˆ๋‹ค.
44
- **๊ด€์‹ฌ์‚ฌ ๋ถ„๋ฆฌ(Separation of Concerns)** ์›์น™์„ ์ ์šฉํ•˜์—ฌ, ๋กœ์ง์ด ์„ž์ด์ง€ ์•Š๊ณ  ์œ ์ง€๋ณด์ˆ˜๊ฐ€ ์‰ฌ์šด ๊ตฌ์กฐ๋ฅผ ์ œ๊ณตํ•ฉ๋‹ˆ๋‹ค.
45
-
46
- ```text
47
- my-app/
48
- โ”œโ”€โ”€ src/
49
- โ”‚ ย  โ”œโ”€โ”€ config/ ย  ย  ย  ย  ย # โš™๏ธ ํ™˜๊ฒฝ๋ณ€์ˆ˜ ๋ฐ DB ์—ฐ๊ฒฐ ์„ค์ •
50
- โ”‚ ย  โ”œโ”€โ”€ controllers/ ย  ย  # ๐Ÿ•น๏ธ ์š”์ฒญ๊ณผ ์‘๋‹ต ์ฒ˜๋ฆฌ (Controller Layer)
51
- โ”‚ ย  โ”œโ”€โ”€ models/ ย  ย  ย  ย  ย # ๐Ÿ—„๏ธ ๋ฐ์ดํ„ฐ๋ฒ ์ด์Šค ์Šคํ‚ค๋งˆ (Data Access Layer)
52
- โ”‚ ย  โ”œโ”€โ”€ routes/ ย  ย  ย  ย  ย # ๐Ÿšฆ API ๋ผ์šฐํŒ… ์ •์˜ (Route Definitions)
53
- โ”‚ ย  โ”œโ”€โ”€ services/ ย  ย  ย  ย # ๐Ÿง  ๋น„์ฆˆ๋‹ˆ์Šค ๋กœ์ง (Service Layer) - ํ•ต์‹ฌ ๋กœ์ง!
54
- โ”‚ ย  โ”œโ”€โ”€ app.js ย  ย  ย  ย  ย  # ๐Ÿ—๏ธ Express App ์„ค์ • (Middleware, CORS ๋“ฑ)
55
- โ”‚ ย  โ””โ”€โ”€ server.js ย  ย  ย  ย # ๐Ÿš€ ์„œ๋ฒ„ ์‹คํ–‰ ์ง„์ž…์  (Entry Point)
56
- โ”œโ”€โ”€ .env ย  ย  ย  ย  ย  ย  ย  ย  # ๐Ÿ” ํ™˜๊ฒฝ ๋ณ€์ˆ˜ (Port, DB Key ๋“ฑ)
57
- โ”œโ”€โ”€ .gitignore ย  ย  ย  ย  ย  # ๐Ÿ™ˆ Git ๋ฌด์‹œ ์„ค์ •
58
- โ””โ”€โ”€ package.json ย  ย  ย  ย  # ๐Ÿ“ฆ ํ”„๋กœ์ ํŠธ ์˜์กด์„ฑ ๋ฐ ์Šคํฌ๋ฆฝํŠธ
59
- ```
60
-
61
- ## ๐Ÿ›  Tech Stack (๊ธฐ์ˆ  ์Šคํƒ)
62
-
63
- - **Runtime**: Node.js
64
- - **Framework**: Express.js
65
- - **Architecture**: Layered Pattern (Controller-Service-Model)
66
- - **Language**: JavaScript (ES6+ Modules)
67
- - **Tools**:
68
- - `dotenv` (ํ™˜๊ฒฝ๋ณ€์ˆ˜ ๊ด€๋ฆฌ)
69
- - `cors` (Cross-Origin ๋ฆฌ์†Œ์Šค ๊ณต์œ  ์„ค์ •)
70
- - `morgan` (HTTP ์š”์ฒญ ๋กœ๊ทธ ๊ธฐ๋ก)
71
- - `nodemon` (๊ฐœ๋ฐœ ์ƒ์‚ฐ์„ฑ ํ–ฅ์ƒ/์ž๋™ ์žฌ์‹œ์ž‘)
72
-
73
- ## ๐Ÿ› ๏ธ Engineering Deep Dive
74
-
75
- ๋‹จ์ˆœํ•œ ๋„๊ตฌ ๊ฐœ๋ฐœ์„ ๋„˜์–ด, Node.js์˜ ๋Ÿฐํƒ€์ž„ ํ™˜๊ฒฝ๊ณผ ๋ชจ๋“ˆ ์‹œ์Šคํ…œ์˜ ์ฐจ์ด๋ฅผ ๊นŠ์ด ์žˆ๊ฒŒ ์ดํ•ดํ•˜๊ธฐ ์œ„ํ•ด ์ง„ํ–‰ํ•œ ํ”„๋กœ์ ํŠธ์ž…๋‹ˆ๋‹ค.
76
-
77
- ### - Project Motivation (Why)
78
-
79
- ๋Œ€๋ถ€๋ถ„์˜ Express ์˜ˆ์ œ๋‚˜ ์Šค์บํด๋”ฉ ๋„๊ตฌ๋“ค์ด ์—ฌ์ „ํžˆ CommonJS(require) ๋ฐฉ์‹์„ ์‚ฌ์šฉํ•˜๊ณ  ์žˆ์Šต๋‹ˆ๋‹ค. ํ•˜์ง€๋งŒ ์ตœ์‹  Node.js ์ƒํƒœ๊ณ„๋Š” ES Modules(import)๋กœ ์ „ํ™˜๋˜๊ณ  ์žˆ์Šต๋‹ˆ๋‹ค.
80
- **"์ตœ์‹  ๋ฌธ๋ฒ•์„ ์ง€์›ํ•˜๋Š” ํ™˜๊ฒฝ์„ ๋งค๋ฒˆ ์ˆ˜๋™์œผ๋กœ ์„ค์ •ํ•˜๋Š” ๋น„ํšจ์œจ์„ ํ•ด๊ฒฐํ•˜์ž"**๋Š” ๋ชฉํ‘œ๋กœ ์‹œ์ž‘ํ–ˆ์œผ๋ฉฐ, ํ”„๋ ˆ์ž„์›Œํฌ ์—†์ด ์ˆœ์ˆ˜ Node.js์˜ `fs`, `path` ๋ชจ๋“ˆ์„ ๋‹ค๋ฃจ๋ฉฐ CLI ๋„๊ตฌ์˜ ๋™์ž‘ ์›๋ฆฌ๋ฅผ ์ฒด๋“ํ•˜๊ณ ์ž ํ–ˆ์Šต๋‹ˆ๋‹ค.
81
-
82
- ### - Architecture
83
-
84
- 1. **๋น„๋™๊ธฐ ํŒŒ์ผ ์‹œ์Šคํ…œ ์ œ์–ด**: ๋Œ€๋Ÿ‰์˜ ํ…œํ”Œ๋ฆฟ ํŒŒ์ผ์„ ์ƒ์„ฑํ•  ๋•Œ I/O ๋ธ”๋กœํ‚น์„ ๋ง‰๊ธฐ ์œ„ํ•ด `fs.promises`์™€ `async/await`๋ฅผ ์‚ฌ์šฉํ•˜์—ฌ ์•ˆ์ •์ ์ธ ํŒŒ์ผ ์“ฐ๊ธฐ ์ž‘์—…์„ ๊ตฌํ˜„ํ–ˆ์Šต๋‹ˆ๋‹ค.
85
- 2. **๋™์  ํ…œํ”Œ๋ฆฟ ์ƒ์„ฑ**: ์‚ฌ์šฉ์ž๊ฐ€ ์ž…๋ ฅํ•œ ํ”„๋กœ์ ํŠธ ์ด๋ฆ„์„ `package.json` ๋“ฑ์— ๋™์ ์œผ๋กœ ์ฃผ์ž…ํ•˜์—ฌ, ์ƒ์„ฑ ์ฆ‰์‹œ ์‹คํ–‰ ๊ฐ€๋Šฅํ•œ ์ƒํƒœ๋ฅผ ๋ณด์žฅํ•ฉ๋‹ˆ๋‹ค.
86
-
87
- ## ๐Ÿ”ฅ Challenges & Troubleshooting
88
-
89
- ๊ฐœ๋ฐœ ๊ณผ์ •์—์„œ ๋งˆ์ฃผ์นœ ๊ธฐ์ˆ ์  ๋‚œ๊ด€๊ณผ ํ•ด๊ฒฐ ๊ณผ์ •์ž…๋‹ˆ๋‹ค.
90
-
91
- ### 1. ESM ํ™˜๊ฒฝ์—์„œ์˜ ๊ฒฝ๋กœ ์‹œ์Šคํ…œ ๊ตฌ์ถ• (Transition to ESM)
92
-
93
- > **๐Ÿ”ด Problem:** `ReferenceError: __dirname is not defined`
94
-
95
- ํ”„๋กœ์ ํŠธ๋ฅผ `type: "module"`(ESM)๋กœ ์„ค์ •ํ•œ ํ›„, ํ…œํ”Œ๋ฆฟ ํด๋” ๊ฒฝ๋กœ๋ฅผ ์ฐธ์กฐํ•˜๊ธฐ ์œ„ํ•ด ๊ด€์Šต์ ์œผ๋กœ `__dirname`์„ ์‚ฌ์šฉํ–ˆ์œผ๋‚˜ ์—๋Ÿฌ๊ฐ€ ๋ฐœ์ƒํ•˜๋ฉฐ ์•ฑ์ด ์ข…๋ฃŒ๋˜์—ˆ์Šต๋‹ˆ๋‹ค.
96
-
97
- **๐Ÿ” Root Cause & Solution**
98
-
99
- Node.js์˜ CommonJS ํ™˜๊ฒฝ์—์„œ๋Š” `__dirname`์ด ๊ธฐ๋ณธ์ ์œผ๋กœ ์ฃผ์ž…๋˜์ง€๋งŒ, ESM ํ‘œ์ค€ ์ŠคํŽ™์—๋Š” ์ด ๋ณ€์ˆ˜๊ฐ€ ์กด์žฌํ•˜์ง€ ์•Š์Šต๋‹ˆ๋‹ค. ์ด๋ฅผ ํ•ด๊ฒฐํ•˜๊ธฐ ์œ„ํ•ด `url`๊ณผ `path` ๋ชจ๋“ˆ์„ ์กฐํ•ฉํ•˜์—ฌ Polyfill(์ง์ ‘ ๊ตฌํ˜„) ํ–ˆ์Šต๋‹ˆ๋‹ค.
100
-
101
- ```javascript
102
- import path from "path";
103
- import { fileURLToPath } from "url";
104
-
105
- // 1. ํ˜„์žฌ ํŒŒ์ผ์˜ URL(file://...)์„ ์‹œ์Šคํ…œ ๊ฒฝ๋กœ๋กœ ๋ณ€ํ™˜
106
- const __filename = fileURLToPath(import.meta.url);
107
-
108
- // 2. ํŒŒ์ผ ๊ฒฝ๋กœ์—์„œ ๋””๋ ‰ํ† ๋ฆฌ ๊ฒฝ๋กœ๋งŒ ์ถ”์ถœํ•˜์—ฌ __dirname ๊ตฌํ˜„
109
- const __dirname = path.dirname(__filename);
110
- ```
111
-
112
- ### 2. CLI์˜ ์‹คํ–‰ ์œ„์น˜์™€ ํŒŒ์ผ ์œ„์น˜์˜ ํ˜ผ๋™ (Execution Context)
113
-
114
- > **๐Ÿ”ด Problem:** `ENOENT: no such file or directory`
115
-
116
- ๋กœ์ปฌ ํ…Œ์ŠคํŠธ(`node bin/cli.js`) ์‹œ์—๋Š” ์ •์ƒ ์ž‘๋™ํ–ˆ์œผ๋‚˜, `npm link` ํ›„ ๋‹ค๋ฅธ ๊ฒฝ๋กœ(๋ฐ”ํƒ•ํ™”๋ฉด ๋“ฑ)์—์„œ ์‹คํ–‰ํ–ˆ์„ ๋•Œ ํ…œํ”Œ๋ฆฟ ํด๋”๋ฅผ ์ฐพ์ง€ ๋ชปํ•˜๋Š” ๋ฌธ์ œ๊ฐ€ ๋ฐœ์ƒํ–ˆ์Šต๋‹ˆ๋‹ค.
117
-
118
- **๐Ÿ” Root Cause**
119
-
120
- CLI ๋„๊ตฌ ๊ฐœ๋ฐœ ์‹œ **'์ฝ”๋“œ์˜ ์œ„์น˜(Source)'**์™€ **'๋ช…๋ น์–ด ์‹คํ–‰ ์œ„์น˜(Target)'**๊ฐ€ ๋‹ค๋ฅผ ์ˆ˜ ์žˆ์Œ์„ ๊ฐ„๊ณผํ–ˆ์Šต๋‹ˆ๋‹ค. ํ…œํ”Œ๋ฆฟ์„ ์ฐพ์„ ๋•Œ `process.cwd()`(์‚ฌ์šฉ์ž ์œ„์น˜)๋ฅผ ๊ธฐ์ค€์œผ๋กœ ํƒ์ƒ‰ํ–ˆ๊ธฐ ๋•Œ๋ฌธ์— ๋ฐœ์ƒํ•œ ๋ฌธ์ œ์˜€์Šต๋‹ˆ๋‹ค.
121
-
122
- **โœ… Solution**
123
-
124
- ๋ฆฌ์†Œ์Šค์˜ ์„ฑ๊ฒฉ์— ๋”ฐ๋ผ ๊ธฐ์ค€ ๊ฒฝ๋กœ๋ฅผ ๋ช…ํ™•ํžˆ ๋ถ„๋ฆฌํ•˜์—ฌ ํ•ด๊ฒฐํ–ˆ์Šต๋‹ˆ๋‹ค.
125
-
126
- - **Source (Template)**: ์ฝ”๋“œ๊ฐ€ ์„ค์น˜๋œ ๊ณณ์— ํ•ญ์ƒ ํ•จ๊ป˜ ์กด์žฌํ•˜๋ฏ€๋กœ `__dirname` ๊ธฐ์ค€.
127
- - **Target (Project)**: ์‚ฌ์šฉ์ž๊ฐ€ ๋ช…๋ น์–ด๋ฅผ ์‹คํ–‰ํ•œ ์œ„์น˜์— ์ƒ์„ฑ๋˜์–ด์•ผ ํ•˜๋ฏ€๋กœ `process.cwd()` ๊ธฐ์ค€.
128
-
129
- ```javascript
130
- // [Source] ํ…œํ”Œ๋ฆฟ ๊ฒฝ๋กœ: ๋‚ด ์ฝ”๋“œ๊ฐ€ ์„ค์น˜๋œ ๊ณณ ๊ธฐ์ค€
131
- const templateDir = path.join(__dirname, "../template");
132
-
133
- // [Target] ์ƒ์„ฑ ๊ฒฝ๋กœ: ์‚ฌ์šฉ์ž๊ฐ€ ๋ช…๋ น์–ด๋ฅผ ์‹คํ–‰ํ•œ ๊ณณ ๊ธฐ์ค€
134
- const targetDir = path.join(process.cwd(), projectName);
135
-
136
- // Copy: Source -> Target
137
- await copyDir(templateDir, targetDir);
138
- ```
139
-
140
- ### 3. ๋ฐฐํฌ ํŒŒ์ดํ”„๋ผ์ธ ์ตœ์ ํ™” (Appropriate Technology)
141
-
142
- > **๐Ÿ”ด Problem**
143
-
144
- ์žฆ์€ ์—…๋ฐ์ดํŠธ ๊ณผ์ •์—์„œ `๋ฒ„์ „ ์ˆ˜์ •` -> `ํƒœ๊ทธ ์ƒ์„ฑ` -> `๊นƒ ํ‘ธ์‹œ` -> `npm ๋ฐฐํฌ`๋ผ๋Š” 4๋‹จ๊ณ„ ํ”„๋กœ์„ธ์Šค๋ฅผ ์ˆ˜๋™์œผ๋กœ ๋ฐ˜๋ณตํ•˜๋‹ค ๋ณด๋‹ˆ, ์ˆœ์„œ๋ฅผ ๋ˆ„๋ฝํ•˜๊ฑฐ๋‚˜ ๋ฒ„์ „์„ ์ž˜๋ชป ๊ธฐ์ž…ํ•˜๋Š” ํœด๋จผ ์—๋Ÿฌ๊ฐ€ ๋ฐœ์ƒํ–ˆ์Šต๋‹ˆ๋‹ค.
145
-
146
- **๐Ÿ” Approach & Solution**
147
-
148
- GitHub Actions์™€ ๊ฐ™์€ CI/CD ๋„๊ตฌ ๋„์ž…์„ ๊ณ ๋ คํ–ˆ์œผ๋‚˜, 1์ธ ๊ฐœ๋ฐœ ํ”„๋กœ์ ํŠธ ๊ทœ๋ชจ์— ๋น„ํ•ด ์„ค์ • ๋น„์šฉ์ด ํฌ๊ณ  ์˜ค๋ฒ„์—”์ง€๋‹ˆ์–ด๋ง์ด๋ผ๋Š” ํŒ๋‹จ์ด ๋“ค์—ˆ์Šต๋‹ˆ๋‹ค. ๋Œ€์‹  Node.js์˜ ๋‚ด์žฅ ๊ธฐ๋Šฅ์ธ **NPM Scripts**๋ฅผ ํ™œ์šฉํ•˜๋Š” ๊ฒƒ์ด ๊ฐ€์žฅ ํšจ์œจ์ ์ธ **'์ ์ • ๊ธฐ์ˆ (Appropriate Technology)'**์ด๋ผ ํŒ๋‹จํ–ˆ์Šต๋‹ˆ๋‹ค.
149
-
150
- ```json
151
- // package.json
152
- "scripts": {
153
- // patch ๋ฒ„์ „ ์—… -> ๊นƒ ํƒœ๊ทธ ํ‘ธ์‹œ -> npm ๋ฐฐํฌ๋ฅผ ๋ช…๋ น์–ด ํ•œ ์ค„๋กœ ์›์ž์ (Atomic) ์‹คํ–‰
154
- "deploy": "npm version patch && git push origin main --tags && npm publish"
155
- }
156
- ```
157
-
158
- ## ๐Ÿ“ Retrospective
159
-
160
- - **Legacy to Modern**: Node.js ์ƒํƒœ๊ณ„๊ฐ€ CJS์—์„œ ESM์œผ๋กœ ์ „ํ™˜๋˜๋Š” ๊ณผ๋„๊ธฐ์—์„œ ๋ฐœ์ƒํ•˜๋Š” ํ˜ธํ™˜์„ฑ ๋ฌธ์ œ๋ฅผ ์ง์ ‘ ๊ฒช๊ณ  ํ•ด๊ฒฐํ•˜๋ฉฐ, ๋ชจ๋˜ ์ž๋ฐ”์Šคํฌ๋ฆฝํŠธ ํ™˜๊ฒฝ์— ๋Œ€ํ•œ ์ดํ•ด๋„๋ฅผ ๋†’์˜€์Šต๋‹ˆ๋‹ค.
161
- - **Consumer to Producer**: ํ•ญ์ƒ ๋ผ์ด๋ธŒ๋Ÿฌ๋ฆฌ๋ฅผ ์‚ฌ์šฉํ•˜๊ธฐ๋งŒ ํ•˜๋˜ ์ž…์žฅ์—์„œ, ์ง์ ‘ ๋„๊ตฌ๋ฅผ ๋งŒ๋“ค์–ด ๋ฐฐํฌํ•˜๋Š” ์ƒ์‚ฐ์ž๊ฐ€ ๋˜์–ด๋ด„์œผ๋กœ์จ ์˜คํ”ˆ์†Œ์Šค ์ƒํƒœ๊ณ„์˜ ์ˆœํ™˜ ๊ตฌ์กฐ๋ฅผ ์ฒด๊ฐํ–ˆ์Šต๋‹ˆ๋‹ค.
162
- - **JavaScript to TypeScript (Next Step)**: ์ˆœ์ˆ˜ ESM ํ™˜๊ฒฝ์„ ๊ตฌ์ถ•ํ•˜๋ฉฐ ๋ชจ๋“ˆ ์‹œ์Šคํ…œ์€ ํ‘œ์ค€ํ™”ํ–ˆ์œผ๋‚˜, ์ปดํŒŒ์ผ ๋‹จ๊ณ„์—์„œ ์˜ค๋ฅ˜๋ฅผ ์žก์„ ์ˆ˜ ์—†๋Š” ๋™์  ํƒ€์ž… ์–ธ์–ด์˜ ํ•œ๊ณ„๋ฅผ ์ฒด๊ฐํ–ˆ์Šต๋‹ˆ๋‹ค. ํ”„๋กœ์ ํŠธ์˜ ์•ˆ์ •์„ฑ๊ณผ ์œ ์ง€๋ณด์ˆ˜์„ฑ์„ ๋†’์ด๊ธฐ ์œ„ํ•ด **TypeScript** ๋„์ž…๊ณผ ์—„๊ฒฉํ•œ ํƒ€์ž… ์‹œ์Šคํ…œ์˜ ํ•„์š”์„ฑ์„ ๊นจ๋‹ฌ์•˜์œผ๋ฉฐ, ์ฐจ๊ธฐ ๋ฒ„์ „์—์„œ๋Š” ์ด๋ฅผ ์ง€์›ํ•  ๊ณ„ํš์ž…๋‹ˆ๋‹ค.
163
-
164
- ## ๐Ÿ—บ๏ธ Roadmap (Future Plans)
165
-
166
- - [ ] **TypeScript Support**: `tsconfig.json` ์ž๋™ ์„ค์ • ๋ฐ `.ts` ํ…œํ”Œ๋ฆฟ ์ง€์›
167
- - [ ] **Test Environment**: Jest/Supertest ์„ค์ • ์ž๋™ํ™”
168
-
169
- ## ๐Ÿ“ License
170
-
171
- This project is MIT licensed.
1
+ # ๐Ÿš€ Create Express ESM (CLI)
2
+
3
+ > **Modern Express Generator**
4
+ >
5
+ > "1์ดˆ ๋งŒ์— ์™„์„ฑํ•˜๋Š” Modern Express(ESM) ํ™˜๊ฒฝ"
6
+ >
7
+ > CommonJS(require)๊ฐ€ ์•„๋‹Œ, ์ตœ์‹  ES Modules(import/export) ๋ฌธ๋ฒ•์„ ๊ธฐ๋ฐ˜์œผ๋กœ ํ•˜๋Š” Express ํ”„๋กœ์ ํŠธ ๊ตฌ์กฐ๋ฅผ ์ž๋™์œผ๋กœ ์ƒ์„ฑํ•ด์ฃผ๋Š” CLI ๋„๊ตฌ์ž…๋‹ˆ๋‹ค.
8
+
9
+ [![npm version](https://img.shields.io/npm/v/create-express-esm.svg)](https://www.npmjs.com/package/create-express-esm)
10
+ [![License: MIT](https://img.shields.io/badge/License-MIT-yellow.svg)](https://opensource.org/licenses/MIT)
11
+
12
+ ## โœจ Demo
13
+
14
+ ![Image](https://github.com/user-attachments/assets/10a7c289-ba5b-42a0-bfb8-6319610fef78)
15
+
16
+ ## โœจ Key Features (ํ•ต์‹ฌ ๊ธฐ๋Šฅ)
17
+
18
+ ๊ธฐ์กด `express-generator`์˜ ํ•œ๊ณ„๋ฅผ ๋ถ„์„ํ•˜๊ณ  ๊ฐœ์„ ํ•˜์—ฌ ๊ฐœ๋ฐœํ–ˆ์Šต๋‹ˆ๋‹ค.
19
+
20
+ - **โšก๏ธ 100% ES Modules**: ๊ตฌ์‹ CommonJS(`require`)๋ฅผ ๋ฒ„๋ฆฌ๊ณ  ์ตœ์‹  `import/export` ๋ฌธ๋ฒ•์„ ๊ธฐ๋ณธ์œผ๋กœ ์ฑ„ํƒํ–ˆ์Šต๋‹ˆ๋‹ค.
21
+ - **๐Ÿ— Layered Architecture**: ์‹ค๋ฌด ํ‘œ์ค€์ธ `Controller` - `Service` - `Model` ๊ตฌ์กฐ๋ฅผ ์ž๋™์œผ๋กœ ์žก์•„์ค๋‹ˆ๋‹ค.
22
+ - **๐Ÿ“ฆ Auto Installation**: ํ”„๋กœ์ ํŠธ ์ƒ์„ฑ ํ›„ ๊ท€์ฐฎ์€ `npm install` ๊ณผ์ •์„ ์ž๋™์œผ๋กœ ์ˆ˜ํ–‰ํ•ฉ๋‹ˆ๋‹ค.
23
+ - **๐Ÿ›  Ready-to-Use**: `dotenv`, `cors`, `morgan`, `nodemon` ๋“ฑ ํ•„์ˆ˜ ๊ฐœ๋ฐœ ํ™˜๊ฒฝ์ด ์„ธํŒ…๋˜์–ด ์žˆ์Šต๋‹ˆ๋‹ค.
24
+
25
+ ## ๐Ÿš€ Quick Start (์‚ฌ์šฉ๋ฒ•)
26
+
27
+ ๋ณ„๋„์˜ ์„ค์น˜ ์—†์ด `npx` ๋ช…๋ น์–ด๋กœ ์ฆ‰์‹œ ์‹คํ–‰ํ•  ์ˆ˜ ์žˆ์Šต๋‹ˆ๋‹ค.
28
+
29
+ ```bash
30
+ npx create-express-esm
31
+ npm create express-esm
32
+ ```
33
+
34
+ ๋˜๋Š” ์ „์—ญ์œผ๋กœ ์„ค์น˜ํ•˜์—ฌ ์‚ฌ์šฉํ•  ์ˆ˜๋„ ์žˆ์Šต๋‹ˆ๋‹ค
35
+
36
+ ```
37
+ npm install -g create-express-esm
38
+ create-express-esm
39
+ ```
40
+
41
+ ## ๐Ÿ“‚ Project Structure (ํด๋” ๊ตฌ์กฐ)
42
+
43
+ ์ด ๋„๊ตฌ๋Š” **Layered Architecture (๊ณ„์ธตํ˜• ์•„ํ‚คํ…์ฒ˜)**๋ฅผ ๊ธฐ๋ฐ˜์œผ๋กœ ํ”„๋กœ์ ํŠธ๋ฅผ ์ƒ์„ฑํ•ฉ๋‹ˆ๋‹ค.
44
+ **๊ด€์‹ฌ์‚ฌ ๋ถ„๋ฆฌ(Separation of Concerns)** ์›์น™์„ ์ ์šฉํ•˜์—ฌ, ๋กœ์ง์ด ์„ž์ด์ง€ ์•Š๊ณ  ์œ ์ง€๋ณด์ˆ˜๊ฐ€ ์‰ฌ์šด ๊ตฌ์กฐ๋ฅผ ์ œ๊ณตํ•ฉ๋‹ˆ๋‹ค.
45
+
46
+ ```text
47
+ my-app/
48
+ โ”œโ”€โ”€ src/
49
+ โ”‚ ย  โ”œโ”€โ”€ config/ ย  ย  ย  ย  ย # โš™๏ธ ํ™˜๊ฒฝ๋ณ€์ˆ˜ ๋ฐ DB ์—ฐ๊ฒฐ ์„ค์ •
50
+ โ”‚ ย  โ”œโ”€โ”€ controllers/ ย  ย  # ๐Ÿ•น๏ธ ์š”์ฒญ๊ณผ ์‘๋‹ต ์ฒ˜๋ฆฌ (Controller Layer)
51
+ โ”‚ ย  โ”œโ”€โ”€ models/ ย  ย  ย  ย  ย # ๐Ÿ—„๏ธ ๋ฐ์ดํ„ฐ๋ฒ ์ด์Šค ์Šคํ‚ค๋งˆ (Data Access Layer)
52
+ โ”‚ ย  โ”œโ”€โ”€ routes/ ย  ย  ย  ย  ย # ๐Ÿšฆ API ๋ผ์šฐํŒ… ์ •์˜ (Route Definitions)
53
+ โ”‚ ย  โ”œโ”€โ”€ services/ ย  ย  ย  ย # ๐Ÿง  ๋น„์ฆˆ๋‹ˆ์Šค ๋กœ์ง (Service Layer) - ํ•ต์‹ฌ ๋กœ์ง!
54
+ โ”‚ ย  โ”œโ”€โ”€ app.js ย  ย  ย  ย  ย  # ๐Ÿ—๏ธ Express App ์„ค์ • (Middleware, CORS ๋“ฑ)
55
+ โ”‚ ย  โ””โ”€โ”€ server.js ย  ย  ย  ย # ๐Ÿš€ ์„œ๋ฒ„ ์‹คํ–‰ ์ง„์ž…์  (Entry Point)
56
+ โ”œโ”€โ”€ .env ย  ย  ย  ย  ย  ย  ย  ย  # ๐Ÿ” ํ™˜๊ฒฝ ๋ณ€์ˆ˜ (Port, DB Key ๋“ฑ)
57
+ โ”œโ”€โ”€ .gitignore ย  ย  ย  ย  ย  # ๐Ÿ™ˆ Git ๋ฌด์‹œ ์„ค์ •
58
+ โ””โ”€โ”€ package.json ย  ย  ย  ย  # ๐Ÿ“ฆ ํ”„๋กœ์ ํŠธ ์˜์กด์„ฑ ๋ฐ ์Šคํฌ๋ฆฝํŠธ
59
+ ```
60
+
61
+ ## ๐Ÿ›  Tech Stack (๊ธฐ์ˆ  ์Šคํƒ)
62
+
63
+ - **Runtime**: Node.js
64
+ - **Framework**: Express.js
65
+ - **Architecture**: Layered Pattern (Controller-Service-Model)
66
+ - **Language**: JavaScript (ES6+ Modules)
67
+ - **Tools**:
68
+ - `dotenv` (ํ™˜๊ฒฝ๋ณ€์ˆ˜ ๊ด€๋ฆฌ)
69
+ - `cors` (Cross-Origin ๋ฆฌ์†Œ์Šค ๊ณต์œ  ์„ค์ •)
70
+ - `morgan` (HTTP ์š”์ฒญ ๋กœ๊ทธ ๊ธฐ๋ก)
71
+ - `nodemon` (๊ฐœ๋ฐœ ์ƒ์‚ฐ์„ฑ ํ–ฅ์ƒ/์ž๋™ ์žฌ์‹œ์ž‘)
72
+
73
+ ## ๐Ÿ› ๏ธ Engineering Deep Dive
74
+
75
+ ๋‹จ์ˆœํ•œ ๋„๊ตฌ ๊ฐœ๋ฐœ์„ ๋„˜์–ด, Node.js์˜ ๋Ÿฐํƒ€์ž„ ํ™˜๊ฒฝ๊ณผ ๋ชจ๋“ˆ ์‹œ์Šคํ…œ์˜ ์ฐจ์ด๋ฅผ ๊นŠ์ด ์žˆ๊ฒŒ ์ดํ•ดํ•˜๊ธฐ ์œ„ํ•ด ์ง„ํ–‰ํ•œ ํ”„๋กœ์ ํŠธ์ž…๋‹ˆ๋‹ค.
76
+
77
+ ### - Project Motivation (Why)
78
+
79
+ ๋Œ€๋ถ€๋ถ„์˜ Express ์˜ˆ์ œ๋‚˜ ์Šค์บํด๋”ฉ ๋„๊ตฌ๋“ค์ด ์—ฌ์ „ํžˆ CommonJS(require) ๋ฐฉ์‹์„ ์‚ฌ์šฉํ•˜๊ณ  ์žˆ์Šต๋‹ˆ๋‹ค. ํ•˜์ง€๋งŒ ์ตœ์‹  Node.js ์ƒํƒœ๊ณ„๋Š” ES Modules(import)๋กœ ์ „ํ™˜๋˜๊ณ  ์žˆ์Šต๋‹ˆ๋‹ค.
80
+ **"์ตœ์‹  ๋ฌธ๋ฒ•์„ ์ง€์›ํ•˜๋Š” ํ™˜๊ฒฝ์„ ๋งค๋ฒˆ ์ˆ˜๋™์œผ๋กœ ์„ค์ •ํ•˜๋Š” ๋น„ํšจ์œจ์„ ํ•ด๊ฒฐํ•˜์ž"**๋Š” ๋ชฉํ‘œ๋กœ ์‹œ์ž‘ํ–ˆ์œผ๋ฉฐ, ํ”„๋ ˆ์ž„์›Œํฌ ์—†์ด ์ˆœ์ˆ˜ Node.js์˜ `fs`, `path` ๋ชจ๋“ˆ์„ ๋‹ค๋ฃจ๋ฉฐ CLI ๋„๊ตฌ์˜ ๋™์ž‘ ์›๋ฆฌ๋ฅผ ์ฒด๋“ํ•˜๊ณ ์ž ํ–ˆ์Šต๋‹ˆ๋‹ค.
81
+
82
+ ### - Architecture
83
+
84
+ 1. **๋น„๋™๊ธฐ ํŒŒ์ผ ์‹œ์Šคํ…œ ์ œ์–ด**: ๋Œ€๋Ÿ‰์˜ ํ…œํ”Œ๋ฆฟ ํŒŒ์ผ์„ ์ƒ์„ฑํ•  ๋•Œ I/O ๋ธ”๋กœํ‚น์„ ๋ง‰๊ธฐ ์œ„ํ•ด `fs.promises`์™€ `async/await`๋ฅผ ์‚ฌ์šฉํ•˜์—ฌ ์•ˆ์ •์ ์ธ ํŒŒ์ผ ์“ฐ๊ธฐ ์ž‘์—…์„ ๊ตฌํ˜„ํ–ˆ์Šต๋‹ˆ๋‹ค.
85
+ 2. **๋™์  ํ…œํ”Œ๋ฆฟ ์ƒ์„ฑ**: ์‚ฌ์šฉ์ž๊ฐ€ ์ž…๋ ฅํ•œ ํ”„๋กœ์ ํŠธ ์ด๋ฆ„์„ `package.json` ๋“ฑ์— ๋™์ ์œผ๋กœ ์ฃผ์ž…ํ•˜์—ฌ, ์ƒ์„ฑ ์ฆ‰์‹œ ์‹คํ–‰ ๊ฐ€๋Šฅํ•œ ์ƒํƒœ๋ฅผ ๋ณด์žฅํ•ฉ๋‹ˆ๋‹ค.
86
+
87
+ ## ๐Ÿ”ฅ Challenges & Troubleshooting
88
+
89
+ ๊ฐœ๋ฐœ ๊ณผ์ •์—์„œ ๋งˆ์ฃผ์นœ ๊ธฐ์ˆ ์  ๋‚œ๊ด€๊ณผ ํ•ด๊ฒฐ ๊ณผ์ •์ž…๋‹ˆ๋‹ค.
90
+
91
+ ### 1. ESM ํ™˜๊ฒฝ์—์„œ์˜ ๊ฒฝ๋กœ ์‹œ์Šคํ…œ ๊ตฌ์ถ• (Transition to ESM)
92
+
93
+ > **๐Ÿ”ด Problem:** `ReferenceError: __dirname is not defined`
94
+
95
+ ํ”„๋กœ์ ํŠธ๋ฅผ `type: "module"`(ESM)๋กœ ์„ค์ •ํ•œ ํ›„, ํ…œํ”Œ๋ฆฟ ํด๋” ๊ฒฝ๋กœ๋ฅผ ์ฐธ์กฐํ•˜๊ธฐ ์œ„ํ•ด ๊ด€์Šต์ ์œผ๋กœ `__dirname`์„ ์‚ฌ์šฉํ–ˆ์œผ๋‚˜ ์—๋Ÿฌ๊ฐ€ ๋ฐœ์ƒํ•˜๋ฉฐ ์•ฑ์ด ์ข…๋ฃŒ๋˜์—ˆ์Šต๋‹ˆ๋‹ค.
96
+
97
+ **๐Ÿ” Root Cause & Solution**
98
+
99
+ Node.js์˜ CommonJS ํ™˜๊ฒฝ์—์„œ๋Š” `__dirname`์ด ๊ธฐ๋ณธ์ ์œผ๋กœ ์ฃผ์ž…๋˜์ง€๋งŒ, ESM ํ‘œ์ค€ ์ŠคํŽ™์—๋Š” ์ด ๋ณ€์ˆ˜๊ฐ€ ์กด์žฌํ•˜์ง€ ์•Š์Šต๋‹ˆ๋‹ค. ์ด๋ฅผ ํ•ด๊ฒฐํ•˜๊ธฐ ์œ„ํ•ด `url`๊ณผ `path` ๋ชจ๋“ˆ์„ ์กฐํ•ฉํ•˜์—ฌ Polyfill(์ง์ ‘ ๊ตฌํ˜„) ํ–ˆ์Šต๋‹ˆ๋‹ค.
100
+
101
+ ```javascript
102
+ import path from "path";
103
+ import { fileURLToPath } from "url";
104
+
105
+ // 1. ํ˜„์žฌ ํŒŒ์ผ์˜ URL(file://...)์„ ์‹œ์Šคํ…œ ๊ฒฝ๋กœ๋กœ ๋ณ€ํ™˜
106
+ const __filename = fileURLToPath(import.meta.url);
107
+
108
+ // 2. ํŒŒ์ผ ๊ฒฝ๋กœ์—์„œ ๋””๋ ‰ํ† ๋ฆฌ ๊ฒฝ๋กœ๋งŒ ์ถ”์ถœํ•˜์—ฌ __dirname ๊ตฌํ˜„
109
+ const __dirname = path.dirname(__filename);
110
+ ```
111
+
112
+ ### 2. CLI์˜ ์‹คํ–‰ ์œ„์น˜์™€ ํŒŒ์ผ ์œ„์น˜์˜ ํ˜ผ๋™ (Execution Context)
113
+
114
+ > **๐Ÿ”ด Problem:** `ENOENT: no such file or directory`
115
+
116
+ ๋กœ์ปฌ ํ…Œ์ŠคํŠธ(`node bin/cli.js`) ์‹œ์—๋Š” ์ •์ƒ ์ž‘๋™ํ–ˆ์œผ๋‚˜, `npm link` ํ›„ ๋‹ค๋ฅธ ๊ฒฝ๋กœ(๋ฐ”ํƒ•ํ™”๋ฉด ๋“ฑ)์—์„œ ์‹คํ–‰ํ–ˆ์„ ๋•Œ ํ…œํ”Œ๋ฆฟ ํด๋”๋ฅผ ์ฐพ์ง€ ๋ชปํ•˜๋Š” ๋ฌธ์ œ๊ฐ€ ๋ฐœ์ƒํ–ˆ์Šต๋‹ˆ๋‹ค.
117
+
118
+ **๐Ÿ” Root Cause**
119
+
120
+ CLI ๋„๊ตฌ ๊ฐœ๋ฐœ ์‹œ **'์ฝ”๋“œ์˜ ์œ„์น˜(Source)'**์™€ **'๋ช…๋ น์–ด ์‹คํ–‰ ์œ„์น˜(Target)'**๊ฐ€ ๋‹ค๋ฅผ ์ˆ˜ ์žˆ์Œ์„ ๊ฐ„๊ณผํ–ˆ์Šต๋‹ˆ๋‹ค. ํ…œํ”Œ๋ฆฟ์„ ์ฐพ์„ ๋•Œ `process.cwd()`(์‚ฌ์šฉ์ž ์œ„์น˜)๋ฅผ ๊ธฐ์ค€์œผ๋กœ ํƒ์ƒ‰ํ–ˆ๊ธฐ ๋•Œ๋ฌธ์— ๋ฐœ์ƒํ•œ ๋ฌธ์ œ์˜€์Šต๋‹ˆ๋‹ค.
121
+
122
+ **โœ… Solution**
123
+
124
+ ๋ฆฌ์†Œ์Šค์˜ ์„ฑ๊ฒฉ์— ๋”ฐ๋ผ ๊ธฐ์ค€ ๊ฒฝ๋กœ๋ฅผ ๋ช…ํ™•ํžˆ ๋ถ„๋ฆฌํ•˜์—ฌ ํ•ด๊ฒฐํ–ˆ์Šต๋‹ˆ๋‹ค.
125
+
126
+ - **Source (Template)**: ์ฝ”๋“œ๊ฐ€ ์„ค์น˜๋œ ๊ณณ์— ํ•ญ์ƒ ํ•จ๊ป˜ ์กด์žฌํ•˜๋ฏ€๋กœ `__dirname` ๊ธฐ์ค€.
127
+ - **Target (Project)**: ์‚ฌ์šฉ์ž๊ฐ€ ๋ช…๋ น์–ด๋ฅผ ์‹คํ–‰ํ•œ ์œ„์น˜์— ์ƒ์„ฑ๋˜์–ด์•ผ ํ•˜๋ฏ€๋กœ `process.cwd()` ๊ธฐ์ค€.
128
+
129
+ ```javascript
130
+ // [Source] ํ…œํ”Œ๋ฆฟ ๊ฒฝ๋กœ: ๋‚ด ์ฝ”๋“œ๊ฐ€ ์„ค์น˜๋œ ๊ณณ ๊ธฐ์ค€
131
+ const templateDir = path.join(__dirname, "../template");
132
+
133
+ // [Target] ์ƒ์„ฑ ๊ฒฝ๋กœ: ์‚ฌ์šฉ์ž๊ฐ€ ๋ช…๋ น์–ด๋ฅผ ์‹คํ–‰ํ•œ ๊ณณ ๊ธฐ์ค€
134
+ const targetDir = path.join(process.cwd(), projectName);
135
+
136
+ // Copy: Source -> Target
137
+ await copyDir(templateDir, targetDir);
138
+ ```
139
+
140
+ ### 3. ๋ฐฐํฌ ํŒŒ์ดํ”„๋ผ์ธ ์ตœ์ ํ™” (Appropriate Technology)
141
+
142
+ > **๐Ÿ”ด Problem**
143
+
144
+ ์žฆ์€ ์—…๋ฐ์ดํŠธ ๊ณผ์ •์—์„œ `๋ฒ„์ „ ์ˆ˜์ •` -> `ํƒœ๊ทธ ์ƒ์„ฑ` -> `๊นƒ ํ‘ธ์‹œ` -> `npm ๋ฐฐํฌ`๋ผ๋Š” 4๋‹จ๊ณ„ ํ”„๋กœ์„ธ์Šค๋ฅผ ์ˆ˜๋™์œผ๋กœ ๋ฐ˜๋ณตํ•˜๋‹ค ๋ณด๋‹ˆ, ์ˆœ์„œ๋ฅผ ๋ˆ„๋ฝํ•˜๊ฑฐ๋‚˜ ๋ฒ„์ „์„ ์ž˜๋ชป ๊ธฐ์ž…ํ•˜๋Š” ํœด๋จผ ์—๋Ÿฌ๊ฐ€ ๋ฐœ์ƒํ–ˆ์Šต๋‹ˆ๋‹ค.
145
+
146
+ **๐Ÿ” Approach & Solution**
147
+
148
+ GitHub Actions์™€ ๊ฐ™์€ CI/CD ๋„๊ตฌ ๋„์ž…์„ ๊ณ ๋ คํ–ˆ์œผ๋‚˜, 1์ธ ๊ฐœ๋ฐœ ํ”„๋กœ์ ํŠธ ๊ทœ๋ชจ์— ๋น„ํ•ด ์„ค์ • ๋น„์šฉ์ด ํฌ๊ณ  ์˜ค๋ฒ„์—”์ง€๋‹ˆ์–ด๋ง์ด๋ผ๋Š” ํŒ๋‹จ์ด ๋“ค์—ˆ์Šต๋‹ˆ๋‹ค. ๋Œ€์‹  Node.js์˜ ๋‚ด์žฅ ๊ธฐ๋Šฅ์ธ **NPM Scripts**๋ฅผ ํ™œ์šฉํ•˜๋Š” ๊ฒƒ์ด ๊ฐ€์žฅ ํšจ์œจ์ ์ธ **'์ ์ • ๊ธฐ์ˆ (Appropriate Technology)'**์ด๋ผ ํŒ๋‹จํ–ˆ์Šต๋‹ˆ๋‹ค.
149
+
150
+ ```json
151
+ // package.json
152
+ "scripts": {
153
+ // patch ๋ฒ„์ „ ์—… -> ๊นƒ ํƒœ๊ทธ ํ‘ธ์‹œ -> npm ๋ฐฐํฌ๋ฅผ ๋ช…๋ น์–ด ํ•œ ์ค„๋กœ ์›์ž์ (Atomic) ์‹คํ–‰
154
+ "deploy": "npm version patch && git push origin main --tags && npm publish"
155
+ }
156
+ ```
157
+
158
+ ## ๐Ÿ“ Retrospective
159
+
160
+ - **Legacy to Modern**: Node.js ์ƒํƒœ๊ณ„๊ฐ€ CJS์—์„œ ESM์œผ๋กœ ์ „ํ™˜๋˜๋Š” ๊ณผ๋„๊ธฐ์—์„œ ๋ฐœ์ƒํ•˜๋Š” ํ˜ธํ™˜์„ฑ ๋ฌธ์ œ๋ฅผ ์ง์ ‘ ๊ฒช๊ณ  ํ•ด๊ฒฐํ•˜๋ฉฐ, ๋ชจ๋˜ ์ž๋ฐ”์Šคํฌ๋ฆฝํŠธ ํ™˜๊ฒฝ์— ๋Œ€ํ•œ ์ดํ•ด๋„๋ฅผ ๋†’์˜€์Šต๋‹ˆ๋‹ค.
161
+ - **Consumer to Producer**: ํ•ญ์ƒ ๋ผ์ด๋ธŒ๋Ÿฌ๋ฆฌ๋ฅผ ์‚ฌ์šฉํ•˜๊ธฐ๋งŒ ํ•˜๋˜ ์ž…์žฅ์—์„œ, ์ง์ ‘ ๋„๊ตฌ๋ฅผ ๋งŒ๋“ค์–ด ๋ฐฐํฌํ•˜๋Š” ์ƒ์‚ฐ์ž๊ฐ€ ๋˜์–ด๋ด„์œผ๋กœ์จ ์˜คํ”ˆ์†Œ์Šค ์ƒํƒœ๊ณ„์˜ ์ˆœํ™˜ ๊ตฌ์กฐ๋ฅผ ์ฒด๊ฐํ–ˆ์Šต๋‹ˆ๋‹ค.
162
+ - **JavaScript to TypeScript (Next Step)**: ์ˆœ์ˆ˜ ESM ํ™˜๊ฒฝ์„ ๊ตฌ์ถ•ํ•˜๋ฉฐ ๋ชจ๋“ˆ ์‹œ์Šคํ…œ์€ ํ‘œ์ค€ํ™”ํ–ˆ์œผ๋‚˜, ์ปดํŒŒ์ผ ๋‹จ๊ณ„์—์„œ ์˜ค๋ฅ˜๋ฅผ ์žก์„ ์ˆ˜ ์—†๋Š” ๋™์  ํƒ€์ž… ์–ธ์–ด์˜ ํ•œ๊ณ„๋ฅผ ์ฒด๊ฐํ–ˆ์Šต๋‹ˆ๋‹ค. ํ”„๋กœ์ ํŠธ์˜ ์•ˆ์ •์„ฑ๊ณผ ์œ ์ง€๋ณด์ˆ˜์„ฑ์„ ๋†’์ด๊ธฐ ์œ„ํ•ด **TypeScript** ๋„์ž…๊ณผ ์—„๊ฒฉํ•œ ํƒ€์ž… ์‹œ์Šคํ…œ์˜ ํ•„์š”์„ฑ์„ ๊นจ๋‹ฌ์•˜์œผ๋ฉฐ, ์ฐจ๊ธฐ ๋ฒ„์ „์—์„œ๋Š” ์ด๋ฅผ ์ง€์›ํ•  ๊ณ„ํš์ž…๋‹ˆ๋‹ค.
163
+
164
+ ## ๐Ÿ—บ๏ธ Roadmap (Future Plans)
165
+
166
+ - [ ] **TypeScript Support**: `tsconfig.json` ์ž๋™ ์„ค์ • ๋ฐ `.ts` ํ…œํ”Œ๋ฆฟ ์ง€์›
167
+ - [ ] **Test Environment**: Jest/Supertest ์„ค์ • ์ž๋™ํ™”
168
+
169
+ ## ๐Ÿ“ License
170
+
171
+ This project is MIT licensed.
package/bin/cli.js CHANGED
@@ -1,110 +1,164 @@
1
1
  #!/usr/bin/env node
2
-
3
- import { input, select } from '@inquirer/prompts'; // ํ˜„๋Œ€์ ์ธ ๋ฐฉ์‹์œผ๋กœ ๊ต์ฒด
4
- import fs from 'fs-extra';
5
- import path from 'path';
6
- import chalk from 'chalk';
7
- import { execSync } from 'child_process';
8
- import { fileURLToPath } from 'url';
9
-
10
- // ESM์—์„œ __dirname ์‚ฌ์šฉํ•˜๊ธฐ ์œ„ํ•œ ์„ค์ •
11
- const __filename = fileURLToPath(import.meta.url);
12
- const __dirname = path.dirname(__filename);
13
-
14
- // 1.2.0 ๋ฒ„์ „ ์ •๋ณด ๋ฐ ๋ฉ”์ธ ๋กœ์ง
15
- async function run() {
16
- console.log(chalk.blue.bold('\n๐Ÿš€ Create Express ESM ์‹œ์ž‘!\n'));
17
-
18
- try {
19
- // 1. ์‚ฌ์šฉ์ž ์งˆ๋ฌธ (๋น„๋™๊ธฐ ํ•จ์ˆ˜ ๋ฐฉ์‹์œผ๋กœ ๋ณ€๊ฒฝ)
20
- const projectName = await input({
21
- message: '์ƒ์„ฑํ•  ํ”„๋กœ์ ํŠธ ์ด๋ฆ„์„ ์ž…๋ ฅํ•˜์„ธ์š”:',
22
- default: 'my-app',
23
- });
24
-
25
- const language = await select({
26
- message: '์‚ฌ์šฉํ•  ์–ธ์–ด๋ฅผ ์„ ํƒํ•˜์„ธ์š”:',
27
- choices: [
28
- { name: 'JavaScript (ESM)', value: 'js' },
29
- { name: 'TypeScript', value: 'ts' },
30
- ],
31
- });
32
-
33
- const targetPath = path.join(process.cwd(), projectName);
34
- const templatePath = path.join(__dirname, '../template', language);
35
-
36
- // 2. ํด๋” ์กด์žฌ ์—ฌ๋ถ€ ํ™•์ธ
37
- if (fs.existsSync(targetPath)) {
38
- console.error(chalk.red(`\nโŒ ์˜ค๋ฅ˜: '${projectName}' ํด๋”๊ฐ€ ์ด๋ฏธ ์กด์žฌํ•ฉ๋‹ˆ๋‹ค.`));
39
- process.exit(1);
40
- }
41
-
42
- // 3. ํ…œํ”Œ๋ฆฟ ๋ณต์‚ฌ
43
- console.log(chalk.cyan(`\n๐Ÿ“‚ [${language.toUpperCase()}] ํ…œํ”Œ๋ฆฟ์„ ๋ณต์‚ฌํ•˜๋Š” ์ค‘...`));
44
-
45
- if (!fs.existsSync(templatePath)) {
46
- console.error(chalk.red(`\nโŒ ์˜ค๋ฅ˜: ${language} ํ…œํ”Œ๋ฆฟ ํด๋”๋ฅผ ์ฐพ์„ ์ˆ˜ ์—†์Šต๋‹ˆ๋‹ค.`));
47
- console.log(chalk.gray(`๊ฒฝ๋กœ ํ™•์ธ: ${templatePath}`));
48
- process.exit(1);
49
- }
50
-
51
- await fs.copy(templatePath, targetPath);
52
-
53
- // 4. ๋„ํŠธ ํŒŒ์ผ ๋ณ€ํ™˜ ๋ฐ ํ™˜๊ฒฝ ์„ค์ •
54
- const renameMap = {
55
- 'gitignore': '.gitignore',
56
- '_gitignore': '.gitignore',
57
- '_env': '.env'
58
- };
59
-
60
- for (const [oldName, newName] of Object.entries(renameMap)) {
61
- const oldFilePath = path.join(targetPath, oldName);
62
- const newFilePath = path.join(targetPath, newName);
63
-
64
- if (await fs.pathExists(oldFilePath)) {
65
- await fs.move(oldFilePath, newFilePath, { overwrite: true });
66
- if (newName === '.env') {
67
- const exampleEnvPath = path.join(targetPath, '.env.example');
68
- await fs.copy(newFilePath, exampleEnvPath);
69
- }
70
- }
71
- }
72
-
73
- // 5. package.json ํ”„๋กœ์ ํŠธ ์ด๋ฆ„ ์ˆ˜์ •
74
- const pkgPath = path.join(targetPath, 'package.json');
75
- if (await fs.pathExists(pkgPath)) {
76
- const pkg = await fs.readJson(pkgPath);
77
- pkg.name = projectName;
78
- await fs.writeJson(pkgPath, pkg, { spaces: 2 });
79
- }
80
-
81
- console.log(chalk.green(`โœ… ํ…œํ”Œ๋ฆฟ ๊ตฌ์„ฑ ์™„๋ฃŒ!`));
82
-
83
- // 6. ํŒจํ‚ค์ง€ ์ž๋™ ์„ค์น˜
84
- console.log(chalk.yellow(`\n๐Ÿ“ฆ ํŒจํ‚ค์ง€ ์ž๋™ ์„ค์น˜๋ฅผ ์ง„ํ–‰ํ•ฉ๋‹ˆ๋‹ค... (npm install)`));
85
-
86
- execSync('npm install', {
87
- cwd: targetPath,
88
- stdio: 'inherit'
89
- });
90
-
91
- console.log(chalk.green(`\nโœจ ๋ชจ๋“  ์„ค์น˜๊ฐ€ ์™„๋ฃŒ๋˜์—ˆ์Šต๋‹ˆ๋‹ค!`));
92
- console.log(chalk.white(`\n๋‹ค์Œ ๋ช…๋ น์–ด๋กœ ์‹œ์ž‘ํ•˜์„ธ์š”:\n`));
93
- console.log(chalk.cyan(` cd ${projectName}`));
94
- if (language === 'ts') {
95
- console.log(chalk.cyan(` npm run dev (๋˜๋Š” npm run build)`));
96
- } else {
97
- console.log(chalk.cyan(` npm run dev`));
98
- }
99
- console.log('\n');
100
-
101
- } catch (error) {
102
- if (error.name === 'ExitPromptError') {
103
- console.log(chalk.yellow('\n\n๐Ÿ‘‹ ์„ค์น˜๋ฅผ ์ค‘๋‹จํ–ˆ์Šต๋‹ˆ๋‹ค.'));
104
- } else {
105
- console.error(chalk.red('\nโŒ ์˜ค๋ฅ˜ ๋ฐœ์ƒ:'), error);
106
- }
107
- }
108
- }
109
-
2
+
3
+ import { input, select, confirm } from '@inquirer/prompts';
4
+ import fs from 'fs-extra';
5
+ import path from 'path';
6
+ import chalk from 'chalk';
7
+ import { execSync } from 'child_process';
8
+ import { fileURLToPath } from 'url';
9
+
10
+ const __filename = fileURLToPath(import.meta.url);
11
+ const __dirname = path.dirname(__filename);
12
+
13
+ async function run() {
14
+ console.log(chalk.blue.bold('\n๐Ÿš€ Create Express ESM ์‹œ์ž‘!\n'));
15
+
16
+ try {
17
+ // 1. ์‚ฌ์šฉ์ž ์งˆ๋ฌธ
18
+ const projectName = await input({
19
+ message: '์ƒ์„ฑํ•  ํ”„๋กœ์ ํŠธ ์ด๋ฆ„์„ ์ž…๋ ฅํ•˜์„ธ์š”:',
20
+ default: 'my-app',
21
+ });
22
+
23
+ const language = await select({
24
+ message: '์‚ฌ์šฉํ•  ์–ธ์–ด๋ฅผ ์„ ํƒํ•˜์„ธ์š”:',
25
+ choices: [
26
+ { name: 'JavaScript (ESM)', value: 'js' },
27
+ { name: 'TypeScript', value: 'ts' },
28
+ ],
29
+ });
30
+
31
+ const useTest = await confirm({
32
+ message: 'Vitest ํ…Œ์ŠคํŠธ ํ™˜๊ฒฝ์„ ์ถ”๊ฐ€ํ•˜์‹œ๊ฒ ์Šต๋‹ˆ๊นŒ?',
33
+ default: true,
34
+ });
35
+
36
+ const targetPath = path.join(process.cwd(), projectName);
37
+ const templatePath = path.join(__dirname, '../template', language);
38
+
39
+ // 2. ํด๋” ์กด์žฌ ์—ฌ๋ถ€ ํ™•์ธ
40
+ if (fs.existsSync(targetPath)) {
41
+ console.error(chalk.red(`\nโŒ ์˜ค๋ฅ˜: '${projectName}' ํด๋”๊ฐ€ ์ด๋ฏธ ์กด์žฌํ•ฉ๋‹ˆ๋‹ค.`));
42
+ process.exit(1);
43
+ }
44
+
45
+ // 3. ๊ธฐ๋ณธ ํ…œํ”Œ๋ฆฟ ๋ณต์‚ฌ
46
+ console.log(chalk.cyan(`\n๐Ÿ“‚ [${language.toUpperCase()}] ํ…œํ”Œ๋ฆฟ ๊ตฌ์„ฑ์„ ์‹œ์ž‘ํ•ฉ๋‹ˆ๋‹ค...`));
47
+ await fs.copy(templatePath, targetPath);
48
+
49
+ // 4. ๋„ํŠธ ํŒŒ์ผ ๋ณ€ํ™˜ (_env -> .env ๋“ฑ)
50
+ const renameMap = {
51
+ 'gitignore': '.gitignore',
52
+ '_gitignore': '.gitignore',
53
+ '_env': '.env'
54
+ };
55
+
56
+ for (const [oldName, newName] of Object.entries(renameMap)) {
57
+ const oldFilePath = path.join(targetPath, oldName);
58
+ const newFilePath = path.join(targetPath, newName);
59
+ if (await fs.pathExists(oldFilePath)) {
60
+ await fs.move(oldFilePath, newFilePath, { overwrite: true });
61
+ if (newName === '.env') {
62
+ await fs.copy(newFilePath, path.join(targetPath, '.env.example'));
63
+ }
64
+ }
65
+ }
66
+
67
+ // 5. package.json ๋™์  ์ˆ˜์ •
68
+ const pkgPath = path.join(targetPath, 'package.json');
69
+ const pkg = await fs.readJson(pkgPath);
70
+ pkg.name = projectName;
71
+
72
+ // [์ถ”๊ฐ€๋œ ๋ถ€๋ถ„] TypeScript ํ™˜๊ฒฝ์—์„œ ESM ์—๋Ÿฌ๋ฅผ ๋ฐฉ์ง€ํ•˜๊ธฐ ์œ„ํ•œ tsx ์„ค์ •
73
+ if (language === 'ts') {
74
+ console.log(chalk.yellow(`โš™๏ธ TypeScript ESM ์‹คํ–‰ ํ™˜๊ฒฝ(tsx)์„ ์ตœ์ ํ™”ํ•˜๋Š” ์ค‘...`));
75
+
76
+ // ts-node ๋Œ€์‹  tsx๋ฅผ ์‚ฌ์šฉํ•˜์—ฌ .js ํ™•์žฅ์ž ์ž„ํฌํŠธ ๋ฌธ์ œ ํ•ด๊ฒฐ
77
+ pkg.scripts.dev = "nodemon --exec tsx src/server.ts";
78
+
79
+ // ์˜์กด์„ฑ ๊ต์ฒด
80
+ pkg.devDependencies = {
81
+ ...pkg.devDependencies,
82
+ "tsx": "^4.7.0"
83
+ };
84
+
85
+ // ๊ธฐ์กด์— ts-node๊ฐ€ ์žˆ๋‹ค๋ฉด ์ œ๊ฑฐ (์ค‘๋ณต ๋ฐฉ์ง€)
86
+ delete pkg.devDependencies['ts-node'];
87
+ }
88
+
89
+ // Vitest ์„ค์ • (์ด์Šˆ #3 ๊ตฌํ˜„ ๋ถ€๋ถ„)
90
+ if (useTest) {
91
+ console.log(chalk.yellow(`๐Ÿงช Vitest ์„ค์ • ๋ฐ ์ƒ˜ํ”Œ ํ…Œ์ŠคํŠธ๋ฅผ ์ƒ์„ฑํ•˜๋Š” ์ค‘...`));
92
+
93
+ pkg.scripts = {
94
+ ...pkg.scripts,
95
+ "test": "vitest",
96
+ "test:ui": "vitest --ui",
97
+ "test:run": "vitest run"
98
+ };
99
+
100
+ const testDeps = {
101
+ "vitest": "^1.0.0",
102
+ "supertest": "^6.3.3"
103
+ };
104
+
105
+ if (language === 'ts') {
106
+ testDeps["@types/supertest"] = "^2.0.12";
107
+ }
108
+
109
+ pkg.devDependencies = {
110
+ ...pkg.devDependencies,
111
+ ...testDeps
112
+ };
113
+
114
+ // Vitest ์„ค์ • ํŒŒ์ผ ์ƒ์„ฑ
115
+ const configExt = language === 'ts' ? 'ts' : 'js';
116
+ const configContent = `import { defineConfig } from 'vitest/config';
117
+
118
+ export default defineConfig({
119
+ test: {
120
+ globals: true,
121
+ environment: 'node',
122
+ },
123
+ });`;
124
+ await fs.writeFile(path.join(targetPath, `vitest.config.${configExt}`), configContent);
125
+
126
+ // ์ƒ˜ํ”Œ ํ…Œ์ŠคํŠธ ํŒŒ์ผ ์ƒ์„ฑ
127
+ const testFileExt = language === 'ts' ? 'ts' : 'js';
128
+ const testContent = `import { describe, it, expect } from 'vitest';
129
+ import request from 'supertest';
130
+ import app from './app.js';
131
+
132
+ describe('API Health Check Test', () => {
133
+ it('GET / ์š”์ฒญ์ด ์„ฑ๊ณตํ•ด์•ผ ํ•œ๋‹ค', async () => {
134
+ const res = await request(app).get('/');
135
+ expect(res.status).toBe(200);
136
+ expect(res.text).toContain('Server is Running');
137
+ });
138
+ });`;
139
+ await fs.writeFile(path.join(targetPath, `src/app.test.${testFileExt}`), testContent);
140
+ }
141
+
142
+ await fs.writeJson(pkgPath, pkg, { spaces: 2 });
143
+ console.log(chalk.green(`โœ… ๋ชจ๋“  ๊ตฌ์„ฑ ์™„๋ฃŒ!`));
144
+
145
+ // 6. ํŒจํ‚ค์ง€ ์ž๋™ ์„ค์น˜
146
+ console.log(chalk.yellow(`\n๐Ÿ“ฆ ์˜์กด์„ฑ ํŒจํ‚ค์ง€๋ฅผ ์„ค์น˜ํ•ฉ๋‹ˆ๋‹ค... (npm install)`));
147
+ execSync('npm install', { cwd: targetPath, stdio: 'inherit' });
148
+
149
+ console.log(chalk.green(`\nโœจ ํ”„๋กœ์ ํŠธ ์ƒ์„ฑ ์„ฑ๊ณต!`));
150
+ console.log(chalk.white(`\n๋‹ค์Œ ๋ช…๋ น์–ด๋ฅผ ์ž…๋ ฅํ•ด ๋ณด์„ธ์š”:\n`));
151
+ console.log(chalk.cyan(` cd ${projectName}`));
152
+ if (useTest) console.log(chalk.cyan(` npm test`));
153
+ console.log(chalk.cyan(` npm run dev\n`));
154
+
155
+ } catch (error) {
156
+ if (error.name === 'ExitPromptError') { // ์˜คํƒ€ ์ˆ˜์ •: ExitPnromptError -> ExitPromptError
157
+ console.log(chalk.yellow('\n\n๐Ÿ‘‹ ์„ค์น˜๋ฅผ ์ค‘๋‹จํ–ˆ์Šต๋‹ˆ๋‹ค.'));
158
+ } else {
159
+ console.error(chalk.red('\nโŒ ์˜ค๋ฅ˜ ๋ฐœ์ƒ:'), error);
160
+ }
161
+ }
162
+ }
163
+
110
164
  run();
package/package.json CHANGED
@@ -1,40 +1,44 @@
1
- {
2
- "name": "create-express-esm",
3
- "version": "1.1.4",
4
- "description": "A modern CLI tool to bootstrap Express.js applications with ES Modules and Layered Architecture.",
5
- "main": "index.js",
6
- "type": "module",
7
- "bin": {
8
- "create-express-esm": "bin/cli.js"
9
- },
10
- "repository": {
11
- "type": "git",
12
- "url": "git+https://github.com/munjuin/create-express-esm.git"
13
- },
14
- "scripts": {
15
- "test": "echo \"Error: no test specified\" && exit 1",
16
- "deploy": "npm version patch && git push origin main --tags && npm publish"
17
- },
18
- "keywords": [
19
- "express",
20
- "esm",
21
- "cli",
22
- "boilerplate"
23
- ],
24
- "author": "munjuin",
25
- "license": "MIT",
26
- "dependencies": {
27
- "chalk": "^5.6.2",
28
- "commander": "^14.0.2",
29
- "fs-extra": "^11.3.2",
30
- "inquirer": "^13.0.1"
31
- },
32
- "files": [
33
- "bin",
34
- "template"
35
- ],
36
- "bugs": {
37
- "url": "https://github.com/munjuin/create-express-esm/issues"
38
- },
39
- "homepage": "https://github.com/munjuin/create-express-esm#readme"
40
- }
1
+ {
2
+ "name": "create-express-esm",
3
+ "version": "1.1.8",
4
+ "description": "A modern CLI tool to bootstrap Express.js applications with ES Modules and Layered Architecture.",
5
+ "main": "index.js",
6
+ "type": "module",
7
+ "bin": {
8
+ "create-express-esm": "bin/cli.js"
9
+ },
10
+ "repository": {
11
+ "type": "git",
12
+ "url": "git+https://github.com/munjuin/create-express-esm.git"
13
+ },
14
+ "scripts": {
15
+ "test": "echo \"Error: no test specified\" && exit 1",
16
+ "release": "npm version patch && git push origin main --tags"
17
+ },
18
+ "keywords": [
19
+ "express",
20
+ "esm",
21
+ "cli",
22
+ "boilerplate"
23
+ ],
24
+ "author": "munjuin",
25
+ "license": "MIT",
26
+ "dependencies": {
27
+ "@inquirer/prompts": "^8.1.0",
28
+ "chalk": "^5.6.2",
29
+ "commander": "^14.0.2",
30
+ "fs-extra": "^11.3.2",
31
+ "inquirer": "^13.0.1"
32
+ },
33
+ "files": [
34
+ "bin",
35
+ "template"
36
+ ],
37
+ "bugs": {
38
+ "url": "https://github.com/munjuin/create-express-esm/issues"
39
+ },
40
+ "homepage": "https://github.com/munjuin/create-express-esm#readme",
41
+ "devDependencies": {
42
+ "tsx": "^4.21.0"
43
+ }
44
+ }
package/template/js/_env CHANGED
@@ -1,2 +1,2 @@
1
- PORT=8080PORT=3000
1
+ PORT=8080PORT=3000
2
2
  NODE_ENV=development
@@ -1,18 +1,18 @@
1
- # Dependency directories
2
- node_modules/
3
-
4
- # Build output
5
- dist/
6
-
7
- # Environment variables
8
- .env
9
- .env.example
10
-
11
- # Debug logs
12
- npm-debug.log*
13
- yarn-debug.log*
14
- yarn-error.log*
15
-
16
- # OS files
17
- .DS_Store
1
+ # Dependency directories
2
+ node_modules/
3
+
4
+ # Build output
5
+ dist/
6
+
7
+ # Environment variables
8
+ .env
9
+ .env.example
10
+
11
+ # Debug logs
12
+ npm-debug.log*
13
+ yarn-debug.log*
14
+ yarn-error.log*
15
+
16
+ # OS files
17
+ .DS_Store
18
18
  Thumbs.db
@@ -1,19 +1,19 @@
1
- {
2
- "name": "my-express-app",
3
- "version": "1.0.0",
4
- "type": "module",
5
- "scripts": {
6
- "start": "node src/server.js",
7
- "dev": "nodemon src/server.js"
8
- },
9
- "dependencies": {
10
- "express": "^4.18.2",
11
- "cors": "^2.8.5",
12
- "dotenv": "^16.3.1",
13
- "morgan": "^1.10.0",
14
- "helmet": "^7.1.0"
15
- },
16
- "devDependencies": {
17
- "nodemon": "^3.0.2"
18
- }
1
+ {
2
+ "name": "my-express-app",
3
+ "version": "1.0.0",
4
+ "type": "module",
5
+ "scripts": {
6
+ "start": "node src/server.js",
7
+ "dev": "nodemon src/server.js"
8
+ },
9
+ "dependencies": {
10
+ "express": "^4.18.2",
11
+ "cors": "^2.8.5",
12
+ "dotenv": "^16.3.1",
13
+ "morgan": "^1.10.0",
14
+ "helmet": "^7.1.0"
15
+ },
16
+ "devDependencies": {
17
+ "nodemon": "^3.0.2"
18
+ }
19
19
  }
@@ -1,19 +1,19 @@
1
- import express from 'express';
2
- import cors from 'cors';
3
- import morgan from 'morgan';
4
- import dotenv from 'dotenv';
5
-
6
- import userRoutes from './routes/userRoutes.js';
7
-
8
- const app = express();
9
-
10
- app.use(cors());
11
- app.use(morgan('dev'));
12
- app.use(express.json());
13
-
14
- // ๋ผ์šฐํ„ฐ ์—ฐ๊ฒฐ (Layered Arch ์‹œ์ž‘์ )
15
- app.use('/api/users', userRoutes);
16
-
17
- app.get('/', (req, res) => res.send('Welcome to Modern Express!'));
18
-
1
+ import express from 'express';
2
+ import cors from 'cors';
3
+ import morgan from 'morgan';
4
+ import dotenv from 'dotenv';
5
+
6
+ import userRoutes from './routes/userRoutes.js';
7
+
8
+ const app = express();
9
+
10
+ app.use(cors());
11
+ app.use(morgan('dev'));
12
+ app.use(express.json());
13
+
14
+ // ๋ผ์šฐํ„ฐ ์—ฐ๊ฒฐ (Layered Arch ์‹œ์ž‘์ )
15
+ app.use('/api/users', userRoutes);
16
+
17
+ app.get('/', (req, res) => res.send('Welcome to Modern Express!'));
18
+
19
19
  export default app;
@@ -1,10 +1,10 @@
1
- import * as userService from '../services/userService.js';
2
-
3
- export const getUsers = async (req, res) => {
4
- try {
5
- const users = await userService.findAllUsers(); // ์„œ๋น„์Šค ํ˜ธ์ถœ
6
- res.json(users);
7
- } catch (error) {
8
- res.status(500).json({ message: error.message });
9
- }
1
+ import * as userService from '../services/userService.js';
2
+
3
+ export const getUsers = async (req, res) => {
4
+ try {
5
+ const users = await userService.findAllUsers(); // ์„œ๋น„์Šค ํ˜ธ์ถœ
6
+ res.json(users);
7
+ } catch (error) {
8
+ res.status(500).json({ message: error.message });
9
+ }
10
10
  };
@@ -1,8 +1,8 @@
1
- import express from 'express';
2
- import * as userController from '../controllers/userController.js';
3
-
4
- const router = express.Router();
5
-
6
- router.get('/', userController.getUsers);
7
-
1
+ import express from 'express';
2
+ import * as userController from '../controllers/userController.js';
3
+
4
+ const router = express.Router();
5
+
6
+ router.get('/', userController.getUsers);
7
+
8
8
  export default router;
@@ -1,9 +1,9 @@
1
- import app from './app.js';
2
- import dotenv from 'dotenv';
3
-
4
- dotenv.config();
5
- const PORT = process.env.PORT || 3000;
6
-
7
- app.listen(PORT, () => {
8
- console.log(`๐Ÿš€ Server running on http://localhost:${PORT}`);
1
+ import app from './app.js';
2
+ import dotenv from 'dotenv';
3
+
4
+ dotenv.config();
5
+ const PORT = process.env.PORT || 3000;
6
+
7
+ app.listen(PORT, () => {
8
+ console.log(`๐Ÿš€ Server running on http://localhost:${PORT}`);
9
9
  });
@@ -1,8 +1,8 @@
1
- // ๋‚˜์ค‘์—๋Š” ์—ฌ๊ธฐ์„œ Model(DB)์„ ํ˜ธ์ถœํ•ฉ๋‹ˆ๋‹ค.
2
- export const findAllUsers = async () => {
3
- // ๋น„์ฆˆ๋‹ˆ์Šค ๋กœ์ง ์ฒ˜๋ฆฌ...
4
- return [
5
- { id: 1, name: "Developer Lee", role: "Fullstack" },
6
- { id: 2, name: "Genius AI", role: "Assistant" }
7
- ];
1
+ // ๋‚˜์ค‘์—๋Š” ์—ฌ๊ธฐ์„œ Model(DB)์„ ํ˜ธ์ถœํ•ฉ๋‹ˆ๋‹ค.
2
+ export const findAllUsers = async () => {
3
+ // ๋น„์ฆˆ๋‹ˆ์Šค ๋กœ์ง ์ฒ˜๋ฆฌ...
4
+ return [
5
+ { id: 1, name: "Developer Lee", role: "Fullstack" },
6
+ { id: 2, name: "Genius AI", role: "Assistant" }
7
+ ];
8
8
  };
@@ -1,23 +1,23 @@
1
- {
2
- "name": "temp-name",
3
- "version": "1.0.0",
4
- "type": "module",
5
- "scripts": {
6
- "start": "node dist/server.js",
7
- "dev": "nodemon --exec ts-node src/server.ts",
8
- "build": "tsc"
9
- },
10
- "dependencies": {
11
- "express": "^4.18.2",
12
- "dotenv": "^16.3.1",
13
- "cors": "^2.8.5"
14
- },
15
- "devDependencies": {
16
- "typescript": "^5.0.0",
17
- "@types/express": "^4.17.17",
18
- "@types/node": "^20.0.0",
19
- "@types/cors": "^2.8.13",
20
- "ts-node": "^10.9.1",
21
- "nodemon": "^3.0.1"
22
- }
1
+ {
2
+ "name": "temp-name",
3
+ "version": "1.0.0",
4
+ "type": "module",
5
+ "scripts": {
6
+ "dev": "nodemon --exec tsx src/server.ts",
7
+ "build": "tsc",
8
+ "start": "node dist/server.js"
9
+ },
10
+ "dependencies": {
11
+ "express": "^4.18.2",
12
+ "dotenv": "^16.3.1",
13
+ "cors": "^2.8.5"
14
+ },
15
+ "devDependencies": {
16
+ "typescript": "^5.0.0",
17
+ "@types/express": "^4.17.17",
18
+ "@types/node": "^20.0.0",
19
+ "@types/cors": "^2.8.13",
20
+ "ts-node": "^10.9.1",
21
+ "nodemon": "^3.0.1"
22
+ }
23
23
  }
@@ -1,20 +1,20 @@
1
- import express, { Application, Request, Response } from 'express';
2
- import cors from 'cors';
3
- import userRoutes from './routes/userRoutes.js';
4
-
5
- const app: Application = express();
6
-
7
- // Middleware
8
- app.use(cors());
9
- app.use(express.json());
10
- app.use(express.urlencoded({ extended: true }));
11
-
12
- // Routes
13
- app.use('/api/users', userRoutes);
14
-
15
- // Health Check
16
- app.get('/', (req: Request, res: Response) => {
17
- res.send('Create Express ESM (TypeScript) Server is Running!');
18
- });
19
-
1
+ import express, { Application, Request, Response } from 'express';
2
+ import cors from 'cors';
3
+ import userRoutes from './routes/userRoutes.js';
4
+
5
+ const app: Application = express();
6
+
7
+ // Middleware
8
+ app.use(cors());
9
+ app.use(express.json());
10
+ app.use(express.urlencoded({ extended: true }));
11
+
12
+ // Routes
13
+ app.use('/api/users', userRoutes);
14
+
15
+ // Health Check
16
+ app.get('/', (req: Request, res: Response) => {
17
+ res.send('Create Express ESM (TypeScript) Server is Running!');
18
+ });
19
+
20
20
  export default app;
@@ -1,17 +1,17 @@
1
- import { Request, Response } from 'express';
2
- import * as userService from '../services/userService.js';
3
-
4
- export const getUsers = async (req: Request, res: Response): Promise<void> => {
5
- try {
6
- const users = await userService.fetchAllUsers();
7
- res.status(200).json({
8
- success: true,
9
- data: users,
10
- });
11
- } catch (error) {
12
- res.status(500).json({
13
- success: false,
14
- message: 'Internal Server Error',
15
- });
16
- }
1
+ import { Request, Response } from 'express';
2
+ import * as userService from '../services/userService.js';
3
+
4
+ export const getUsers = async (req: Request, res: Response): Promise<void> => {
5
+ try {
6
+ const users = await userService.fetchAllUsers();
7
+ res.status(200).json({
8
+ success: true,
9
+ data: users,
10
+ });
11
+ } catch (error) {
12
+ res.status(500).json({
13
+ success: false,
14
+ message: 'Internal Server Error',
15
+ });
16
+ }
17
17
  };
@@ -1,8 +1,8 @@
1
- import { Router } from 'express';
2
- import * as userController from '../controllers/userController.js';
3
-
4
- const router = Router();
5
-
6
- router.get('/', userController.getUsers);
7
-
1
+ import { Router } from 'express';
2
+ import * as userController from '../controllers/userController.js';
3
+
4
+ const router = Router();
5
+
6
+ router.get('/', userController.getUsers);
7
+
8
8
  export default router;
@@ -1,8 +1,8 @@
1
- import 'dotenv/config';
2
- import app from './app.js';
3
-
4
- const PORT = process.env.PORT || 3000;
5
-
6
- app.listen(PORT, () => {
7
- console.log(`๐Ÿš€ Server is running on http://localhost:${PORT}`);
1
+ import 'dotenv/config';
2
+ import app from './app.js';
3
+
4
+ const PORT = process.env.PORT || 3000;
5
+
6
+ app.listen(PORT, () => {
7
+ console.log(`๐Ÿš€ Server is running on http://localhost:${PORT}`);
8
8
  });
@@ -1,14 +1,14 @@
1
- // ์œ ์ € ๋ฐ์ดํ„ฐ ํƒ€์ž… ์ •์˜
2
- interface User {
3
- id: number;
4
- name: string;
5
- email: string;
6
- }
7
-
8
- export const fetchAllUsers = async (): Promise<User[]> => {
9
- // ์‹ค์ œ DB ์—ฐ๋™ ๋Œ€์‹  ์ƒ˜ํ”Œ ๋ฐ์ดํ„ฐ๋ฅผ ๋ฐ˜ํ™˜ํ•ฉ๋‹ˆ๋‹ค.
10
- return [
11
- { id: 1, name: 'Owner', email: 'owner@example.com' },
12
- { id: 2, name: 'Gemini', email: 'gemini@ai.com' },
13
- ];
1
+ // ์œ ์ € ๋ฐ์ดํ„ฐ ํƒ€์ž… ์ •์˜
2
+ interface User {
3
+ id: number;
4
+ name: string;
5
+ email: string;
6
+ }
7
+
8
+ export const fetchAllUsers = async (): Promise<User[]> => {
9
+ // ์‹ค์ œ DB ์—ฐ๋™ ๋Œ€์‹  ์ƒ˜ํ”Œ ๋ฐ์ดํ„ฐ๋ฅผ ๋ฐ˜ํ™˜ํ•ฉ๋‹ˆ๋‹ค.
10
+ return [
11
+ { id: 1, name: 'Owner', email: 'owner@example.com' },
12
+ { id: 2, name: 'Gemini', email: 'gemini@ai.com' },
13
+ ];
14
14
  };
@@ -1,15 +1,15 @@
1
- {
2
- "compilerOptions": {
3
- "target": "ESNext",
4
- "module": "NodeNext",
5
- "moduleResolution": "NodeNext",
6
- "outDir": "./dist",
7
- "rootDir": "./src",
8
- "strict": true,
9
- "esModuleInterop": true,
10
- "skipLibCheck": true,
11
- "forceConsistentCasingInFileNames": true
12
- },
13
- "include": ["src/**/*"],
14
- "exclude": ["node_modules"]
1
+ {
2
+ "compilerOptions": {
3
+ "target": "ESNext",
4
+ "module": "NodeNext",
5
+ "moduleResolution": "NodeNext",
6
+ "outDir": "./dist",
7
+ "rootDir": "./src",
8
+ "strict": true,
9
+ "esModuleInterop": true,
10
+ "skipLibCheck": true,
11
+ "forceConsistentCasingInFileNames": true
12
+ },
13
+ "include": ["src/**/*"],
14
+ "exclude": ["node_modules"]
15
15
  }