fastapi+vue中的用户权限管理设计
数据库设计:RBAC数据模型
这是一个典型的基于SQLAlchemy的RBAC权限系统数据模型实现,各模型分工明确,共同构成完整的权限管理系统。
图解说明:
-
实体关系:
-
用户(USER)和角色(ROLE)通过 USER_ROLE 中间表实现多对多关系
-
角色(ROLE)和权限(PERMISSION)通过 ROLE_PERMISSION 中间表实现多对多关系
-
-
关键连接点:
USER → USER_ROLE ← ROLE → ROLE_PERMISSION ← PERMISSION
这个链条完整表达了"用户→角色→权限"的授权路径
-
箭头含义:
-
||--o{
表示一对多关系 -
PK
表示主键 -
FK
表示外键
-
-
实际业务流:
当检查用户是否有某接口的访问权限时,系统会:用户ID → 查询USER_ROLE → 获取角色列表 → 查询ROLE_PERMISSION → 获取权限列表 → 匹配权限中的resource字段与当前请求接口
这个图示清晰地展示了您提供的四个模型如何协作完成RBAC权限控制。如果需要更详细的流程图或时序图,我可以进一步补充。
1. Role (角色模型)
-
核心作用:定义系统中的角色实体
-
关键字段:
-
role_name
: 角色唯一标识(如admin/hr/interviewer) -
description
: 角色描述信息
-
-
关联关系:
-
users
: 通过UserRole关联到用户(一对多) -
permissions
: 通过RolePermission关联到权限(一对多)
-
-
业务意义:角色是权限分配的中间层,用户通过角色间接获得权限
2. Permission (权限模型)
-
核心作用:定义系统中的具体权限项
-
关键字段:
-
permission_name
: 权限唯一标识(如knowledge_create) -
resource
: 关联的资源路径/API端点
-
-
关联关系:
-
roles
: 通过RolePermission关联到角色(一对多)
-
-
业务意义:权限是系统中最细粒度的访问控制单元,直接对应具体操作
3. UserRole (用户-角色关联模型)
-
核心作用:维护用户与角色的多对多关系
-
关键设计:
-
联合唯一约束(
uq_user_role
): 防止重复分配相同角色 -
级联删除: 用户删除时自动解除关联
-
-
关联关系:
-
user
: 关联到用户表 -
role
: 关联到角色表
-
-
业务意义:实现"用户拥有哪些角色"的映射关系
4. RolePermission (角色-权限关联模型)
-
核心作用:维护角色与权限的多对多关系
-
关键设计:
-
联合唯一约束(
uq_role_permission
): 防止重复分配相同权限 -
级联删除: 角色删除时自动解除关联
-
-
关联关系:
-
role
: 关联到角色表 -
permission
: 关联到权限表
-
-
业务意义:实现"角色包含哪些权限"的映射关系
整体协作流程
-
管理员创建权限(Permission)并定义其对应的资源
-
创建角色(Role)并将相关权限关联到角色(RolePermission)
-
为用户分配角色(UserRole)
-
系统通过检查用户的角色→权限→资源路径链来确定访问权限
这种设计实现了权限管理的解耦,使系统能够灵活应对权限变更,同时通过中间表实现了多对多关系的维护。
class User(Base):__tablename__ = 'user'__table_args__ = {'extend_existing': True}user_id = Column(Integer, primary_key=True, autoincrement=True, comment='Primary Key')user_name = Column(String(100), nullable=False)password = Column(String(200), nullable=False)user_email = Column(String(100), nullable=False, unique=True)user_department = Column(String(100), nullable=True)join_time = Column(DateTime, nullable=False, default=datetime.now)# 关联 Session 模型,代表该用户的所有会话sessions = relationship("Session", back_populates="user", cascade="all, delete-orphan")# 关联 Knowledge 模型,代表该用户关联的所有知识knowledge = relationship("Knowledge", back_populates="user", cascade="all, delete-orphan")# 添加与UserRole的关系roles = relationship("UserRole", back_populates="user", cascade="all, delete-orphan")class Role(Base):__tablename__ = 'role'__table_args__ = {'extend_existing': True}role_id = Column(Integer, primary_key=True, autoincrement=True, comment='Primary Key')role_name = Column(String(50), nullable=False, unique=True, comment='角色名称,如admin/hr/interviewer')description = Column(String(200), comment='角色描述')create_time = Column(DateTime, nullable=False, default=datetime.now)modify_time = Column(DateTime, nullable=False, default=datetime.now, onupdate=datetime.now)# 关联到UserRole和RolePermissionusers = relationship("UserRole", back_populates="role", cascade="all, delete-orphan")permissions = relationship("RolePermission", back_populates="role", cascade="all, delete-orphan")class Permission(Base):__tablename__ = 'permission'__table_args__ = {'extend_existing': True}permission_id = Column(Integer, primary_key=True, autoincrement=True, comment='Primary Key')permission_name = Column(String(100), nullable=False, unique=True, comment='权限名称,如knowledge_create/session_delete')description = Column(String(200), comment='权限描述')resource = Column(String(200), comment='对应的资源路径/API端点')create_time = Column(DateTime, nullable=False, default=datetime.now)modify_time = Column(DateTime, nullable=False, default=datetime.now, onupdate=datetime.now)# 关联到RolePermissionroles = relationship("RolePermission", back_populates="permission", cascade="all, delete-orphan")class UserRole(Base):__tablename__ = 'user_role'__table_args__ = (UniqueConstraint('user_id', 'role_id', name='uq_user_role'),{'extend_existing': True})id = Column(Integer, primary_key=True, autoincrement=True)user_id = Column(Integer, ForeignKey('user.user_id', ondelete='CASCADE'), nullable=False)role_id = Column(Integer, ForeignKey('role.role_id', ondelete='CASCADE'), nullable=False)create_time = Column(DateTime, nullable=False, default=datetime.now)# 关联到User和Roleuser = relationship("User", back_populates="roles")role = relationship("Role", back_populates="users")class RolePermission(Base):__tablename__ = 'role_permission'__table_args__ = (UniqueConstraint('role_id', 'permission_id', name='uq_role_permission'),{'extend_existing': True})id = Column(Integer, primary_key=True, autoincrement=True)role_id = Column(Integer, ForeignKey('role.role_id', ondelete='CASCADE'), nullable=False)permission_id = Column(Integer, ForeignKey('permission.permission_id', ondelete='CASCADE'), nullable=False)create_time = Column(DateTime, nullable=False, default=datetime.now)# 关联到Role和Permissionrole = relationship("Role", back_populates="permissions")permission = relationship("Permission", back_populates="roles")
后端技术实现流程:
JWT (JSON Web Token)
-
一种紧凑的、URL安全的令牌格式
-
包含三部分:Header(头部)、Payload(负载)和Signature(签名)
-
用于在各方之间安全地传输信息
OAuth 2.0
-
一个授权框架,允许第三方应用获取对资源的有限访问权限
-
定义了四种授权流程(授权码、隐式、密码、客户端凭证)
典型 JWT + OAuth 2.0 流程 (授权码模式)
-
用户发起授权请求
-
客户端将用户重定向到授权服务器
-
携带参数:
response_type=code
,client_id
,redirect_uri
,scope
等
-
-
用户认证与同意
-
用户在授权服务器登录并同意请求的权限
-
-
获取授权码
-
授权服务器返回授权码到客户端的
redirect_uri
-
-
用授权码交换令牌
-
客户端向授权服务器的令牌端点发送请求
-
携带:
grant_type=authorization_code
,code
,redirect_uri
,client_id
,client_secret
-
-
返回JWT令牌
-
授权服务器验证后返回访问令牌(通常是JWT)和刷新令牌
-
响应示例:
{"access_token": "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9...","token_type": "Bearer","expires_in": 3600,"refresh_token": "def50200ae2f..." }
-
-
使用访问令牌访问资源
-
客户端在请求头中加入:
Authorization: Bearer <access_token>
-
资源服务器验证JWT签名和声明
-
-
刷新令牌(可选)
-
当访问令牌过期时,使用刷新令牌获取新的访问令牌
-
JWT在OAuth中的优势
-
自包含:JWT包含所有必要信息,减少数据库查询
-
可验证性:资源服务器可以独立验证JWT而不需联系授权服务器
-
标准化结构:易于跨语言和平台实现
# app/middlewares/permission.py
from fastapi import Request, HTTPException, Depends
from sqlalchemy.orm import Session
from app.db_init import SessionLocal
from app.models.model import User, Permission, RolePermission, UserRole, Role # 添加 Role 导入
from functools import wrapsdef permission_required(permission_name: str):"""权限检查依赖工厂函数"""async def dependency(request: Request):db: Session = SessionLocal()try:user_id = getattr(request.state, 'user_id', None)if not user_id:raise HTTPException(status_code=401, detail="未认证")# 检查是否是管理员/root用户user = db.query(User).filter(User.user_id == user_id).first()if not user:raise HTTPException(status_code=401, detail="用户不存在")# 检查用户角色admin_role = (db.query(UserRole).join(Role, Role.role_id == UserRole.role_id).filter(UserRole.user_id == user_id).filter(Role.role_name == 'admin').first())if admin_role:return # 管理员直接通过权限检查# 普通用户权限检查has_permission = db.query(Permission).join(RolePermission, RolePermission.permission_id == Permission.permission_id).join(UserRole, UserRole.role_id == RolePermission.role_id).filter(UserRole.user_id == user_id,Permission.permission_name == permission_name).first()if not has_permission:raise HTTPException(status_code=403, detail="权限不足")finally:db.close()return Depends(dependency)
permission_required
函数是一个 FastAPI 的权限检查依赖工厂函数,主要用于实现基于权限的访问控制。它的主要作用和功能如下:
-
功能概述:
-
它是一个装饰器工厂函数,接收一个权限名称作为参数,返回一个 FastAPI 依赖项
-
用于检查当前请求的用户是否拥有指定的权限
-
-
工作流程:
-
从请求中获取用户 ID
-
查询数据库验证用户是否存在
-
检查用户是否是管理员角色(role_name 为 'admin'),如果是则直接通过检查
-
对于非管理员用户,检查用户是否拥有指定的权限
-
如果没有权限则返回 403 错误
-
-
权限检查逻辑:
-
通过连接查询检查用户角色关联的权限
-
查询涉及多个表:User → UserRole → RolePermission → Permission
-
只有当用户至少有一个角色拥有指定的权限时才会通过检查
-
-
异常处理:
-
未认证用户(无 user_id):返回 401
-
用户不存在:返回 401
-
权限不足:返回 403
-
什么是中间件?
在 Web 开发中,中间件(Middleware) 是一种机制,用于在 HTTP 请求到达路由处理函数之前 或 响应返回客户端之前 执行某些逻辑。它类似于一个“拦截器”,可以对请求和响应进行预处理或后处理。
中间件的作用
-
全局处理请求/响应
-
例如:身份验证、日志记录、跨域处理(CORS)、请求限流、数据压缩等。
-
-
修改请求或响应
-
例如:添加请求头、解析请求体、修改响应数据。
-
-
拦截非法请求
-
例如:检查权限(如你的
permission_required
)、阻止未授权的访问。
-
中间件的工作原理
在 FastAPI 或类似的框架(如 Express.js、Django)中,中间件通常按照 洋葱模型(Onion Model) 工作:
请求 → 中间件1 → 中间件2 → ... → 路由处理 → ... → 中间件2 → 中间件1 → 响应
-
请求阶段:中间件按顺序执行(如先验证身份,再检查权限)。
-
响应阶段:中间件按相反顺序执行(如先记录日志,再返回数据)。
FastAPI 中的中间件示例
1. 普通中间件(全局拦截)
from fastapi import FastAPI, Requestapp = FastAPI()@app.middleware("http")
async def log_requests(request: Request, call_next):print(f"收到请求: {request.method} {request.url}")response = await call_next(request) # 继续执行后续中间件或路由print(f"返回响应: {response.status_code}")return response
-
这个中间件会在 每个请求前后 打印日志。
2. 依赖注入式中间件(如你的 permission_required
)
from fastapi import Depends, HTTPExceptiondef check_auth(token: str):if token != "secret":raise HTTPException(status_code=403, detail="无权访问")return True@app.get("/admin")
async def admin_route(auth: bool = Depends(check_auth)):return {"message": "欢迎管理员"}
-
Depends(check_auth)
是一个依赖项,作用类似于中间件,但更灵活(可以针对单个路由使用)。
中间件 vs 依赖注入
特性 | 中间件 | 依赖注入(如 Depends ) |
---|---|---|
作用范围 | 全局或路由组 | 单个路由 |
执行顺序 | 所有请求必经 | 仅对声明了的路由生效 |
典型用途 | 日志、CORS、全局认证 | 细粒度权限检查、数据库会话管理 |
你的 permission_required
是中间件吗?
严格来说,它 不是传统中间件,而是一个 依赖项工厂函数(返回 Depends
)。但它实现了类似中间件的功能(权限检查),只是作用范围更精确(仅对使用它的路由生效)。
总结
-
中间件:适合全局逻辑(如日志、CORS)。
-
依赖注入:适合路由级逻辑(如权限、数据库会话)。
-
你的
permission_required
是依赖注入的典型应用,用于实现 RBAC(基于角色的访问控制)。
@router.post('/login')
async def login(request_data: LoginRequest, db = Depends(get_db)):'''登录界面'''username = request_data.usernamepassword = request_data.password# 根据用户名查询数据库中用户的密码info = AuthService.select_user_info(username, db)if info is None:return {'code': 401,'message': 'user not found'}# 密码验证if password != info['password']:return {'code': 401,'message': 'password error'}# 获取用户角色user_roles = AuthService.get_user_roles(info['user_id'], db)is_admin = 'admin' in [role.lower() for role in user_roles]# 获取用户权限permissions = AuthService.get_user_permissions(info['user_id'], db)if permissions is None:permissions = []# 创建JWT令牌access_token_expires = timedelta(minutes=ACCESS_TOKEN_EXPIRE_MINUTES)access_token = create_access_token(data={"sub": str(info['user_id']), # 确保user_id是字符串"username": info['user_name'],"permissions": permissions,"is_admin": is_admin,"roles": user_roles},expires_delta=access_token_expires)return {'code': 200,'message': 'success','data': {'user_id': info['user_id'],'username': info['user_name'],'access_token': access_token,'token_type': 'bearer','permissions': permissions,'roles': user_roles,'is_admin': is_admin}}
这个 login
函数是一个 FastAPI 路由处理函数,用于实现用户登录认证流程。它的核心作用是:
主要功能
-
用户身份认证
-
接收前端提交的用户名和密码(通过
LoginRequest
模型)。 -
验证用户名是否存在、密码是否正确。
-
返回 401 错误码和提示信息(如 "user not found" 或 "password error")。
-
-
权限和角色信息收集
-
查询用户的角色(如
user_roles
)并检查是否是管理员(is_admin
)。 -
查询用户拥有的权限列表(
permissions
)。
-
-
生成 JWT 令牌
-
使用
create_access_token
生成包含用户信息的 JWT(有效期通过ACCESS_TOKEN_EXPIRE_MINUTES
控制)。 -
令牌中存储的关键数据:
{"sub": "用户ID", # 用户唯一标识"username": "用户名","permissions": ["权限1", "权限2"], # 用户权限列表"is_admin": True/False, # 是否是管理员"roles": ["角色1", "角色2"] # 用户角色列表 }
-
-
返回登录结果
-
返回 200 状态码和用户信息(包括令牌、权限、角色等),供前端后续使用。
-
详细流程
-
参数接收
-
通过
LoginRequest
模型(未展示但应包含username
和password
字段)接收前端提交的登录数据。
-
-
用户验证
-
调用
AuthService.select_user_info
查询数据库,检查用户名是否存在。 -
比对数据库中的密码(明文比对,实际项目中应使用 哈希密码比对,如
bcrypt
)。
-
-
权限角色查询
-
通过
AuthService.get_user_roles
获取用户角色列表,并检查是否包含admin
角色。 -
通过
AuthService.get_user_permissions
获取用户权限列表(用于前端动态路由或按钮权限控制)。
-
-
令牌生成
-
使用
create_access_token
(可能是fastapi-jwt
或类似库)生成 JWT,包含用户关键信息。
-
-
响应返回
-
返回结构化响应,包含令牌和用户信息,例如:
{"code": 200,"message": "success","data": {"user_id": 123,"username": "john","access_token": "xxx.yyy.zzz","permissions": ["read", "write"],"roles": ["user"],"is_admin": false} }
-
关键点说明
-
安全注意事项
-
密码明文存储问题:当前代码直接比对明文密码,实际项目应使用 密码哈希(如
bcrypt
)存储和验证。 -
JWT 敏感信息:令牌中存储了
permissions
和roles
,需确保 JWT 使用 HTTPS 传输并设置合理有效期。
-
-
权限控制设计
-
前端可根据返回的
permissions
和roles
动态渲染界面。 -
后端可通过
permission_required
中间件(如你之前提供的)进一步校验权限。
-
-
扩展性
-
可添加登录日志记录、多设备登录限制、验证码等功能。
-
典型使用场景
-
前端提交登录表单 → 后端验证 → 返回令牌。
-
前端将令牌存储在
localStorage
或Cookie
中,后续请求通过Authorization: Bearer <token>
头部携带。 -
其他受保护的路由通过中间件校验令牌和权限(如
@router.get("/admin", dependencies=[Depends(permission_required("admin_access"))]
)。
这个函数是典型的认证系统核心部分,实现了从登录到权限令牌分发的完整流程。
from fastapi import Depends, HTTPException, status
from fastapi.security import OAuth2PasswordBearer
from jose import JWTError, jwt
from datetime import datetime, timedelta
from utils.configs import SECRET_KEY, ALGORITHM, ACCESS_TOKEN_EXPIRE_MINUTES
from fastapi import Request# OAuth2 配置
oauth2_scheme = OAuth2PasswordBearer(tokenUrl="token")# 创建访问令牌
def create_access_token(data: dict, expires_delta: timedelta = None):to_encode = data.copy()if expires_delta:expire = datetime.utcnow() + expires_deltaelse:expire = datetime.utcnow() + timedelta(minutes=15)to_encode.update({"exp": expire})encoded_jwt = jwt.encode(to_encode, SECRET_KEY, algorithm=ALGORITHM)return encoded_jwtasync def get_current_user(token: str = Depends(oauth2_scheme)):credentials_exception = HTTPException(status_code=status.HTTP_401_UNAUTHORIZED,detail="Could not validate credentials",headers={"WWW-Authenticate": "Bearer"},)try:payload = jwt.decode(token, SECRET_KEY, algorithms=[ALGORITHM])user_id = payload.get("sub")if user_id is None:raise credentials_exceptionpermissions = payload.get("permissions", [])username = payload.get("username")is_admin = payload.get("is_admin", False)roles = payload.get("roles", [])# 返回一个包含所需信息的字典return {"user_id": user_id,"username": username,"permissions": permissions,"is_admin": is_admin,"roles": roles}except JWTError:raise credentials_exceptionasync def auth_middleware(request: Request, call_next):try:token = request.headers.get('Authorization')if token and token.startswith('Bearer '):token = token.split(' ')[1]payload = jwt.decode(token, SECRET_KEY, algorithms=[ALGORITHM])request.state.user_id = payload.get("sub")request.state.permissions = payload.get("permissions", [])except JWTError:passresponse = await call_next(request)return response
这几个函数共同构成了一个 FastAPI 的 JWT 认证系统,分别负责 令牌生成、用户认证和请求拦截。以下是它们的详细作用:
1. create_access_token
- JWT 令牌生成
作用:
生成一个包含用户信息的 JWT(JSON Web Token),用于身份验证和授权。
关键逻辑:
-
接收一个字典
data
(包含用户信息如user_id
、permissions
等)。 -
设置令牌过期时间(默认 15 分钟,或通过
expires_delta
自定义)。 -
使用
SECRET_KEY
和ALGORITHM
(如 HS256)签名令牌。
示例输出:
token = create_access_token({"sub": "123", "username": "john"})
# 类似:eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJzdWIiOiIxMjMiLCJleHAiOjE2...
2. get_current_user
- 当前用户认证
作用:
解析请求中的 JWT 令牌,验证用户身份并返回用户信息。
关键逻辑:
-
通过
OAuth2PasswordBearer
自动从请求头Authorization: Bearer <token>
提取令牌。 -
解码令牌并验证签名(使用
SECRET_KEY
)。 -
提取关键字段(如
user_id
、permissions
、is_admin
)。 -
如果令牌无效或过期,抛出
401 Unauthorized
错误。
典型用法:
@app.get("/protected")
async def protected_route(user: dict = Depends(get_current_user)):return {"user": user}
3. auth_middleware
- 认证中间件
作用:
拦截所有请求,解析 JWT 令牌并将用户信息注入到 request.state
中,供后续路由或中间件使用。
关键逻辑:
-
从请求头提取
Bearer
令牌。 -
解码令牌并提取
user_id
和permissions
,存入request.state
。 -
如果令牌无效,静默跳过(不中断请求,适合非强制校验的场景)。
与 get_current_user
的区别:
特性 | auth_middleware | get_current_user |
---|---|---|
作用范围 | 全局(所有请求) | 单个路由(需显式声明 Depends ) |
认证严格性 | 静默失败(仅注入数据) | 严格失败(401 错误) |
典型用途 | 日志、基础用户信息注入 | 需要强认证的路由 |
协作流程
-
用户登录
-
调用
create_access_token
生成令牌并返回给前端。
-
-
前端请求
-
在请求头中添加
Authorization: Bearer <token>
。
-
-
请求处理
-
auth_middleware
拦截请求,解析令牌并注入用户信息到request.state
。 -
路由层可通过
get_current_user
进一步严格校验(如需要权限的接口)。
-
安全注意事项
-
密钥管理
-
SECRET_KEY
必须保密,且不要硬编码在代码中(从环境变量读取)。
-
-
令牌有效期
-
生产环境中
ACCESS_TOKEN_EXPIRE_MINUTES
不宜过长(通常 15-60 分钟)。
-
-
算法选择
-
推荐使用
HS256
(对称加密)或RS256
(非对称加密)。
-
-
中间件静默失败
-
auth_middleware
不强制拦截无效令牌,需确保关键路由额外依赖get_current_user
。
-
扩展建议
-
刷新令牌:添加
refresh_token
机制实现无感续期。 -
权限校验:结合
permission_required
(你之前的代码)实现细粒度控制。 -
日志记录:在中间件中记录用户操作日志。
这些函数共同构建了一个完整的 JWT 认证体系,覆盖了从令牌生成到请求处理的完整链路。
然后将这个认证中间件添加到接收HTTP请求之前
然后在路由中添加权限名称
在 FastAPI 中,_=permission_required("abilities_read")
是一种 依赖注入(Dependency Injection) 的用法,用于在路由处理函数执行前进行 权限校验。以下是详细解释:
1. 代码含义
-
permission_required("abilities_read")
这是一个依赖项工厂函数(你之前定义的),它会检查当前用户是否拥有abilities_read
权限。 -
_=
将依赖项的返回值赋值给变量_
(下划线是 Python 中约定用于“忽略此变量”的命名方式),表示我们不需要在函数体内使用这个返回值。
2. 执行流程
-
请求到达路由
当客户端访问GET /api/abilities
时,FastAPI 会先执行permission_required("abilities_read")
的校验逻辑。 -
权限校验
-
如果用户 有权限:继续执行
get_abilities
函数体。 -
如果用户 无权限:直接返回
403 Forbidden
错误(由permission_required
抛出)。
-
-
处理业务逻辑
只有通过权限校验后,才会调用get_all_competency_items()
获取数据并返回。
3. 为什么用 _=
?
-
明确忽略返回值:
permission_required
可能返回某些值(如用户信息),但当前路由不需要使用它,用_
表示“故意忽略”。 -
代码可读性:
明确告诉其他开发者:“这里有一个权限检查,但我不需要它的结果”。
前端:基于 Vuex 的前端状态管理 + JWT 认证体系
import axios from 'axios'
import { API_URLS } from '@/config/api'const state = {token: localStorage.getItem('token') || null,user: null,permissions: [],role: null
}const mutations = {SET_TOKEN(state, token) {state.token = tokenlocalStorage.setItem('token', token)axios.defaults.headers.common['Authorization'] = `Bearer ${token}`},CLEAR_TOKEN(state) {state.token = nulllocalStorage.removeItem('token')delete axios.defaults.headers.common['Authorization']},SET_USER(state, user) {state.user = user},SET_PERMISSIONS(state, permissions) {state.permissions = permissions},SET_ROLE(state, role) {state.role = role}
}const actions = {async login({ commit }, { username, password }) {const response = await axios.post(API_URLS.LOGIN, { username, password })commit('SET_TOKEN', response.data.access_token)commit('SET_PERMISSIONS', response.data.permissions)return response.data},async fetchPermissions({ commit, state }) {if (!state.token) returnconst response = await axios.get(API_URLS.PERMISSIONS)commit('SET_PERMISSIONS', response.data.permissions)return response.data},logout({ commit }) {commit('CLEAR_TOKEN')commit('SET_USER', null)commit('SET_PERMISSIONS', [])commit('SET_ROLE', null)},async getPermissions({ commit }) {const response = await axios.get('/auth/permissions')commit('SET_PERMISSIONS', response.data)return response.data}
}export default {namespaced: true,state,mutations,actions
}
import { createStore } from 'vuex'
import user from './modules/user' // 导入user模块export default createStore({state: {userId: '',isAuthenticated: false},mutations: {setUserId(state, id) {state.userId = parseInt(id)state.isAuthenticated = !!id}},modules: {user // 注册user模块},// 添加持久化plugins: [store => {// 初始化时从 localStorage 获取状态const userId = localStorage.getItem('userId')if (userId) {store.commit('setUserId', userId)}// 订阅 mutationstore.subscribe((mutation, state) => {if (mutation.type === 'setUserId') {localStorage.setItem('userId', state.userId)}})}]
})
这两部分代码共同构成了一个 基于 Vuex 的前端状态管理 + JWT 认证体系,主要处理用户认证、权限管理和状态持久化。以下是详细解析:
1. user
模块(核心功能)
作用:管理用户登录状态、令牌(Token)、权限和角色信息。
const state = {token: localStorage.getItem('token') || null, // 从本地存储初始化Tokenuser: null, // 用户信息permissions: [], // 权限列表role: null // 用户角色
}
关键方法:
方法/属性 | 功能说明 |
---|---|
mutations.SET_TOKEN | 存储Token到Vuex和localStorage,并设置Axios全局请求头(Authorization ) |
mutations.CLEAR_TOKEN | 退出登录时清除Token和Axios请求头 |
actions.login | 提交登录请求,保存Token和权限到Vuex |
actions.logout | 清理所有用户状态(Token、用户信息、权限等) |
actions.fetchPermissions | 主动获取用户权限(需Token有效) |
流程示例:
-
用户登录
-
权限控制
-
前端根据
state.permissions
动态渲染菜单/按钮。 -
每次API请求自动携带JWT(通过Axios拦截器)。
-
2. index.js
(Vuex主仓库)
作用:全局状态管理 + 模块整合 + 持久化。
export default createStore({state: {userId: '', // 全局用户IDisAuthenticated: false // 认证状态},modules: {user // 注册user模块(嵌套状态)},plugins: [ /* 持久化逻辑 */ ]
})
关键设计:
-
模块化
-
将用户相关状态(
user
模块)与全局状态分离,避免命名冲突。 -
通过
namespaced: true
启用模块命名空间。
-
-
状态持久化
-
初始化时:从
localStorage
读取userId
恢复登录状态。 -
状态变化时:通过
store.subscribe
监听setUserId
突变,自动同步到localStorage
。
-
-
认证状态同步
-
isAuthenticated
由userId
自动计算得出(!!id
转为布尔值)。
-
相关文章:
fastapi+vue中的用户权限管理设计
数据库设计:RBAC数据模型 这是一个典型的基于SQLAlchemy的RBAC权限系统数据模型实现,各模型分工明确,共同构成完整的权限管理系统。 图解说明: 实体关系: 用户(USER)和角色(ROLE)通过 USER_ROLE 中间表实现多对多关系…...
Space Engineers 太空工程师 [DLC 解锁] [Steam] [Windows]
Space Engineers 太空工程师 [DLC 解锁] [Steam] [Windows] 需要有游戏正版基础本体,安装路径不能带有中文,或其它非常规拉丁字符; DLC 版本 至最新全部 DLC 后续可能无法及时更新文章,具体最新版本见下载文件说明 DLC 解锁列表&…...
随机变量数字特征
主要介绍一维随机变量期望和方差、二维随机变量期望和方差、以及协方差相关公式,及推导。 一维随机变量 以一个抛硬币的场景作为例子,如下: 抛掷两枚均匀硬币,如果两枚都是正面向上,则赢得2元,否则就输掉…...
C++总结01-类型相关
一、数据存储 1.程序数据段 • 静态(全局)数据区:全局变量、静态变量 • 堆内存:程序员手动分配、手动释放 • 栈内存:编译器自动分配、自动释放 • 常量区:编译时大小、值确定不可修改 2.程序代码段 •…...
【多线程】七、POSIX信号量 环形队列的生产者消费者模型
文章目录 Ⅰ. 信号量一、POSIX 信号量的概念二、POSIX 信号量的类型区别三、POSIX 信号量与 SystemV 信号量的区别Ⅱ. 线程信号量基本原理一、为什么要引入信号量❓二、PV 操作三、POSIX 信号量的实现原理四、CAS操作介绍Ⅲ. POSIX未命名信号量接口一、初始化无名信号量二、销毁…...
二维码批量识别—混乱多张二维码识别-物品分拣—-未来之窗-仙盟创梦IDE
仙盟模型 用途 精准分拣:快速准确识别物品上复杂或多个二维码,依据码中信息(如目的地、品类等)实现物品自动化分拣,提高分拣效率与准确性。库存管理:识别入库、出库物品二维码,更新库存数据&am…...
《TensorFlow 与 TensorFlow Lite:协同驱动 AI 应用全景》
《TensorFlow 与 TensorFlow Lite:协同驱动 AI 应用全景》 摘要 :在机器学习技术浪潮中,TensorFlow 与 TensorFlow Lite 作为 Google 技术栈的核心组件,分别占据云端训练与端侧部署的关键位置。本文将系统梳理二者架构特性、功能…...
Spring AI 实战:第三章、Spring AI结构化输出之告别杂乱无章
引言:当程序员遇上剧荒 “周末看什么?” 这个看似简单的问题,往往能让我们在各大影视平台间反复横跳半小时,最后无奈选择重刷《老友记》。本期让我们用技术解决这个"世纪难题":让大模型成为你的私人影视推荐…...
ros2 humble 控制真实机械臂(以lerobot为例)
基础版 0.确保串口访问权限 sudo chmod 666 /dev/ttyARM0 # 确保串口访问权限 1.下载 lerobot 驱动功能包 git clone https://gitee.com/kong-yue1/lerobot_devices.git 2.编写控制节点(完整代码) 主要功能是与 Feetech 电机总线进行通信&#…...
REINFORCE蒙特卡罗策略梯度算法详解:python从零实现
🧠 向所有学习者致敬! “学习不是装满一桶水,而是点燃一把火。” —— 叶芝 我的博客主页: https://lizheng.blog.csdn.net 🌐 欢迎点击加入AI人工智能社区! 🚀 让我们一起努力,共创…...
模拟SIP终端向Freeswitch注册用户
1、简介 使用go语言编写一个程序,模拟SIP-T58终端在Freeswitch上注册用户 2、思路 以客户端向服务端Freeswitch发起REGISTER请求,告知服务器当前的联系地址构造SIP REGISTER请求 创建UDP连接,连接到Freeswitch的5060端口发送初始的REGISTER请…...
Elasticsearch 中的索引模板:如何使用可组合模板
作者:来自 Elastic Kofi Bartlett 探索可组合模板以及如何创建它们。 更多阅读: Elasticsearch:可组合的 Index templates - 7.8 版本之后 想获得 Elastic 认证吗?查看下一期 Elasticsearch Engineer 培训的时间! El…...
一篇文章看懂时间同步服务
Linux 系统时间与时区管理 一、时间与时钟类型 时钟类型说明管理工具系统时钟由 Linux 内核维护的软件时钟,基于时区配置显示时间timedatectl硬件时钟 (RTC)主板上的物理时钟,通常以 UTC 或本地时间存储,用于系统启动时初始化时间hwclock …...
Mysql常用语句汇总
Mysql语句分类 DDL: 数据定义语言,用来定义数据库对象(数据库、表、字段)DML: 数据操作语言,用来对数据库表中的数据进行增删改DQL: 数据查询语言,用来查询数据库中表的记录DCL: 数据控制语言,用来创建数据…...
迭代器的思想和实现细节
1. 迭代器的本质 迭代器是一种行为类似指针的对象,它可能是指针(如 std::vector 的迭代器),也可能是封装了指针的类(如 std::list 的迭代器)。如果是指针那天然就可以用下面的运算,如果是类&am…...
[Vue]编程式导航
在 Vue 中,编程式导航是通过 JavaScript 代码(而非 <router-link> 标签)动态控制路由跳转的核心方式。这个方法依赖于 Vue Router 提供的 API,能更灵活地处理复杂场景(如异步操作、条件跳转等)。 一、…...
使用Node.js搭建https服务器
一、引言 https是是http的安全版本,在http的基础上通过传输加密和身份认证保证了传输过程中的安全性。可以认为:https http tls/ssl。本文讲述使用Node.js搭建https服务器的方法。 二、编译OpenSSL 按照《Openssl在Linux下编译/交叉编译》࿰…...
网络安全:sql注入练习靶场——sqli_labs安装保姆级教程
网络安全:sql注入练习靶场——sqli_labs安装保姆级教程 前言 sqli-labs靶场是一个开源的sql注入练习的综合靶场,包含大部分sql注入漏洞以及注入方式 网络安全学习者可以通过在sqli-labs靶场练习提升对sql注入的理解,以及学习各种绕过姿势。…...
rfsoc petalinux适配调试记录
1。安装虚拟机 2.设置共享文件夹 https://xinzhi.wenda.so.com/a/1668239544201149先设置文件夹路径 vmware 12 下安装 ubuntu 16.04 后,按往常的惯例安装 vmware-tools,安装时提示建议使用 open-vm-tools,于是放弃 vmware-tools 的安装&am…...
Windows下编译WebRTC源码
一、开发环境要求 准备一台64位的win10或win11(我用的是win11)电脑。最好是一台纯净的、没有安装过其它软件的Windows主机,避免已安装的软件和库对编译造成影响。 然后最好预留超过100G的硬盘空间。因为编译WebRTC时会产生大量的临时文件需…...
【vscode】.dart文件没有错误波浪线
解决方法: 新建一个文件夹,在vscode里打开这个文件夹 在这个文件夹里新建.dart文件后打开即可出现错误波浪线...
OpenharmonyOS+RK3568,【编译烧录】
文章目录 1. 摘要 ✨2. 代码下载 📩3. 编译 🖥️4. 修改&适配 ✂️4.1 编译框架基本概念4.2 vendor & device 目录4.3 内核编译4.3.1 如何修改、适配自己的开发板? 4.4 修改外设驱动 5. 烧录&验证 📋参考 1. 摘要 ✨ …...
从零开始理解 C++ 后端编程中的分布式系统
一、什么是“分布式”? 简单来说,分布式系统就是由“多个计算机(或服务器)”组成的一个大系统,它们通过网络协作完成某个任务,就像一个“团队合作”一样。 想象你开了一家餐馆,最初只有 一个厨房 和 一个服务员,所有订单都在这里处理。随着生意变好,你需要: 开分店…...
基于SpringBoot的篮球竞赛预约平台设计与实现
1.1 研究背景 科学技术日新月异的如今,计算机在生活各个领域都占有重要的作用,尤其在信息管理方面,在这样的大背景下,学习计算机知识不仅仅是为了掌握一种技能,更重要的是能够让它真正地使用到实践中去,以…...
具身系列——PPO算法实现CartPole游戏(强化学习)
完整代码参考: https://gitee.com/chencib/ailib/blob/master/rl/ppo_cartpole.py 执行结果: 部分训练得分: (sd) D:\Dev\traditional_nn\feiai\test\rl>python ppo_cartpole_v2_succeed.py Ep: 0 | Reward: 23.0 | Running: 2…...
小程序与快应用:中国移动互联网的渐进式革命——卓伊凡的技术演进观
小程序与快应用:中国移动互联网的渐进式革命——卓伊凡的技术演进观 在知乎看到很多:“懂王”发布的要把内行笑疯了的评论,卓伊凡必须怼一下,真印证那句话,无知者无畏 一、Web与小程序的技术本质差异 1.1 浏览器渲染…...
Socket 编程 UDP
Socket 编程 UDP UDP 网络编程V1 版本 - echo serverV2 版本 - DictServerV3 版本 - 简单聊天室 补充参考内容地址转换函数关于 inet_ntoa UDP 网络编程 声明:下面代码的验证都是用Windows作为客户端的,如果你有两台云服务器可以直接粘贴我在Linux下的客…...
Lua 基础 API与 辅助库函数 中关于创建的方法用法
目录 基础 API 函数1. lua_len(L, index)2. lua_load(L, reader, data, chunkname, mode)3. lua_newstate(allocator, ud)4. lua_newtable(L)5. lua_newthread(L)6. lua_newuserdata(L, size)7. lua_next(L, index) 辅助库函数(luaL_*)8. luaL_len(L, in…...
YOLOv11改进:利用RT-DETR主干网络PPHGNetV2助力轻量化目标检测
这里写自定义目录标题 YOLOv11改进:利用RT-DETR主干网络PPHGNetV2助力轻量化目标检测1. 介绍2. 引言3. 技术背景3.1 YOLOv11概述3.2 RT-DETR与PPHGNetV23.3 相关工作 4. 应用使用场景5. 详细代码实现5.1 环境准备5.2 PPHGNetV2主干网络实现5.3 YOLOv11与PPHGNetV2集…...
centos7.0无法安装php8.2/8.3
在centos安装php8.2报错 configure: error: *** A compiler with support for C17 language features is required. 配置过程检测到你的系统编译器不支持 C17 语言特性,而 PHP 8.2 的编译需要编译器支持 C17 sudo yum update -y sudo yum install centos-releas…...
工业传动核心部件深度剖析:丝杆升降机与气缸的技术特性及选型指南
在工业自动化技术飞速发展的当下,丝杆升降机与气缸作为关键的直线传动部件,广泛应用于各类机械设备中。对于工程师而言,深入了解它们的技术特性、优缺点及适用场景,是实现高效、精准设备设计的重要前提。本文将从技术原理出发&…...
flask 获取各种请求数据:GET form-data x-www-form-urlencoded JSON headers 上传文件
在 Flask 里,能使用多种方法获取不同类型的请求数据,下面详细介绍常见请求数据的获取方式。 获取查询字符串参数(GET 请求) 查询字符串参数一般在 URL 里,以 ?key1value1&key2value2 这种形式存在。可通过 requ…...
c++_2011 NOIP 普及组 (1)
P1307 [NOIP 2011 普及组] 数字反转 P1307 [NOIP 2011 普及组] 数字反转 - 洛谷 # P1307 [NOIP 2011 普及组] 数字反转 ## 题目描述 给定一个整数 $N$,请将该数各个位上数字反转得到一个新数。新数也应满足整数的常见形式,即除非给定的原数为零&…...
信息泄露:网站敏感文件泄漏的隐形危机与防御之道
在网络安全领域,信息泄露常被称为“沉默的杀手”。攻击者无需复杂漏洞,仅通过网站无意暴露的敏感文件(如源码备份、配置文件、版本控制记录),即可获取数据库密码、API密钥甚至服务器权限。本文将深入剖析信息泄…...
C++笔记-多态(包含虚函数,纯虚函数和虚函数表等)
1.多态的概念 多态(polymorphism)的概念:通俗来说,就是多种形态。多态分为编译时多态(静态多态)和运行时多态(动态多态),这里我们重点讲运行时多态,编译时多态(静态多态)和运行时多态(动态多态)。编译时多态(静态多态)主要就是我们前面讲的函…...
2025年- H22-Lc130-206. 反转链表(链表)---java版
1.题目描述 2.思路 使用迭代法 (1)定义一个前指针 (2)然后定义两个变量 curr(head),curr.next。 (3)curr和curr.next交换位置(只要当前指针不为空,执行两两交换) 3.代码实现 /*** Definition for singly-…...
智能家居的OneNet云平台
一、声明 该项目只需要创建一个产品,然后这个产品里面包含几个设备,而不是直接创建几个产品 注意:传输数据使用到了不同的power,还有一定要手机先联网才能使用云平台 二、OneNet云平台创建 (1)Temperatur…...
二、shell脚本--变量与数据类型
1. 变量的定义与使用 定义变量:简单直接 在 Shell 里定义变量相当容易: 基本格式: variable_namevalue关键点 ❗:赋值号 的两边绝对不能有空格!这绝对是初学者最容易踩的坑之一 😨,务必留意!…...
GitHub Actions 和 GitLab CI/CD 流水线设计
以下是关于 GitHub Actions 和 GitLab CI/CD 流水线设计 的基本知识总结: 一、核心概念对比 维度GitHub ActionsGitLab CI/CD配置方式YAML 文件(.github/workflows/*.yml).gitlab-ci.yml执行环境GitHub 托管 Runner / 自托管GitLab 共享 Runner / 自托管市场生态Actions Mar…...
穿越数据森林与网络迷宫:树与图上动态规划实战指南
在 C 算法的浩瀚宇宙中,树与图就像是神秘的迷宫和茂密的森林,充满了未知与挑战。而动态规划则是我们探索其中的神奇罗盘,帮助我们找到最优路径。今天,就让我们一起深入这片神秘领域,揭开树与图上动态规划的神秘面纱&am…...
Java学习手册:Spring 生态其他组件介绍
一、微服务架构相关组件 Spring Cloud 服务注册与发现 : Eureka :由 Netflix 开源,包含 Eureka Server 和 Eureka Client 两部分。Eureka Server 作为服务注册表,接收服务实例的注册请求并管理其信息;Eureka Client 负…...
[android]MT6835 Android 移植brctl指令
说明 android默认brctl不支持showmacs选项,需要移植brctl-utils软件包 移除toybox中brctl编译 mssi/external/toybox/Android.bp 将 toybox_symlinks ["[","acpi","base64","basename","blockdev","br…...
安卓基础(悬浮窗分级菜单和弹窗)
initializeViews() 初始化 把全部的按钮都弄出来 // 主菜单按钮ImageButton mainButton floatingMenuView.findViewById(R.id.main_button);// 二级菜单按钮subButtons new ImageButton[3];subButtons[0] floatingMenuView.findViewById(R.id.sub_button_1);subButtons[1]…...
HTTP基础介绍+OSI七层参考模型+HTTP协议介绍
图片来源于网络 图片来源于网络 浏览器 Chrome:谷歌浏览器,推荐 Safari(WebKit):苹果浏览器,iOS,macOS Firefox:火狐浏览器,开源插件特别多(FireBug) IE:Wi…...
【项目实践】boost 搜索引擎
1. 项目展示 boost搜索引擎具体讲解视频 2. 项目背景 对于boost库,官方是没有提供搜索功能的,我们这个项目就是来为它添加一个站内搜索的功能。 3. 项目环境与技术栈 • 项目环境: ubuntu22.04、vscode • 技术栈: C/C、C11、S…...
接口隔离原则(ISP)
非常好,**接口隔离原则(ISP: Interface Segregation Principle)是 SOLID 五大原则中的第四个,它专门解决“一个接口太臃肿”**导致的麻烦。 我来从以下几个维度详细拆解: 🧠 什么是接口隔离原则࿱…...
Leetcode刷题记录29——矩阵置零
题源:https://leetcode.cn/problems/set-matrix-zeroes/description/?envTypestudy-plan-v2&envIdtop-100-liked 题目描述: 思路一: 💡 解题思路 本题中我们采用如下策略: 第一次遍历整个矩阵,记…...
复刻低成本机械臂 SO-ARM100 组装篇(打螺丝喽)
视频讲解: 复刻低成本机械臂 SO-ARM100 组装篇(打螺丝喽) 组装的视频有很多,参考大佬的《手把手复刻HuggingFace开源神作之Follower机械臂组装,资料已整理》_哔哩哔哩_bilibili,跟着视频做,大体…...
[更新完毕]2025东三省B题深圳杯B题数学建模挑战赛数模思路代码文章教学:LED显示屏颜色转换设计与校正
完整内容请看文章最下面的推广群 已经更新完整的文章代码 基于非线性映射与深度模型的多通道LED显示屏色彩校正 摘要 本研究聚焦于高动态色彩空间下LED显示屏的色彩映射与逐点校正问题,结合非线性回归理论与深度学习模型,构建了一套涵盖BT.2020映射、RG…...
Easy云盘总结篇-登录注册
**说在前面:该项目是跟着B站一位大佬写的,不分享源码,支持项目付费 ** 获取图形验证码 可以看到这里有2两种图形验证码,分为: type0:如上图下面那个,是完成操作后要进行注册的验证码 type1: 如…...