Post-blog-Client & Server
Client
๊ฐ๋จํ ํฌ์คํธ๋ฅผ ์์ฑ, ์ฝ๊ธฐ, ์์ , ์ญ์ ๋ฑ ํ ์ ์๋ CRUD ๋ธ๋ก๊ทธ ์น ์ฑ์ client์ ๋๋ค.
์คํ์ผ
styled-components๋ฅผ ์ฌ์ฉํ์ฌ ์คํ์ผ์ ์ ์ฉํ์๊ณ
๋ผ์ฐํฐ
react-router-dom๋ฅผ ์ฌ์ฉํ์ฌ
LoginPage, RegisterPage, WritePage, PostPage, PostListPage ์ด 5๊ฐ์ ํ์ด์ง๋ฅผ ๊ตฌํํ์์ต๋๋ค.
์๋ฒ์์ ์ ๊ณตํ๋ HTTP ํค๋์ โlast-pageโ๋ฅผ ๋ฐ์์์ PostListPage์์ ํ์ด์ง๋ค์ด์ ๊ธฐ๋ฅ์ ๊ตฌํํ์์ต๋๋ค.
์ํ๊ด๋ฆฌ
React-redux๋ฅผ ์ฌ์ฉํ์ฌ ์ํ๊ด๋ฆฌ๋ฅผ ํฉ๋๋ค
Ducks ํจํด์ผ๋ก redux๊ด๋ จ ํ์ผ์ ๋ชจ๋ modules์ ์์ฑํ์ต๋๋ค.
Container-Presenter ๋์์ธ ํจํด์ผ๋ก ๋ฐ์ดํฐ ์ฒ๋ฆฌ์ ๋ฐ์ดํฐ ์ถ๋ ฅ์ ํ๋ ํ์ผ๋ค์ ๋๋ ์ ์์ฑํ์์ต๋๋ค.
Container Components๋ containers
Presentational Components๋ components
ํด๋์์ ํ์ธํ ์ ์์ต๋๋ค.
API ์๋ฒ์ ์ฐ๋
axios๋ฅผ ์ฌ์ฉํ์ฌ API๋ฅผ ํธ์ถํฉ๋๋ค.
Redux-saga ๋ฏธ๋ค์จ์ด๋ฅผ ์ฌ์ฉํ์ฌ API ์์ฒญ๊ณผ ๊ฐ์ ๋น๋๊ธฐ์ ์์ ์ ๊ด๋ฆฌํฉ๋๋ค.
๋ฐฐํฌ
๋ฐฐํฌํ๋ ์น์๋ฒ๋ฅผ nginx๋ฅผ ์ฌ์ฉํ์ฌ ์๋ก ๊ตฌํํ์ง ์๊ณ API๋ฅผ ์ ๊ณตํ๋ post-blog-server ์๋ฒ์์ koa-static์ ์ฌ์ฉํ์ฌ ์ ์ ํ์ผ์ ์ ๊ณตํ๋๋ก ํตํฉํด์ ๊ตฌํํ์์ต๋๋ค.
๊ทธํ์ Docker๋ฅผ ์ฌ์ฉํ์ฌ ๋น๋๋ ํด๋ผ์ด์ธํธ์ ์๋ฒ๋ฅผ ์ด๋ฏธ์ง๋ก ๋ง๋ค์๊ณ
Google Cloud Run์ ๋น๋๋ ์ด๋ฏธ์ง๋ฅผ ์ฌ๋ ค์ ๋ฐฐํฌํ์์ต๋๋ค.
Young-blog ๋ฐฐํฌ๋ ์ฌ์ดํธ์ ๋๋ค.
page
ํ์๊ฐ์
โ/registerโ
๋ก๊ทธ์ธ
โ/loginโ
ํฌ์คํธ ์์ฑ
โ/writeโ
ํฌ์คํธ ๋ชฉ๋ก ์กฐํ
โ/โ ,โ/?tagโ , โ/@:usernameโ
ํฌ์คํธ ์กฐํ
โ/@:username/:postIdโ
Environments
- axios@0.21.1
- immer@9.0.1
- qs@6.10.1
- quill@1.3.7
- react@17.0.1
- react-dom@17.0.1
- react-helmet-async@1.0.9
- react-redux@7.2.2
- react-router-dom@5.2.0
- react-scripts@4.0.3
- redux@4.0.5
- redux-actions@2.6.5
- redux-devtools-extension@2.13.9
- redux-saga@1.1.3
- styled-components@5.2.1
Server
๊ฐ๋จํ ํฌ์คํธ๋ฅผ ์์ฑ, ์ฝ๊ธฐ, ์์ , ์ญ์ ๋ฑ ํ ์ ์๋ CRUD ๋ธ๋ก๊ทธ ์น ์ฑ์ server์ ๋๋ค.
REST API
Node.js์ koa ํ๋ ์์ํฌ๋ฅผ ์ฌ์ฉํ์ฌ REST API๋ฅผ ๊ตฌํํ์์ต๋๋ค.
ํฌ์คํธ ๊ด๋ฆฌ
Joi ๋ผ์ด๋ธ๋ฌ๋ฆฌ๋ฅผ ์ฌ์ฉํ์ฌ ํด๋ผ์ด์ธํธ์์ ์์ฑ๋ ํฌ์คํธ๋ฅผ ๊ฒ์ฆํ์ฌ ์ฌ๋ฐ๋ฅธ ์์ฒญ์ด๋ฉด ๋ฐ์ดํฐ๋ฒ ์ด์ค์ ๋ฐ์ดํฐ๋ฅผ ์ ์ฅํ๊ณ ๊ทธ๋ ์ง ์์ผ๋ฉด ์ค๋ฅ๋ฉ์ธ์ง๋ฅผ ๋ณด๋ด๋๋ก ๊ตฌํํ์์ต๋๋ค.
sanitize-html ๋ผ์ด๋ธ๋ฌ๋ฆฌ๋ฅผ ์ฌ์ฉํ์ฌ ํด๋ผ์ด์ธํธ์์ ๊ธ์ฐ๊ธฐ๋ฅผ ํ ๋ ์ ์ฑ์ฝ๋ ์ฝ์ ๋ฐฉ์ง๋ฅผ ์ํ์ฌ HTML์ ํํฐ๋งํ์์ต๋๋ค.
์ ์ ๊ด๋ฆฌ
jsonwebtoken ๋ผ์ด๋ธ๋ฌ๋ฆฌ๋ฅผ ์ฌ์ฉํ์ฌ JSON ํ ํฐ์ ๋ฐ๊ธํ๊ณ ์ฟ ํค์ ์ ์ ์ ๋ณด๋ฅผ ์ ์ฅํ์ฌ ํ์๊ฐ์ , ๋ก๊ทธ์ธ , ๋ก๊ทธ์์ ๊ธฐ๋ฅ์ ํ์ํ ํ์ ์ธ์ฆ ์์คํ ์ ๊ตฌํํ์์ต๋๋ค.
๋ฐ์ดํฐ ๋ฒ ์ด์ค
mongoDB Atlas๋ฅผ ์ฌ์ฉํ์ฌ ์๋ฒ์ ๋ฐ์ดํฐ๋ฒ ์ด์ค๋ฅผ ์ฐ๊ฒฐํ์์ต๋๋ค.
mongoDB Atlas๋ mongodb์์ ์ ๊ณตํ๋ ํด๋ผ์ฐ๋ ๋ฐ์ดํฐ๋ฒ ์ด์ค๋ก ํ๊ฒฝ๋ณ์์ ์ ์ฅ๋ URI๋ฅผ ํตํด ์ฐ๋ํ์ต๋๋ค.
ํ๊ฒฝ๋ณ์ ํ์ผ .env๋ ๋ณด์์ ์ํด git์ ์ฌ๋ฆฌ์ง์์์ต๋๋ค.
๋ฐฐํฌ
โป 3 / 30 ์ถ๊ฐ๋ด์ฉ
koa-static ๋ผ์ด๋ธ๋ฌ๋ฆฌ๋ฅผ ์ฌ์ฉํ์ฌ ๋ณดํต ํด๋ผ์ด์ธํธ๋ฅผ ๋ฐฐํฌํ ๋ ์ฌ์ฉํ๋ nginx ์๋ฒ๋ฅผ ๋ฐ๋ก ๊ตฌํํ์ง์๊ณ
ํ์ฌ ์ด API์๋ฒ์ ์ ์ ์ธ ํ์ด์ง๋ฅผ ์ ๊ณตํ๊ฒ๋ ํตํฉํ์ฌ ๊ตฌํ ํ์์ต๋๋ค.
Docker๋ฅผ ์ฌ์ฉํ์ฌ ๋น๋๋ ํด๋ผ์ด์ธํธ์ ์๋ฒ๋ฅผ ์ด๋ฏธ์ง๋ก ๋ง๋ค์๊ณ
Google Cloud Run์ ๋น๋๋ ์ด๋ฏธ์ง๋ฅผ ์ฌ๋ ค์ ๋ฐฐํฌํ์์ต๋๋ค.
Young-blog ๋ฐฐํฌ๋ ์ฌ์ดํธ์ ๋๋ค.
posts API
ํฌ์คํธ ๋ฑ๋กํ๊ธฐ
- POST /api/posts
/* Request body */
{
"title": "์ ๋ชฉ",
"body": "๋ด์ฉ",
"tags": ["ํ๊ทธ1", "ํ๊ทธ2"]
}
ํฌ์คํธ ์ ๋ถ๋ค ๊ฐ์ ธ์ค๊ธฐ
- GET /api/posts/
ํด๋น id๊ฐ์ ํฌ์คํธ ๊ฐ์ ธ์ค๊ธฐ
- GET /api/posts/:id
ํด๋น ํ์ด์ง์ ํฌ์คํธ ๊ฐ์ ธ์ค๊ธฐ
- GET /api/posts?username=&tag=&page=
page๋ง๋ค ์ต๋ ํฌ์คํธ : 10
โLast-pageโ ์ ๋ณด๋ Request Headers์ ์์.
username๊ณผ tag๋ก ํด๋น ํ์ด์ง ์ฐพ์์ฌ์์์.
ํด๋น id๊ฐ์ ํฌ์คํธ ์ญ์ ํ๊ธฐ
- DELETE /api/posts/:id
ํด๋น id๊ฐ์ ํฌ์คํธ ์์ ํ๊ธฐ
- PATCH /api/posts/:id
/* Request body */
{
"title": "์์ ",
"body": "์์ ๋ด์ฉ",
"tags": ["์์ ํ๊ทธ1", "์์ ํ๊ทธ2"]
}
user API
ํ์๊ฐ์
- POST /api/auth/register
/* Request body */
{
"username": "username",
"password": "password"
}
๋ก๊ทธ์ธ
- POST /api/auth/login
/* Request body */
{
"username": "username",
"password": "password"
}
๋ก๊ทธ์ธ ์ํ ํ์ธํ๊ธฐ
- GET /api/auth/check
๋ก๊ทธ์์
- POST /api/auth/logout
Environments
- koa@2.13.1
- koa-router@10.0.0
- koa-static@5.0.0
- koa-bodyparser@4.3.0
- bcrypt@5.0.1
- dotenv@8.2.0
- esm@3.2.25
- joi@17.4.0
- jsonwebtoken@8.5.1
- mongoose@5.11.19
- sanitize-html@2.3.3