token.go 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160
  1. package utils
  2. import (
  3. "bytes"
  4. "encoding/json"
  5. "fmt"
  6. "io"
  7. "net/http"
  8. "time"
  9. "github.com/beego/beego/v2/server/web"
  10. )
  11. // TokenUserInfo 从token验证API返回的用户信息
  12. type TokenUserInfo struct {
  13. AccountID string `json:"accountID"`
  14. ID int64 `json:"id"`
  15. Name string `json:"name"`
  16. UserCode string `json:"userCode"`
  17. ContactNumber string `json:"contactNumber"`
  18. TokenType string `json:"token_type"`
  19. Exp int64 `json:"exp"`
  20. Iat int64 `json:"iat"`
  21. }
  22. // VerifyToken 验证token并返回用户信息
  23. func VerifyToken(token string) (*TokenUserInfo, error) {
  24. if token == "" {
  25. return nil, fmt.Errorf("token不能为空")
  26. }
  27. // 从配置文件读取token验证API URL
  28. authAPIURL, err := web.AppConfig.String("auth_api_url")
  29. if err != nil || authAPIURL == "" {
  30. // 如果配置不存在,使用默认值
  31. authAPIURL = "https://aqai.shudaodsj.com:22000/api/auth/verify"
  32. }
  33. fmt.Printf("🔐 开始验证Token...\n")
  34. fmt.Printf(" 验证API: %s\n", authAPIURL)
  35. // 构建请求体
  36. requestBody := map[string]string{
  37. "token": token,
  38. }
  39. jsonData, err := json.Marshal(requestBody)
  40. if err != nil {
  41. return nil, fmt.Errorf("序列化请求体失败: %v", err)
  42. }
  43. // 创建HTTP请求
  44. req, err := http.NewRequest("POST", authAPIURL, bytes.NewBuffer(jsonData))
  45. if err != nil {
  46. return nil, fmt.Errorf("创建请求失败: %v", err)
  47. }
  48. req.Header.Set("Content-Type", "application/json")
  49. // 重要:认证API需要在请求头中也携带 Authorization
  50. req.Header.Set("Authorization", "Bearer "+token)
  51. fmt.Printf(" 请求头 Authorization: Bearer %s...\n", token[:20])
  52. // 发送请求
  53. client := &http.Client{Timeout: 10 * time.Second}
  54. resp, err := client.Do(req)
  55. if err != nil {
  56. fmt.Printf("❌ 请求token验证API失败: %v\n", err)
  57. return nil, fmt.Errorf("请求token验证API失败: %v", err)
  58. }
  59. defer resp.Body.Close()
  60. // 读取响应体
  61. body, err := io.ReadAll(resp.Body)
  62. if err != nil {
  63. fmt.Printf("❌ 读取响应失败: %v\n", err)
  64. return nil, fmt.Errorf("读取响应失败: %v", err)
  65. }
  66. fmt.Printf(" 响应状态码: %d\n", resp.StatusCode)
  67. fmt.Printf(" 响应内容: %s\n", string(body))
  68. // 检查HTTP状态码
  69. if resp.StatusCode != http.StatusOK {
  70. return nil, fmt.Errorf("token验证失败,状态码: %d, 响应: %s", resp.StatusCode, string(body))
  71. }
  72. // 解析响应
  73. var userInfo TokenUserInfo
  74. if err := json.Unmarshal(body, &userInfo); err != nil {
  75. fmt.Printf("❌ 解析响应失败: %v\n", err)
  76. return nil, fmt.Errorf("解析token验证响应失败: %v", err)
  77. }
  78. // 检查token是否过期
  79. if userInfo.Exp > 0 && time.Now().Unix() > userInfo.Exp {
  80. fmt.Printf("❌ Token已过期 (exp: %d, now: %d)\n", userInfo.Exp, time.Now().Unix())
  81. return nil, fmt.Errorf("token已过期")
  82. }
  83. fmt.Printf("✅ Token验证成功\n")
  84. return &userInfo, nil
  85. }
  86. // GetUserInfoFromToken 从请求头中获取token并验证,返回用户信息
  87. func GetUserInfoFromToken(headerFunc func(string) string) (*TokenUserInfo, error) {
  88. // 尝试从多个可能的header字段获取token
  89. token := headerFunc("token")
  90. fmt.Print("token", token)
  91. if token == "" {
  92. token = headerFunc("Token")
  93. }
  94. if token == "" {
  95. token = headerFunc("Authorization")
  96. // 如果是Bearer token格式,去掉"Bearer "前缀
  97. if len(token) > 7 && token[:7] == "Bearer " {
  98. token = token[7:]
  99. }
  100. }
  101. if token == "" {
  102. return nil, fmt.Errorf("请求头中未找到token")
  103. }
  104. return VerifyToken(token)
  105. }
  106. // GetUserInfoFromContext 从Beego Context中获取已验证的用户信息
  107. // 该函数假定中间件已经验证过token并将用户信息存储在context中
  108. func GetUserInfoFromContext(input interface{}) (*TokenUserInfo, error) {
  109. // 添加调试信息
  110. fmt.Printf("\n🔍 [GetUserInfoFromContext] 开始解析用户信息\n")
  111. fmt.Printf(" 输入类型: %T\n", input)
  112. fmt.Printf(" 输入值: %+v\n", input)
  113. // 检查input是否为nil
  114. if input == nil {
  115. fmt.Printf("❌ [GetUserInfoFromContext] input为nil\n\n")
  116. return nil, fmt.Errorf("未找到用户信息,context中userInfo为nil,请确保已经过token认证")
  117. }
  118. // 从context中获取userInfo
  119. userInfo, ok := input.(*TokenUserInfo)
  120. if !ok {
  121. fmt.Printf("❌ [GetUserInfoFromContext] 类型断言失败,期望 *TokenUserInfo,实际得到 %T\n\n", input)
  122. return nil, fmt.Errorf("用户信息类型错误,期望 *TokenUserInfo,实际得到 %T", input)
  123. }
  124. if userInfo == nil {
  125. fmt.Printf("❌ [GetUserInfoFromContext] userInfo为nil\n\n")
  126. return nil, fmt.Errorf("用户信息为空")
  127. }
  128. fmt.Printf("✅ [GetUserInfoFromContext] 成功解析用户信息\n")
  129. fmt.Printf(" - AccountID: %s\n", userInfo.AccountID)
  130. fmt.Printf(" - ID: %d\n", userInfo.ID)
  131. fmt.Printf(" - Name: %s\n\n", userInfo.Name)
  132. return userInfo, nil
  133. }