Files
HSAP/platform/as_platform/auth/users.py
Chengfang Lu e72bc061c5 feat: HSAP platform v2 — modular navigation, quality review, audit log, world model simulation
Major changes:
- New frontend (platform/web/): Vite + React 18 + TypeScript + Tailwind
- 4-module navigation: 数据送标 / 模型管理 / 车队管理 / 系统管理
- Data catalog with charts (DMS/ADAS/Lane 3-tab view)
- Quality review workflow (标注质检): Good/Fine/Bad scoring with auto-advance
- Audit enhancements: batch operations, rejection categories, Feishu notifications
- Operation audit log (操作日志)
- World model simulation studio (仿真工坊)
- Dataset version management with snapshots and diff
- ADAS 7-class dataset integration (138K images organized + compressed)
- User management with Feishu integration and pagination
- CRUD/search/filter on all pages, card layout redesign
- PIL-optimized image overlay rendering
- Auto-snapshot on build, in_review workflow stage
- Removed embedded algorithm code (now in workspace)
2026-06-03 11:40:21 +08:00

87 lines
2.8 KiB
Python

"""用户服务。"""
from __future__ import annotations
import json
from typing import Any
from sqlalchemy.orm import Session, joinedload
from as_platform.db.init_db import assign_default_role
from as_platform.db.models import Role, User
def get_user_by_id(db: Session, user_id: int) -> User | None:
return (
db.query(User)
.options(joinedload(User.roles).joinedload(Role.permissions))
.filter(User.id == user_id)
.first()
)
def upsert_feishu_user(db: Session, info: dict[str, Any]) -> User:
open_id = info.get("open_id")
user = db.query(User).filter(User.feishu_open_id == open_id).first()
if not user:
user = User(feishu_open_id=open_id)
db.add(user)
user.feishu_union_id = info.get("union_id") or user.feishu_union_id
user.feishu_user_id = info.get("user_id") or user.feishu_user_id
user.feishu_tenant_key = info.get("tenant_key") or user.feishu_tenant_key
department_ids = info.get("department_ids")
if isinstance(department_ids, list):
user.feishu_department_ids_json = json.dumps(department_ids, ensure_ascii=False)
user.name = info.get("name") or user.name
user.email = info.get("email") or user.email
user.avatar_url = info.get("avatar_url") or user.avatar_url
user.is_active = True
db.flush()
assign_default_role(db, user)
db.refresh(user)
return get_user_by_id(db, user.id) # type: ignore
def get_or_create_dev_user(db: Session, name: str = "开发用户") -> User:
user = db.query(User).filter(User.name == name, User.feishu_open_id.is_(None)).first()
if not user:
user = User(name=name, email="dev@local")
db.add(user)
db.flush()
admin = db.query(Role).filter_by(code="admin").first()
if admin:
user.roles = [admin]
db.flush()
return get_user_by_id(db, user.id) # type: ignore
def list_users(db: Session) -> list[User]:
return db.query(User).options(joinedload(User.roles)).order_by(User.id).all()
def list_users_paginated(
db: Session,
search: str = "",
role_code: str = "",
offset: int = 0,
limit: int = 20,
) -> tuple[list[User], int]:
q = db.query(User).options(joinedload(User.roles))
if search:
like = f"%{search}%"
q = q.filter((User.name.ilike(like)) | (User.email.ilike(like)))
if role_code:
q = q.join(User.roles).filter(Role.code == role_code)
total = q.count()
users = q.order_by(User.id).offset(offset).limit(limit).all()
return users, total
def set_user_roles(db: Session, user_id: int, role_codes: list[str]) -> User | None:
user = get_user_by_id(db, user_id)
if not user:
return None
roles = db.query(Role).filter(Role.code.in_(role_codes)).all()
user.roles = roles
db.flush()
return user