oauth.go 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150
  1. package oauth
  2. import (
  3. "encoding/json"
  4. "fmt"
  5. "net/http"
  6. "net/url"
  7. "github.com/yaotian/gowechat/util"
  8. "github.com/yaotian/gowechat/wxcontext"
  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. *wxcontext.Context
  20. }
  21. //NewOauth 实例化授权信息
  22. func NewOauth(context *wxcontext.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. util.CommonError
  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. urlStr := fmt.Sprintf(accessTokenURL, oauth.AppID, oauth.AppSecret, code)
  52. var response []byte
  53. response, err = util.HTTPGet(urlStr)
  54. if err != nil {
  55. return
  56. }
  57. err = json.Unmarshal(response, &result)
  58. if err != nil {
  59. return
  60. }
  61. if result.ErrCode != 0 {
  62. err = fmt.Errorf("GetUserAccessToken error : errcode=%v , errmsg=%v", result.ErrCode, result.ErrMsg)
  63. return
  64. }
  65. return
  66. }
  67. //RefreshAccessToken 刷新access_token
  68. func (oauth *Oauth) RefreshAccessToken(refreshToken string) (result ResAccessToken, err error) {
  69. urlStr := fmt.Sprintf(refreshAccessTokenURL, oauth.AppID, refreshToken)
  70. var response []byte
  71. response, err = util.HTTPGet(urlStr)
  72. if err != nil {
  73. return
  74. }
  75. err = json.Unmarshal(response, &result)
  76. if err != nil {
  77. return
  78. }
  79. if result.ErrCode != 0 {
  80. err = fmt.Errorf("GetUserAccessToken error : errcode=%v , errmsg=%v", result.ErrCode, result.ErrMsg)
  81. return
  82. }
  83. return
  84. }
  85. //CheckAccessToken 检验access_token是否有效
  86. func (oauth *Oauth) CheckAccessToken(accessToken, openID string) (b bool, err error) {
  87. urlStr := fmt.Sprintf(checkAccessTokenURL, accessToken, openID)
  88. var response []byte
  89. response, err = util.HTTPGet(urlStr)
  90. if err != nil {
  91. return
  92. }
  93. var result util.CommonError
  94. err = json.Unmarshal(response, &result)
  95. if err != nil {
  96. return
  97. }
  98. if result.ErrCode != 0 {
  99. b = false
  100. return
  101. }
  102. b = true
  103. return
  104. }
  105. //UserInfo 用户授权获取到用户信息
  106. type UserInfo struct {
  107. util.CommonError
  108. OpenID string `json:"openid"`
  109. Nickname string `json:"nickname"`
  110. Sex int32 `json:"sex"`
  111. Province string `json:"province"`
  112. City string `json:"city"`
  113. Country string `json:"country"`
  114. HeadImgURL string `json:"headimgurl"`
  115. Privilege []string `json:"privilege"`
  116. Unionid string `json:"unionid"`
  117. }
  118. //GetUserInfo 如果scope为 snsapi_userinfo 则可以通过此方法获取到用户基本信息
  119. func (oauth *Oauth) GetUserInfo(accessToken, openID string) (result UserInfo, err error) {
  120. urlStr := fmt.Sprintf(userInfoURL, accessToken, openID)
  121. var response []byte
  122. response, err = util.HTTPGet(urlStr)
  123. if err != nil {
  124. return
  125. }
  126. err = json.Unmarshal(response, &result)
  127. if err != nil {
  128. return
  129. }
  130. if result.ErrCode != 0 {
  131. err = fmt.Errorf("GetUserInfo error : errcode=%v , errmsg=%v", result.ErrCode, result.ErrMsg)
  132. return
  133. }
  134. return
  135. }