登录鉴权与 Token 管理

一、前言:为什么选择 TSRPC 做登录鉴权?

  • TSRPC 核心优势:类型安全、前后端一体化、无冗余编码
  • 登录鉴权场景适配性:Token 机制与 TSRPC 接口拦截的契合度

二、前期准备:环境搭建与基础配置

    1. 技术栈选型说明
    • 前端:Vue
    • 后端:Node.js + TSRPC
    • 数据库:MongoDB(存储用户信息与 Token)
    • 依赖:jsonwebtoken(生成 Token)
    1. TSRPC 前后端初始化
    • 服务端项目创建与接口基础配置
    • 前端项目创建与 TSRPC 客户端连接
    • 前后端类型同步(TSRPC 核心特性,确保数据格式一致)
    1. 数据库表设计
    • 用户表:存储用户名、密码等基础信息
    • Token 表:存储用户 ID、Token 字符串、过期时间等

三、核心实现:登录流程全解析

  • 3.1 第一步:前端登录表单实现与数据提交
    • 登录表单组件开发(用户名、密码输入框 + 提交按钮)
    • 表单验证:非空校验、格式合法性校验
    • TSRPC 客户端调用登录接口(apiLogin),提交用户名密码
  • 3.2 第二步:后端 apiLogin 接口实现(免鉴权配置)
    • TSRPC 接口定义:请求参数类型(用户名、密码)与响应类型(Token)
    • 免鉴权配置:通过 TSRPC 中间件排除 apiLogin 接口,无需 Token 即可访问
    • 后端校验逻辑,查询数据库,验证用户名是否存在,验证密码是否正确
  • 3.3 第三步:Token 生成与数据库存储
    • 基于 jsonwebtoken 生成 Token:传入用户 ID 等核心信息,设置过期时间
    • Token 存储到数据库:关联用户 ID,记录生成时间与过期时间
    • 后端响应结果:将生成的 Token 返回给前端

四、关键拓展:全局接口鉴权实现

  • 4.1 前端全局请求拦截

    • 响应拦截

      TypeScript 复制代码
      const TokenStorageKey = 'tsrpc-token'
      // 响应拦截器
      client.flows.preApiReturnFlow.push((v) => {
        if (v.return.isSucc) {
          if (v.return.res.__ssoToken) {
            localStorage.setItem(TokenStorageKey, JSON.stringify(v.return.res.__ssoToken))
          }
        } else {
          const errorMsg = v.return.err.message
          // 解析错误码
          if (errorMsg.startsWith('NEED_LOGIN:')) {
            ElMessage.error('请先登录')
            router.push('/login')
          } else {
            ElMessage.error(errorMsg.replace(/^[\w_]+:\s*/, '')) // 移除错误码前缀
          }
        }
      
        return v
      })
    • 请求拦截

      TypeScript 复制代码
      client.flows.preCallApiFlow.push((v) => {
        let tokenStr = localStorage.getItem(TokenStorageKey)
        v.req.__ssoToken = tokenStr ? JSON.parse(tokenStr) : undefined
        return v
      })
  • 4.2 后端全局鉴权中间件:验证 Token 合法性

    • 拦截器截获接口中带的 __ssoToken

      TypeScript 复制代码
      /**
       * 在 API 调用前解析当前用户身份
       * @param server HttpServer
       */
      export function parseCurrentUser(server: HttpServer) {
        // API 调用前的拦截器 
        server.flows.preApiCallFlow.push(async (call) => {
          let req = call.req as BaseRequest
          if (req.__ssoToken) {
            // 验证并解析 SSO Token
            const user = await UserService.verifyToken(req.__ssoToken)
            if (user) {
              call.currentUser = user
            }
          }
          return call
        })
      }
    • 验证是用户身份

      复制代码
        ```TypeScript
        /**
         * 验证用户身份
         * @param server HttpServer
         */
        export function enableAuthentication(server: HttpServer) {
          // API 调用前的拦截器
          server.flows.preApiCallFlow.push((call) => {
            let conf: BaseConf | undefined = call.service.conf
        
            // 默认需要登录,除非明确设置 needLogin: false
            if (conf?.needLogin !== false && !call.currentUser) {
              call.error('NEED_LOGIN: 请先登录')
              return undefined
            }
        
            return call
          })
        }
        ```
    • 校验结果处理:通过则放行,不通过则call.error

五、补充功能:Token 注销与过期处理

    1. 退出登录功能:前端清除本地 Token + 后端删除数据库中对应 Token 记录
    1. Token 过期处理:
    • 前端:拦截器中检测到以 NEED_LOGIN: 开头的消息,跳转到登录页
    TypeScript 复制代码
    const errorMsg = v.return.err.message
    // 解析错误码
    if (errorMsg.startsWith('NEED_LOGIN:')) {
      ElMessage.error('请先登录')
      router.push('/login')
    } else {
      ElMessage.error(errorMsg.replace(/^[\w_]+:\s*/, '')) // 移除错误码前缀
    }
    1. 数据库基本设计,两个表,token 和 user 两个表
    TypeScript 复制代码
    export interface db_Token {
      _id?: ObjectId
      userId: string
      token: string
      createdAt?: Date
      expiresAt?: Date
    }
    TypeScript 复制代码
    export interface db_User {
      _id: ObjectId | string;
      username: string;  // 用户名
      password: string; // 密码
      thumb?: string;  // 头像
      state: number;  // 账户状态
      updatedAt?: Date | string
      createdAt?: Date | string
    }

六:部分截图


gitee 开源地址:https://gitee.com/liucenter/tsrpc-manage-system

相关推荐
乐于分享的阿乐14 小时前
最新长期支持版本nodejs安装及环境配置(保姆级图文+安装包)
node.js
m0_5358175514 小时前
macOS上Claude Code安装配置保姆级教程:国内直连API,从0到1跑通(附避坑指南)
gpt·macos·ai·node.js·claude·claudecode·88api
五月君_16 小时前
放弃 Python,Kimi 用 TS + Node.js 重写了一个 Kimi Code
开发语言·python·node.js
涛声依旧-底层原理研究所18 小时前
Node.js在高并发低延迟场景中的优势
java·人工智能·python·node.js
晓杰'19 小时前
从0到1实现Balatro游戏后端(5):得分计算与单局结算流程实现
后端·typescript·node.js·游戏开发·项目实战·nestjs·webscoket
参宿719 小时前
Shell 脚本语言(Bash/Sh)基础 与 应用
node.js
weifengma-wish21 小时前
通过NPM安装claude code
前端·npm·node.js
不总是1 天前
Windows 系统 Node.js 免安装版(zip)安装与配置教程(2026 最新)
前端·windows·node.js
蓝乐2 天前
Express 知识点总结
node.js·express
kylinmin2 天前
Node.js安装及环境配置超详细教程(以win11为例子)
node.js