使用typescript改造koa開發框架

来源:https://www.cnblogs.com/edwardloveyou/archive/2020/02/04/12259245.html
-Advertisement-
Play Games

原文地址: "使用typescript改造koa開發框架" 強類型的 TypeScript 開發體驗和維護項目上相比 JavaScript 有著明顯的優勢,那麼對常用的腳手架進行改造也就勢在必行了。 接下來開始對基於 koa 框架的 node 後端腳手架進行改造: 1. 項目開發環境 和 types ...


原文地址:使用typescript改造koa開發框架
強類型的 TypeScript 開發體驗和維護項目上相比 JavaScript 有著明顯的優勢,那麼對常用的腳手架進行改造也就勢在必行了。

接下來開始對基於 koa 框架的 node 後端腳手架進行改造:

  1. 項目開發環境 和 typescript 編譯環境的搭建;
  2. nodekoa、koa中間件和使用到的庫 添加類型化支持;
  3. 基於 typesript 的特性改造項目。

項目開發環境搭建

基於 gulp 搭建開發編譯環境,gulp-typescript 插件用於編譯 typescript 文件, gulp-nodemon 則可以監控文件內容的變更,自動編譯和重啟node服務,提升開發效率。

npm install -D gulp gulp-nodemon gulp-typescript ts-node typescript

gulp 的配置

gulpfile.js 的設置

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 監控 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 的設置

{
  "compilerOptions": {
    "baseUrl": ".", // import的相對起始路徑
    "outDir": "./dist", // 構建輸出目錄
    "module": "commonjs",
    "target": "esnext",// node 環境支持 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, // 開啟裝飾器的使用
    "emitDecoratorMetadata": true,
    "allowJs": true,
    "sourceMap": true,
    "paths": {
      "@/*": [ "src/*" ]
    }
  },
  "include": [
    "src/**/*"
  ],
  "exclude": [
    "node_modules",
    "dist"
  ]
}

eslint 的配置

當然 eslint 也要添加對 typescript 對支持

npm install -D @typescript-eslint/eslint-plugin @typescript-eslint/parser

.eslintrc.json 的設置

{
  "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 運行配置

最後就是設置 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
  },

添加類型化支持

項目主要使用到了以下的組件

  • jsonwebtoken
  • koa
  • koa-body
  • koa-compress
  • koa-favicon
  • koa-logger
  • koa-router
  • koa-static
  • koa2-cors
  • log4js

那麼就要安裝對應的 type 文件,當然別忘了 @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 裝飾器 改造項目

.net mvc 框架有個很便利的地方就是 使用裝飾器對控制器進行配置,現在通過 typescript 的裝飾器也可以實現相同的功能。這裡需要使用到反射相關的庫 reflect-metadata,用過 JavaC# 的小伙伴,對反射的原理一定不陌生。

定義http請求的裝飾器

我們再也不需要在路由配置和控制器方法之間來回查找和匹配了

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 作為參數
  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');
    };
  };
}

// 導出 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: '註冊成功!',
      data
    }
  }
  
}

收集元數據和添加路由

我們已經把裝飾器添加到對應控制器的方法上了,那麼怎麼把元數據收集起來呢?這就需要用到 node 提供的 fs 文件模塊,node服務第一次啟動的時候,掃描一遍controller文件夾,收集到所有控制器模塊,結合裝飾器收集到的metadata,就可以把對應的方法添加到 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)
    }
  });
  // 結合meta數據添加路由
  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

最後

這樣對koa項目腳手架的改造基本完成,源碼請查看 koa-server


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • $衝突的解決方案 遇到其他js文件也用$包裝了函數。可以把jQuery放在後面,並釋放下$的控制權,也可以換個字元替代原來的$,例如$$ 或者,jQuery //jQuery釋放$的控制權 $$ = $.noConflict(); <!DOCTYPE html> <html lang="zh-CN" ...
  • javascript 函數 return 後面沒有返回值 代表提早退出語句,return後面的語句都不再執行 此時返回值為undefined <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Document< ...
  • each方法 each方法 //each方法 $("li").each(function (index, element) { $(element).css("opacity", (index + 1) / 10); }) for迴圈方法 for (var i = 0; i < $("li").le ...
  • 五角星評分案例 1. 滑鼠經過li的時候,當前的位置是實心五角星,前面的是實心。當前位置後面的是空心。註意此處不能完全用鏈式編程寫到底 2. 滑鼠離開,comment的所有孩子變為空心五角星。額外,找到current, 讓current和它的前面都是實心。 3. 點擊li的時候,當前的位置增加個cu ...
  • if() else if() else() alert() 彈出警告框 prompt() 輸入框,確定:返回輸入信息;取消:返回null <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Document</t ...
  • 鏈式編程 設置性操作:可以鏈式編程 獲取性操作,不能鏈式,因為獲取性操作,數值,字元串,返回值是不是一個jq對象。 $(function () { //設置性操作:可以鏈式編程 //獲取性操作,不能鏈式,因為獲取性操作,數值,字元串,返回值是不是一個jq對象。 $("div").width(200) ...
  • HTTP的歷史 3 項 WWW 構建技術,分別是:把 SGML(Standard Generalized Markup Language,標準通用標記語言)作為頁面的文本標記語言的 HTML(HyperText Markup Language,超文本標記語言);作為文檔傳遞協議的 HTTP ; 指定 ...
  • 鋼琴案例 (按下1-9數字鍵,能觸發對應的mouseenter事件) 1. 結合之前的學習,主要內容,就是on註冊keyup事件,函數里傳入e, 用e.keyCode,來獲取1-9的數字的範圍。 如果所按的按鍵在49-57的範圍里,就去觸發mouseenter(or mouseleave)事件。 怎 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...