經過重啟北護課程查詢系統 Day 1 - 頁面爬蟲的小試身手,Day 2 要來決定如何實作網頁啦!
筆者自己想要學點新的東西,所以決定使用標題打的技術來實作~~ 可能有些框架大家不是很熟悉,下面我一一的請 GPT 介紹
- Fastify: 是一個高性能、可擴展、開發人員友好的 Node.js 後端 web server 框架。
其主要特性包括:
- 高性能:每秒可處理 3 萬個請求。
- 可擴展:通過 hook、plugin 和 decorator 提供可擴展性。
- 開發人員友好:提供簡潔易用的 API 和豐富的文檔。
- HTMX: 是一個輕量級的前端框架,旨在讓開發人員使用 HTML 和 CSS 輕鬆構建動態且交互性強的 Web 應用程序。
- EJS: 是一個 JavaScript 的模板語言,用於生成 HTML、XML、JSON 等各種文本格式的內容。
其主要特性包括:
- 語法簡潔易學,類似於 JavaScript。
- 支持 JavaScript 條件語句和循環等控制結構。
- 可與 Express 等 Web 框架無縫整合。
- Tailwind: 是一個 CSS 框架,旨在幫助開發人員快速構建具有現代化 UI 設計的 Web 應用程序。
其主要特性包括:
- 原子級 CSS:提供一系列可組合的 CSS 類,可快速構建複雜的 UI 元素。
- 響應式設計:提供響應式媒體查詢,可根據屏幕尺寸自動調整 UI 布局。
- 可定制性:提供主題和自定義配置,可打造個性化的 UI 設計。
📢
想少時間的大大建議可以選擇看 github 唷! https://github.com/ntunhs-course-ecosystem/course-hub/tree/day2
建立專案
- 在電腦找個舒適的位置創建資料夾命名為
course-hub
- 初始化專案
pnpm init
- 記得要在 package.json 新增
“type”: “module”
建立 nodejs 的 dev 腳本
- 我們在開發情境下,通常會需要監看檔案是否更改,實時更新重啟專案
- 這時,就需要為 node.js 專案新增 scripts,達到上面提到的需求
修改 package.json
- 修改 package.json,加入與 dev 相關的 scripts
nodemonConfig
則是用到 node.js 常用的重啟工具,在這邊我們只監看 server 底下的檔案
{
...
"scripts": {
"dev": "npm-run-all -p dev:*",
"dev:fastify": "nodemon server/server.js | pino-pretty",
"dev:css": "tailwindcss -i client/bundle/main.pcss -o client/public/_compiled/main.bundle.css --postcss --watch",
"dev:js": "esbuild client/bundle/main.js --outfile=client/public/_compiled/main.bundle.js --bundle --watch",
"test": "echo \"Error: no test specified\" && exit 1"
},
"nodemonConfig": {
"watch": [
"server/"
]
}
...
}
拉出專案基本雛形
└─course-hub
├─client
│ └─bundle # 儲存要進行 bundle 的檔案
│ └─public # 儲存前端靜態內容
└─server
├─modules # 放置以 Domain 為主命名的 routes
├─plugins # 放置 fastify plugins
├─repository # 放置資料庫內容的地方
└─views # 放置 EJS 前端頁面
安裝基本套件
這章節將會安裝"基本”的套件,主要都是參照 fastify-starter 的套件
DEV 套件
pnpm i -D @tailwindcss/forms esbuild npm-run-all postcss postcss-import tailwindcss tailwindcss-debug-screens prettier eslint-config-prettier prettier-plugin-ejs nodemon
一般套件
pnpm i @fastify/autoload @fastify/formbody @fastify/static @fastify/view @fastify/sensible @fastify/under-pressure @fastify/cors dotenv ejs fastify fastify-plugin htmx.org better-sqlite3 desm env-var close-with-grace
初始化 tailwind
創建 tailwind config
- 在專案根目錄創建
tailwind.config.cjs
module.exports = {
content: ["./server/views/**/*.ejs"],
theme: {
container: {
center: true,
padding: "2rem"
},
debugScreens: {
position: ["bottom", "right"]
}
},
plugins: [
require("@tailwindcss/forms"),
require("tailwindcss-debug-screens")
]
};
創建 postcss config
- 在專案根目錄創建
postcss.config.cjs
module.exports = {
plugins: [
require("postcss-import"),
require("tailwindcss/nesting"),
require("tailwindcss")
]
};
創建 main.css
- 在專案的
client/bundle
創建main.pcss
@import "tailwindcss/base";
@import "tailwindcss/components";
@import "tailwindcss/utilities";
@layer base {
html, body {
@apply w-full overflow-x-hidden antialiased font-sans;
}
}
創建 .prettierrc.json
- 在專案根目錄創建
.prettierrc.json
規範一些 code auto format 的規則trailingComma
: 函數參數、Object 的最後一個元素後面不會有逗號tabWidth
: 這個選項控制一個 tab 字符的寬度,以空格為單位,此處設置為 4,表示一個 tab 字符等於 4 個空格semi
: 最後使用分號singleQuote
: 這個選項控制是否使用單引號 ’’,設置為 false 表示使用雙引號 “”quoteProps
: property 的雙引號使用要一致- 設置
prettier-plugin-ejs
讓 vscode 可以自動格式化 ejs 程式碼
{
"trailingComma": "none",
"tabWidth": 4,
"semi": true,
"singleQuote": false,
"quoteProps": "consistent",
"plugins": [
"prettier-plugin-ejs"
]
}
創建 .eslintrc
- 在專案根目錄創建
.eslintrc
以便透過靜態分析程式碼- parserOptions
ecmaVersion
: 指定要使用的語言規範版本,這邊使用 ES2022sourceType
: 指定程式碼類型,這邊使用 module
- env
browser
: 啟用與瀏覽器相關的規則node
: 啟用與 Node.js 相關的規則es2022
: 啟用與 ES2022 語法相關的規則
- extends
eslint:recommended
: 使用 eslint 內建推薦的規則prettier
: 使用 prettier 提供的規則
- rules
semi
: 強制使用分號,並指定為總是使用comma-dangle
: 禁止尾逗號,即函數參數、Object 的最後一個元素後面不應該有逗號no-unused-vars
: 關閉禁止未使用變量的規則no-console
: 允許使用 console 對象(警告級別降低為 0)no-useless-escape
: 關閉禁止不必要的轉義字符的規則no-useless-catch
: 關閉禁止無用的 catch 塊的規則no-var
: 禁止使用 var 聲明變數,強制使用 let 和 constno-control-regex
: 關閉禁止使用控制字符的正則表達式的規則 (ASCII 0~31)。
- parserOptions
{
"parserOptions": {
"ecmaVersion": 2022,
"sourceType": "module"
},
"env": {
"browser": true,
"node": true,
"es2022": true
},
"extends": ["eslint:recommended", "prettier"],
"rules": {
"semi": ["error", "always"],
"comma-dangle": ["error", "never"],
"no-unused-vars": "off",
"no-console": 0,
"no-useless-escape": "off",
"no-useless-catch": "off",
"no-var": "error",
"no-control-regex": "off"
}
}
渲染第一個 Hello World 頁面
因為筆者是第一次接觸這個架構,所以就先鎖定在 Hello World 吧!
後端
.env
- 在專案根目錄新建
.env
檔案
SERVER_HOST="0.0.0.0"
SERVER_PORT=3000
server/app.js
- 在專案目錄的
server
資料夾新建app.js
檔案
import fasitfy from "fastify";
import autoLoad from "@fastify/autoload";
import { join } from "desm";
export async function buildApp(options = {}) {
const app = fasitfy(options);
await app.register(autoLoad, {
dir: join(import.meta.url, "plugins")
});
await app.register(autoLoad, {
dir: join(import.meta.url, "modules"),
encapsulate: false,
maxDepth: 1
});
return app;
}
/** @type {import("fastify").FastifyServerOptions} */
const appOptions = {
logger: {
level: "info"
}
};
// 我們希望只有在有人監看(開發)時才使用 pino-pretty
// 否則我們將以換行分隔的 JSON 形式進行記錄,以便輸入至 log 專用的 tool
if (process.stdout.isTTY) {
appOptions.logger.transport = {
target: "pino-pretty"
};
}
export default await buildApp(appOptions);
server/env-class.js
- 在專案目錄的
server
資料夾新建env-class.js
檔案
// 抓取 env 的設定
// 使用 env-var 來驗證以及針對數值轉換 env 的設定
import envVar from "env-var";
import dotenv from "dotenv";
dotenv.config();
export const ServerEnv = function () {
return {
host: envVar.get("SERVER_HOST").required().default("0.0.0.0").asString(),
port: envVar.get("SERVER_PORT").required().default(3000).asPortNumber()
};
};
server/plugins/
👉🏻
以下檔案都會建立在 server/plugins
內
cors.js
import fp from "fastify-plugin";
import cors from "@fastify/cors";
async function createCorsPlugin(fastify, opts) {
fastify.register(cors, {
origin: false
});
}
export default fp(createCorsPlugin, {
name: "cors"
});
formBody.js
import fp from "fastify-plugin";
import formBody from "@fastify/formbody";
export default fp(async (fastify, opts)=> {
fastify.register(formBody);
}, {
name: "form-body"
});
sensible.js
import fp from "fastify-plugin";
import sensible from "@fastify/sensible";
export default fp(async (fastify, opts) => {
await fastify.register(sensible);
}, {
name: "sensible"
});
static.js
import fp from "fastify-plugin";
import { join } from "desm";
import fStatic from "@fastify/static";
export default fp(async (fastify, opts) => {
fastify.register(fStatic, {
root: join(import.meta.url, "../../client/public")
});
}, {
name: "f-static"
});
under-pressure.js
import fp from "fastify-plugin";
import underPressure from "@fastify/under-pressure";
export default fp(async (fastify, opts) => {
/**
* 這個 plugin 會在你的應用程序處於高負載時特別有用
* 你如果有使用 express 或其餘的 web framework,你會發現 1 秒一起發送上萬個 request 可能會造成 timeout 的問題
* 此 plugin 會根據 nodejs event loop, heap memory, rss memory (Resident set size, 作業系統分配到application的記憶體)的狀況(過度壓力無法處裡)來響應 503 (Server Unavailable) 狀態
*/
await fastify.register(underPressure, {
maxEventLoopDelay: 1e3,
maxHeapUsedBytes: 1e9, // 約 1GB
maxRssBytes: 1e9,
maxEventLoopUtilization: 0.98
});
}, {
name: "under-pressure"
});
view-engine.js
import fp from "fastify-plugin";
import { join } from "desm";
import fView from "@fastify/view";
import ejs from "ejs";
export default fp(async (fastify, opts) => {
fastify.register(fView, {
engine: {
ejs: ejs
},
root: join(import.meta.url, "../views")
});
}, {
name: "view-engine"
});
server/module/
👉🏻
以下檔案都會建立在 server/module
內
home.js
/**
*
* @param {import("fastify").FastifyInstance} fastify
* @param {import("fastify").FastifyPluginOptions} opts
*/
export default async function (fastify, opts) {
fastify.get("/", async (request, reply) => {
return reply.view("components/layout", {
page: {
file: "../home",
data: {}
}
});
});
}
server/views/
👉🏻
以下檔案都會建立在 server/views
內
compoments/head.ejs
- 建立可重複使用的 head
<head>
<meta charset="utf-8" />
<meta http-equiv="X-UA-Compatible" content="IE=edge" />
<% if (title) { %>
<title><%= title %></title>
<% } else { %>
<title>NTUNHS CourseHub</title>
<% } %>
<meta name="description" content="" />
<meta name="viewport" content="width=device-width, initial-scale=1" />
<link href="/_compiled/main.bundle.css" rel="stylesheet" />
<script defer src="/_compiled/main.bundle.js"></script>
</head>
compoments/layout.ejs
- 建立可重複使用的 body
<!doctype html>
<html>
<%- include("components/head.ejs", {title: "NTUNHS CourseHub | Home"}) %>
<body hx-boost="true">
<%- include(page.file, page.data) %>
</body>
</html>
home.js
<h1 class="text-3xl font-bold underline">Hello world!</h1>
訪問網頁
啟動專案
- 要訪問網頁,第一步當然是啟動專案啦!
- 因為我們處於開發階段,所以這邊是運行 dev 的腳本
- 另外,我們有安裝 npm-run-all 以及設定好所有 dev 需要的腳本了
- 直接運行
pnpm dev
啟動專案吧!
訪問 127.0.0.1:3000
- 打開你的網頁訪問 http://127.0.0.1:3000 吧
最後的專案結構
|-- client
| |-- bundle
| | |-- main.js
| | `-- main.pcss
| `-- public
| `-- _compiled
| |-- main.bundle.css
| `-- main.bundle.js
|-- package.json
|-- pnpm-lock.yaml
|-- postcss.config.cjs
|-- server
| |-- app.js
| |-- env-class.js
| |-- modules
| | |-- course-hub
| | | |-- index.js
| | | `-- routes
| | `-- home.js
| |-- plugins
| | |-- cors.js
| | |-- formBody.js
| | |-- sensible.js
| | |-- static.js
| | |-- under-pressure.js
| | `-- view-engine.js
| |-- repository
| |-- server.js
| `-- views
| |-- components
| | `-- head.ejs
| `-- home.ejs
`-- tailwind.config.cjs