secure_verification.go 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226
  1. package controller
  2. import (
  3. "fmt"
  4. "net/http"
  5. "time"
  6. "github.com/QuantumNous/new-api/common"
  7. "github.com/QuantumNous/new-api/model"
  8. passkeysvc "github.com/QuantumNous/new-api/service/passkey"
  9. "github.com/QuantumNous/new-api/setting/system_setting"
  10. "github.com/gin-contrib/sessions"
  11. "github.com/gin-gonic/gin"
  12. )
  13. const (
  14. // SecureVerificationSessionKey 安全验证的 session key
  15. SecureVerificationSessionKey = "secure_verified_at"
  16. // SecureVerificationTimeout 验证有效期(秒)
  17. SecureVerificationTimeout = 300 // 5分钟
  18. )
  19. type UniversalVerifyRequest struct {
  20. Method string `json:"method"` // "2fa" 或 "passkey"
  21. Code string `json:"code,omitempty"`
  22. }
  23. type VerificationStatusResponse struct {
  24. Verified bool `json:"verified"`
  25. ExpiresAt int64 `json:"expires_at,omitempty"`
  26. }
  27. // UniversalVerify 通用验证接口
  28. // 支持 2FA 和 Passkey 验证,验证成功后在 session 中记录时间戳
  29. func UniversalVerify(c *gin.Context) {
  30. userId := c.GetInt("id")
  31. if userId == 0 {
  32. c.JSON(http.StatusUnauthorized, gin.H{
  33. "success": false,
  34. "message": "未登录",
  35. })
  36. return
  37. }
  38. var req UniversalVerifyRequest
  39. if err := c.ShouldBindJSON(&req); err != nil {
  40. common.ApiError(c, fmt.Errorf("参数错误: %v", err))
  41. return
  42. }
  43. // 获取用户信息
  44. user := &model.User{Id: userId}
  45. if err := user.FillUserById(); err != nil {
  46. common.ApiError(c, fmt.Errorf("获取用户信息失败: %v", err))
  47. return
  48. }
  49. if user.Status != common.UserStatusEnabled {
  50. common.ApiError(c, fmt.Errorf("该用户已被禁用"))
  51. return
  52. }
  53. // 检查用户的验证方式
  54. twoFA, _ := model.GetTwoFAByUserId(userId)
  55. has2FA := twoFA != nil && twoFA.IsEnabled
  56. passkey, passkeyErr := model.GetPasskeyByUserID(userId)
  57. hasPasskey := passkeyErr == nil && passkey != nil
  58. if !has2FA && !hasPasskey {
  59. common.ApiError(c, fmt.Errorf("用户未启用2FA或Passkey"))
  60. return
  61. }
  62. // 根据验证方式进行验证
  63. var verified bool
  64. var verifyMethod string
  65. switch req.Method {
  66. case "2fa":
  67. if !has2FA {
  68. common.ApiError(c, fmt.Errorf("用户未启用2FA"))
  69. return
  70. }
  71. if req.Code == "" {
  72. common.ApiError(c, fmt.Errorf("验证码不能为空"))
  73. return
  74. }
  75. verified = validateTwoFactorAuth(twoFA, req.Code)
  76. verifyMethod = "2FA"
  77. case "passkey":
  78. if !hasPasskey {
  79. common.ApiError(c, fmt.Errorf("用户未启用Passkey"))
  80. return
  81. }
  82. // Passkey 验证需要先调用 PasskeyVerifyBegin 和 PasskeyVerifyFinish
  83. // 这里只是验证 Passkey 验证流程是否已经完成
  84. // 实际上,前端应该先调用这两个接口,然后再调用本接口
  85. verified = true // Passkey 验证逻辑已在 PasskeyVerifyFinish 中完成
  86. verifyMethod = "Passkey"
  87. default:
  88. common.ApiError(c, fmt.Errorf("不支持的验证方式: %s", req.Method))
  89. return
  90. }
  91. if !verified {
  92. common.ApiError(c, fmt.Errorf("验证失败,请检查验证码"))
  93. return
  94. }
  95. // 验证成功,在 session 中记录时间戳
  96. session := sessions.Default(c)
  97. now := time.Now().Unix()
  98. session.Set(SecureVerificationSessionKey, now)
  99. if err := session.Save(); err != nil {
  100. common.ApiError(c, fmt.Errorf("保存验证状态失败: %v", err))
  101. return
  102. }
  103. // 记录日志
  104. model.RecordLog(userId, model.LogTypeSystem, fmt.Sprintf("通用安全验证成功 (验证方式: %s)", verifyMethod))
  105. c.JSON(http.StatusOK, gin.H{
  106. "success": true,
  107. "message": "验证成功",
  108. "data": gin.H{
  109. "verified": true,
  110. "expires_at": now + SecureVerificationTimeout,
  111. },
  112. })
  113. }
  114. // PasskeyVerifyAndSetSession Passkey 验证完成后设置 session
  115. // 这是一个辅助函数,供 PasskeyVerifyFinish 调用
  116. func PasskeyVerifyAndSetSession(c *gin.Context) {
  117. session := sessions.Default(c)
  118. now := time.Now().Unix()
  119. session.Set(SecureVerificationSessionKey, now)
  120. _ = session.Save()
  121. }
  122. // PasskeyVerifyForSecure 用于安全验证的 Passkey 验证流程
  123. // 整合了 begin 和 finish 流程
  124. func PasskeyVerifyForSecure(c *gin.Context) {
  125. if !system_setting.GetPasskeySettings().Enabled {
  126. c.JSON(http.StatusOK, gin.H{
  127. "success": false,
  128. "message": "管理员未启用 Passkey 登录",
  129. })
  130. return
  131. }
  132. userId := c.GetInt("id")
  133. if userId == 0 {
  134. c.JSON(http.StatusUnauthorized, gin.H{
  135. "success": false,
  136. "message": "未登录",
  137. })
  138. return
  139. }
  140. user := &model.User{Id: userId}
  141. if err := user.FillUserById(); err != nil {
  142. common.ApiError(c, fmt.Errorf("获取用户信息失败: %v", err))
  143. return
  144. }
  145. if user.Status != common.UserStatusEnabled {
  146. common.ApiError(c, fmt.Errorf("该用户已被禁用"))
  147. return
  148. }
  149. credential, err := model.GetPasskeyByUserID(userId)
  150. if err != nil {
  151. c.JSON(http.StatusOK, gin.H{
  152. "success": false,
  153. "message": "该用户尚未绑定 Passkey",
  154. })
  155. return
  156. }
  157. wa, err := passkeysvc.BuildWebAuthn(c.Request)
  158. if err != nil {
  159. common.ApiError(c, err)
  160. return
  161. }
  162. waUser := passkeysvc.NewWebAuthnUser(user, credential)
  163. sessionData, err := passkeysvc.PopSessionData(c, passkeysvc.VerifySessionKey)
  164. if err != nil {
  165. common.ApiError(c, err)
  166. return
  167. }
  168. _, err = wa.FinishLogin(waUser, *sessionData, c.Request)
  169. if err != nil {
  170. common.ApiError(c, err)
  171. return
  172. }
  173. // 更新凭证的最后使用时间
  174. now := time.Now()
  175. credential.LastUsedAt = &now
  176. if err := model.UpsertPasskeyCredential(credential); err != nil {
  177. common.ApiError(c, err)
  178. return
  179. }
  180. // 验证成功,设置 session
  181. PasskeyVerifyAndSetSession(c)
  182. // 记录日志
  183. model.RecordLog(userId, model.LogTypeSystem, "Passkey 安全验证成功")
  184. c.JSON(http.StatusOK, gin.H{
  185. "success": true,
  186. "message": "Passkey 验证成功",
  187. "data": gin.H{
  188. "verified": true,
  189. "expires_at": time.Now().Unix() + SecureVerificationTimeout,
  190. },
  191. })
  192. }