Files
sundynix-agentix/sundynix-gateway/internal/auth/auth_test.go
T
Blizzard 149c35c21b feat(gateway): 真实鉴权片1 —— JWT 注册/登录 + 校验中间件(后端核心)
替掉"裸 X-User-ID 头当身份"的临时方案,落地无状态 JWT 鉴权后端:

- internal/auth:JWT 签发/校验(HS256,密钥 env JWT_SECRET,仅接受 HMAC 防 alg 混淆)
  + bcrypt 密码哈希/校验。纯包,含单测。
- User 模型加 Name + PasswordHash(json:"-" 不外泄);store 加 CreateUser/GetUserByEmail/
  GetUserByID(邮箱唯一冲突 → ErrUserExists)。
- handler/auth:POST /auth/register(建用户+签发)· POST /auth/login(校验+签发,
  用户不存在与密码错同一文案防枚举)· GET /auth/me。
- middleware/auth:解析 Bearer JWT,校验通过把已验证 userID 注入上下文(非阻断)。
- userID(c) 改为优先取 JWT 注入的 uid,兜底 X-User-ID 头(前端尚未接登录,保持可用)。

验证:
- 单测:JWT 签发/解析往返、过期拒绝、篡改/非法拒绝、bcrypt 哈希校验。
- 实跑(nats+pg+gateway):注册→token+user(无密码)、重复注册 409、错密码 401、
  /auth/me 带 token 200 / 无 token 401;owner 隔离改用已验证 uid —— 带 token 建的库
  匿名/伪造 header 都看不到(JWT 用户数据归于雪花 id,header 无法臆测)。

片 2 待做:前端登录页 + 存令牌带 Bearer + 处理 401 + 去掉 header 兜底 + 保护路由。

Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
2026-06-17 16:14:21 +08:00

57 lines
1.1 KiB
Go

package auth
import (
"testing"
"time"
)
func TestJWT_RoundTrip(t *testing.T) {
tok, err := Issue("user-123")
if err != nil {
t.Fatal(err)
}
uid, err := Parse(tok)
if err != nil {
t.Fatalf("应能解析自签令牌: %v", err)
}
if uid != "user-123" {
t.Errorf("subject = %q, want user-123", uid)
}
}
func TestJWT_Expired(t *testing.T) {
tok, _ := issue("u", -time.Hour) // 已过期
if _, err := Parse(tok); err == nil {
t.Error("过期令牌应拒绝")
}
}
func TestJWT_Tampered(t *testing.T) {
tok, _ := Issue("u")
if _, err := Parse(tok + "x"); err == nil {
t.Error("被篡改令牌应拒绝")
}
if _, err := Parse("not.a.jwt"); err == nil {
t.Error("非法令牌应拒绝")
}
if _, err := Parse(""); err == nil {
t.Error("空令牌应拒绝")
}
}
func TestPassword_HashAndCheck(t *testing.T) {
hash, err := HashPassword("s3cret-pw")
if err != nil {
t.Fatal(err)
}
if hash == "s3cret-pw" || hash == "" {
t.Error("应为 bcrypt 哈希,非明文")
}
if !CheckPassword(hash, "s3cret-pw") {
t.Error("正确密码应校验通过")
}
if CheckPassword(hash, "wrong-pw") {
t.Error("错误密码应校验失败")
}
}