oauth.go 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145
  1. package oauth
  2. import (
  3. "dashoo.cn/opms_libary/plugin/wechat/context"
  4. "encoding/json"
  5. "fmt"
  6. "github.com/gogf/gf/frame/g"
  7. "net/http"
  8. "net/url"
  9. )
  10. const (
  11. redirectOauthURL = "https://open.weixin.qq.com/connect/oauth2/authorize?appid=%s&redirect_uri=%s&response_type=code&scope=%s&state=%s#wechat_redirect"
  12. accessTokenURL = "https://api.weixin.qq.com/sns/oauth2/access_token?appid=%s&secret=%s&code=%s&grant_type=authorization_code"
  13. refreshAccessTokenURL = "https://api.weixin.qq.com/sns/oauth2/refresh_token?appid=%s&grant_type=refresh_token&refresh_token=%s"
  14. userInfoURL = "https://api.weixin.qq.com/sns/userinfo?access_token=%s&openid=%s&lang=zh_CN"
  15. checkAccessTokenURL = "https://api.weixin.qq.com/sns/auth?access_token=%s&openid=%s"
  16. )
  17. //Oauth 保存用户授权信息
  18. type Oauth struct {
  19. *context.Context
  20. }
  21. //NewOauth 实例化授权信息
  22. func NewOauth(context *context.Context) *Oauth {
  23. auth := new(Oauth)
  24. auth.Context = context
  25. return auth
  26. }
  27. //GetRedirectURL 获取跳转的url地址
  28. func (oauth *Oauth) GetRedirectURL(redirectURI, scope, state string) string {
  29. //url encode
  30. urlStr := url.QueryEscape(redirectURI)
  31. return fmt.Sprintf(redirectOauthURL, oauth.AppID, urlStr, scope, state)
  32. }
  33. //Redirect 跳转到网页授权
  34. func (oauth *Oauth) Redirect(redirectURI, scope, state string) error {
  35. location := oauth.GetRedirectURL(redirectURI, scope, state)
  36. //location 为完整地址,所以不需要request
  37. http.Redirect(oauth.Writer, oauth.Request, location, 302)
  38. return nil
  39. }
  40. // ResAccessToken 获取用户授权access_token的返回结果
  41. type ResAccessToken struct {
  42. context.WxError
  43. AccessToken string `json:"access_token"`
  44. ExpiresIn int64 `json:"expires_in"`
  45. RefreshToken string `json:"refresh_token"`
  46. OpenID string `json:"openid"`
  47. Scope string `json:"scope"`
  48. }
  49. // GetUserAccessToken 通过网页授权的code 换取access_token(区别于context中的access_token)
  50. func (oauth *Oauth) GetUserAccessToken(code string) (result ResAccessToken, err error) {
  51. url := fmt.Sprintf(accessTokenURL, oauth.AppID, oauth.AppSecret, code)
  52. response := g.Client().GetBytes(url)
  53. if err != nil {
  54. return
  55. }
  56. err = json.Unmarshal(response, &result)
  57. if err != nil {
  58. return
  59. }
  60. if result.Code != 0 {
  61. err = fmt.Errorf("GetUserAccessToken error : errcode=%v , errmsg=%v", result.Code, result.Msg)
  62. return
  63. }
  64. return
  65. }
  66. //RefreshAccessToken 刷新access_token
  67. func (oauth *Oauth) RefreshAccessToken(refreshToken string) (result ResAccessToken, err error) {
  68. url := fmt.Sprintf(refreshAccessTokenURL, oauth.AppID, refreshToken)
  69. response := g.Client().GetBytes(url)
  70. if err != nil {
  71. return
  72. }
  73. err = json.Unmarshal(response, &result)
  74. if err != nil {
  75. return
  76. }
  77. if result.Code != 0 {
  78. err = fmt.Errorf("GetUserAccessToken error : errcode=%v , errmsg=%v", result.Code, result.Msg)
  79. return
  80. }
  81. return
  82. }
  83. //CheckAccessToken 检验access_token是否有效
  84. func (oauth *Oauth) CheckAccessToken(accessToken, openID string) (b bool, err error) {
  85. url := fmt.Sprintf(checkAccessTokenURL, accessToken, openID)
  86. response := g.Client().GetBytes(url)
  87. if err != nil {
  88. return
  89. }
  90. var result context.WxError
  91. err = json.Unmarshal(response, &result)
  92. if err != nil {
  93. return
  94. }
  95. if result.Code != 0 {
  96. b = false
  97. return
  98. }
  99. b = true
  100. return
  101. }
  102. //UserInfo 用户授权获取到用户信息
  103. type UserInfo struct {
  104. context.WxError
  105. OpenID string `json:"openid"`
  106. Nickname string `json:"nickname"`
  107. Sex int `json:"sex"`
  108. Province string `json:"province"`
  109. City string `json:"city"`
  110. Country string `json:"country"`
  111. HeadImgURL string `json:"headimgurl"`
  112. Privilege []string `json:"privilege"`
  113. Unionid string `json:"unionid"`
  114. }
  115. //GetUserInfo 如果scope为 snsapi_userinfo 则可以通过此方法获取到用户基本信息
  116. func (oauth *Oauth) GetUserInfo(accessToken, openID string) (result UserInfo, err error) {
  117. url := fmt.Sprintf(userInfoURL, accessToken, openID)
  118. response := g.Client().GetBytes(url)
  119. if err != nil {
  120. return
  121. }
  122. err = json.Unmarshal(response, &result)
  123. if err != nil {
  124. return
  125. }
  126. if result.Code != 0 {
  127. err = fmt.Errorf("GetUserInfo error : errcode=%v , errmsg=%v", result.Code, result.Msg)
  128. return
  129. }
  130. return
  131. }