使用typescript改造koa開(kāi)發(fā)框架的實(shí)現(xiàn)
強(qiáng)類(lèi)型的 TypeScript 開(kāi)發(fā)體驗(yàn)和維護(hù)項(xiàng)目上相比 JavaScript 有著明顯的優(yōu)勢(shì),那么對(duì)常用的腳手架進(jìn)行改造也就勢(shì)在必行了。
接下來(lái)開(kāi)始對(duì)基于 koa 框架的 node 后端腳手架進(jìn)行改造:
- 項(xiàng)目開(kāi)發(fā)環(huán)境 和 typescript 編譯環(huán)境的搭建;
- 對(duì) node、koa、koa中間件和使用到的庫(kù) 添加類(lèi)型化支持;
- 基于 typesript 的特性改造項(xiàng)目。
項(xiàng)目開(kāi)發(fā)環(huán)境搭建
基于 gulp 搭建開(kāi)發(fā)編譯環(huán)境,gulp-typescript 插件用于編譯 typescript 文件, gulp-nodemon 則可以監(jiān)控文件內(nèi)容的變更,自動(dòng)編譯和重啟node服務(wù),提升開(kāi)發(fā)效率。
npm install -D gulp gulp-nodemon gulp-typescript ts-node typescript
gulp 的配置
gulpfile.js 的設(shè)置
const { src, dest, watch, series, task } = require('gulp');
const del = require('del');
const ts = require('gulp-typescript');
const nodemon = require('gulp-nodemon');
const tsProject = ts.createProject('tsconfig.json');
function clean(cb) {
return del(['dist'], cb);
}
// 輸出 js 到 dist目錄
function toJs() {
return src('src/**/*.ts')
.pipe(tsProject())
.pipe(dest('dist'));
}
// nodemon 監(jiān)控 ts 文件
function runNodemon() {
nodemon({
inspect: true,
script: 'src/app.ts',
watch: ['src'],
ext: 'ts',
env: { NODE_ENV: 'development' },
// tasks: ['build'],
}).on('crash', () => {
console.error('Application has crashed!\n');
});
}
const build = series(clean, toJs);
task('build', build);
exports.build = build;
exports.default = runNodemon;
typescript 的配置
tsconfig.json 的設(shè)置
{
"compilerOptions": {
"baseUrl": ".", // import的相對(duì)起始路徑
"outDir": "./dist", // 構(gòu)建輸出目錄
"module": "commonjs",
"target": "esnext",// node 環(huán)境支持 esnext
"allowSyntheticDefaultImports": true,
"importHelpers": true,
"strict": false,
"moduleResolution": "node",
"esModuleInterop": true,
"forceConsistentCasingInFileNames": true,
"noImplicitAny": true,
"suppressImplicitAnyIndexErrors": true,
"noUnusedParameters": true,
"noUnusedLocals": true,
"noImplicitReturns": true,
"experimentalDecorators": true, // 開(kāi)啟裝飾器的使用
"emitDecoratorMetadata": true,
"allowJs": true,
"sourceMap": true,
"paths": {
"@/*": [ "src/*" ]
}
},
"include": [
"src/**/*"
],
"exclude": [
"node_modules",
"dist"
]
}
eslint 的配置
當(dāng)然 eslint 也要添加對(duì) typescript 對(duì)支持
npm install -D @typescript-eslint/eslint-plugin @typescript-eslint/parser
.eslintrc.json 的設(shè)置
{
"env": {
"es6": true,
"node": true
},
"extends": [
"eslint:recommended",
"plugin:@typescript-eslint/eslint-recommended"
],
"globals": {
"Atomics": "readonly",
"SharedArrayBuffer": "readonly"
},
"parser": "@typescript-eslint/parser",
"parserOptions": {
"ecmaVersion": 2018,
"sourceType": "module"
},
"plugins": [
"@typescript-eslint"
],
"rules": {
"indent": [ "warn", 2 ],
"no-unused-vars": 0
}
}
package.json 運(yùn)行配置
最后就是設(shè)置 package.json 的 scripts
"scripts": {
"start": "gulp",// dev
"build": "gulp build", // output
"eslint": "eslint --fix --ext .js,.ts src/",
"server": "export NODE_ENV=production && node dist/app" // production server
},
添加類(lèi)型化支持
項(xiàng)目主要使用到了以下的組件
jsonwebtoken
koa
koa-body
koa-compress
koa-favicon
koa-logger
koa-router
koa-static
koa2-cors
log4js
那么就要安裝對(duì)應(yīng)的 type 文件,當(dāng)然別忘了 @types/node
npm install -D @types/jsonwebtoken @types/koa @types/koa-compress @types/koa-favicon @types/koa-logger @types/koa-router @types/koa-static @types/koa2-cors @types/log4js @types/node
使用 typescript 裝飾器 改造項(xiàng)目
.net mvc 框架有個(gè)很便利的地方就是 使用裝飾器對(duì)控制器進(jìn)行配置,現(xiàn)在通過(guò) typescript 的裝飾器也可以實(shí)現(xiàn)相同的功能。這里需要使用到反射相關(guān)的庫(kù) reflect-metadata,用過(guò) Java 或 C# 的小伙伴,對(duì)反射的原理一定不陌生。
定義http請(qǐng)求的裝飾器
我們?cè)僖膊恍枰诼酚膳渲煤涂刂破鞣椒ㄖ皝?lái)回查找和匹配了
import 'reflect-metadata'
import { ROUTER_MAP } from '../constant'
/**
* @desc 生成 http method 裝飾器
* @param {string} method - http method,如 get、post、head
* @return Decorator - 裝飾器
*/
function createMethodDecorator(method: string) {
// 裝飾器接收路由 path 作為參數(shù)
return function httpMethodDecorator(path: string) {
return (proto: any, name: string) => {
const target = proto.constructor;
const routeMap = Reflect.getMetadata(ROUTER_MAP, target, 'method') || [];
routeMap.push({ name, method, path });
Reflect.defineMetadata(ROUTER_MAP, routeMap, target, 'method');
};
};
}
// 導(dǎo)出 http method 裝飾器
export const post = createMethodDecorator('post');
export const get = createMethodDecorator('get');
export const del = createMethodDecorator('del');
export const put = createMethodDecorator('put');
export const patch = createMethodDecorator('patch');
export const options = createMethodDecorator('options');
export const head = createMethodDecorator('head');
export const all = createMethodDecorator('all');
裝飾控制器的方法
export default class Sign {
@post('/login')
async login (ctx: Context) {
const { email, password } = ctx.request.body;
const users = await userDao.getUser({ email });
// ...
return ctx.body = {
code: 0,
message: '登錄成功',
data
};
}
@post('/register')
async register (ctx: Context) {
const { email, password } = ctx.request.body;
const salt = makeSalt();
// ...
return ctx.body = {
code: 0,
message: '注冊(cè)成功!',
data
}
}
}
收集元數(shù)據(jù)和添加路由
我們已經(jīng)把裝飾器添加到對(duì)應(yīng)控制器的方法上了,那么怎么把元數(shù)據(jù)收集起來(lái)呢?這就需要用到 node 提供的 fs 文件模塊,node服務(wù)第一次啟動(dòng)的時(shí)候,掃描一遍controller文件夾,收集到所有控制器模塊,結(jié)合裝飾器收集到的metadata,就可以把對(duì)應(yīng)的方法添加到 koa-router。
import 'reflect-metadata'
import fs from 'fs'
import path from 'path'
import { ROUTER_MAP } from './constant'
import { RouteMeta } from './type'
import Router from 'koa-router'
const addRouter = (router: Router) => {
const ctrPath = path.join(__dirname, 'controller');
const modules: ObjectConstructor[] = [];
// 掃描controller文件夾,收集所有controller
fs.readdirSync(ctrPath).forEach(name => {
if (/^[^.]+?\.(t|j)s$/.test(name)) {
modules.push(require(path.join(ctrPath, name)).default)
}
});
// 結(jié)合meta數(shù)據(jù)添加路由
modules.forEach(m => {
const routerMap: RouteMeta[] = Reflect.getMetadata(ROUTER_MAP, m, 'method') || [];
if (routerMap.length) {
const ctr = new m();
routerMap.forEach(route => {
const { name, method, path } = route;
router[method](path, ctr[name]);
})
}
})
}
export default addRouter
最后
這樣對(duì)koa項(xiàng)目腳手架的改造基本完成,源碼請(qǐng)查看koa-server
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
詳細(xì)談?wù)凟S6中的symbol數(shù)據(jù)類(lèi)型
這篇文章主要給大家介紹了關(guān)于ES6中symbol數(shù)據(jù)類(lèi)型的相關(guān)資料,Symbol函數(shù)的特性是每一個(gè)Symbol函數(shù)的返回值都是唯一的,可以通過(guò)給symbol函數(shù)傳遞不同的參數(shù)產(chǎn)生具有不同標(biāo)記的值,需要的朋友可以參考下2021-08-08
詳解使用Next.js構(gòu)建服務(wù)端渲染應(yīng)用
這篇文章主要介紹了詳解使用Next.js構(gòu)建服務(wù)端渲染應(yīng)用,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-07-07
JS實(shí)現(xiàn)頭條新聞的經(jīng)典輪播圖效果示例
這篇文章主要介紹了JS實(shí)現(xiàn)頭條新聞的經(jīng)典輪播圖效果,涉及javascript圖片輪播切換相關(guān)實(shí)現(xiàn)技巧,需要的朋友可以參考下2019-01-01
使用js Math.random()函數(shù)生成n到m間的隨機(jī)數(shù)字
何使用js生成n到m間的隨機(jī)數(shù)字,主要目的是為后期的js生成驗(yàn)證碼做準(zhǔn)備,Math.random()函數(shù)返回0和1之間的偽隨機(jī)數(shù)2014-10-10
javascript查找字符串中出現(xiàn)最多的字符和次數(shù)的小例子
這篇文章介紹了javascript查找字符串中出現(xiàn)最多的字符和次數(shù)的小例子,有需要的朋友可以參考一下2013-10-10
js數(shù)組常用操作方法小結(jié)(增加,刪除,合并,分割等)
這篇文章主要介紹了js數(shù)組常用操作方法,結(jié)合實(shí)例總結(jié)了javascript數(shù)組的增加、刪除、合并、分割等操作技巧,需要的朋友可以參考下2016-08-08
Javascript 判斷客戶(hù)端瀏覽器類(lèi)型代碼
有時(shí)候一些js代碼并不希望在別的瀏覽器下運(yùn)行,就需要事先判斷一下,方便下面的操作。2010-03-03
JavaScript 實(shí)現(xiàn)同時(shí)選取多個(gè)時(shí)間段的方法
這篇文章主要介紹了JavaScript 實(shí)現(xiàn)同時(shí)選取多個(gè)時(shí)間段的方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-10-10

