package weixin import ( "fmt" "github.com/tidwall/gjson" "io" "net/http" "sync" "time" ) const ( code2AccessTokenUrl string = "https://api.weixin.qq.com/sns/oauth2/access_token" jscode2SessionUrl string = "https://api.weixin.qq.com/sns/jscode2session" accessTokenUrl string = "https://api.weixin.qq.com/cgi-bin/token" userPhoneNumberUrl string = "https://api.weixin.qq.com/wxa/business/getuserphonenumber" getWxACodeUnLimitUrl string = "https://api.weixin.qq.com/wxa/getwxacodeunlimit" code2UserinfoUrl string = "https://api.weixin.qq.com/sns/userinfo" oaQrCodeCreateUrl string = "https://api.weixin.qq.com/cgi-bin/qrcode/create" userInfoByOpenid string = "https://api.weixin.qq.com/cgi-bin/user/info" CreateOaMenu string = "https://api.weixin.qq.com/cgi-bin/menu/create" QueryOaMenu string = "https://api.weixin.qq.com/cgi-bin/get_current_selfmenu_info" DeleteOaMenu string = "https://api.weixin.qq.com/cgi-bin/menu/delete" UploadOaMedia string = "https://api.weixin.qq.com/cgi-bin/media/upload" ) const ( OaMenuTypeClick = "click" OaMenuTypeView = "view" ) type UserInfo struct { Openid string Unionid string AccessToken string Nickname string HeadUrl string Sex int //1:为男性 2位女性 Country string City string Province string } type BaseSdk struct { appid string secret string accessToken string expireIn int64 lock sync.Mutex } type OaMenuButton struct { Type string `json:"type,omitempty"` Name string `json:"name"` Key string `json:"key,omitempty"` Url string `json:"url,omitempty"` SubButton []*OaMenuButton `json:"sub_button"` } func (o *BaseSdk) getAccessToken() (string, error) { o.lock.Lock() defer o.lock.Unlock() if time.Now().Unix() < o.expireIn { return o.accessToken, nil } // 获取一个新token url := fmt.Sprintf("%s?grant_type=client_credential&appid=%s&secret=%s", accessTokenUrl, o.appid, o.secret) res, err := http.Get(url) if err != nil { return "", err } defer res.Body.Close() body, err := io.ReadAll(res.Body) if err != nil { return "", err } g := gjson.ParseBytes(body) accessToken := g.Get("access_token").String() if accessToken == "" { return "", fmt.Errorf("%d:%s", g.Get("errcode").Int(), g.Get("errmsg").String()) } o.accessToken = accessToken o.expireIn = time.Now().Unix() + g.Get("expires_in").Int() - 10 return o.accessToken, nil } func (o *BaseSdk) getUserInfoByCode(code string, auth bool) (*UserInfo, error) { url := fmt.Sprintf("%s?appid=%s&secret=%s&code=%s&grant_type=authorization_code", code2AccessTokenUrl, o.appid, o.secret, code) res, err := http.Get(url) if err != nil { return nil, err } defer res.Body.Close() body, err := io.ReadAll(res.Body) if err != nil { return nil, err } g := gjson.ParseBytes(body) errcode := g.Get("errcode").Int() if errcode != 0 { return nil, fmt.Errorf("%d:%s", errcode, g.Get("errmsg").String()) } var user UserInfo user.Unionid = g.Get("unionid").String() user.AccessToken = g.Get("access_token").String() user.Openid = g.Get("openid").String() if auth { err = o.getUserInfo(&user) if err != nil { return nil, err } } return &user, nil } func (o *BaseSdk) getUserInfo(user *UserInfo) error { url := fmt.Sprintf("%s?access_token=%s&openid=%s&lang=zh_CN", code2UserinfoUrl, user.AccessToken, user.Openid) res, err := http.Get(url) if err != nil { return err } defer res.Body.Close() body, err := io.ReadAll(res.Body) if err != nil { return err } g := gjson.ParseBytes(body) errcode := g.Get("errcode").Int() if errcode != 0 { return fmt.Errorf("%d:%s", errcode, g.Get("errmsg").String()) } user.Unionid = g.Get("unionid").String() user.Nickname = g.Get("nickname").String() user.HeadUrl = g.Get("headimgurl").String() user.Province = g.Get("province").String() user.City = g.Get("city").String() user.Country = g.Get("country").String() user.Sex = int(g.Get("sex").Int()) return nil }