You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
Open-IM-Server/internal/push/getui/push.go

248 lines
6.7 KiB

package getui
3 years ago
import (
"Open_IM/internal/push"
3 years ago
"Open_IM/pkg/common/config"
"Open_IM/pkg/common/db"
"Open_IM/pkg/common/log"
"Open_IM/pkg/utils"
"bytes"
3 years ago
"crypto/sha256"
3 years ago
"errors"
3 years ago
//"crypto/sha512"
3 years ago
"encoding/hex"
3 years ago
"encoding/json"
"io/ioutil"
"net/http"
"strconv"
"time"
)
var (
GetuiClient *Getui
3 years ago
TokenExpireError = errors.New("token expire")
3 years ago
)
3 years ago
const (
3 years ago
PushURL = "/push/single/alias"
3 years ago
AuthURL = "/auth"
)
3 years ago
func init() {
GetuiClient = newGetuiClient()
}
type Getui struct{}
type GetuiCommonResp struct {
Code int `json:"code"`
Msg string `json:"msg"`
Data interface{} `json:"data"`
}
type AuthReq struct {
Sign string `json:"sign"`
Timestamp string `json:"timestamp"`
Appkey string `json:"appkey"`
}
type AuthResp struct {
ExpireTime string `json:"expire_time"`
Token string `json:"token"`
}
type PushReq struct {
RequestID string `json:"request_id"`
Audience struct {
3 years ago
Alias []string `json:"alias"`
3 years ago
} `json:"audience"`
3 years ago
PushMessage struct {
Notification Notification `json:"notification,omitempty"`
Transmission string `json:"transmission,omitempty"`
3 years ago
} `json:"push_message"`
3 years ago
PushChannel struct {
Ios Ios `json:"ios"`
Android Android `json:"android"`
} `json:"push_channel"`
}
type Ios struct {
Aps struct {
Sound string `json:"sound"`
Alert Alert `json:"alert"`
} `json:"aps"`
}
type Alert struct {
Title string `json:"title"`
Body string `json:"body"`
}
type Android struct {
Ups struct {
Notification Notification `json:"notification"`
2 years ago
Options Options `json:"options"`
3 years ago
} `json:"ups"`
3 years ago
}
type Notification struct {
Title string `json:"title"`
Body string `json:"body"`
ClickType string `json:"click_type"`
}
2 years ago
type Options struct {
HW struct {
DefaultSound bool `json:"/message/android/notification/default_sound"`
ChannelID string `json:"/message/android/notification/channel_id"`
Sound string `json:"/message/android/notification/sound"`
Importance string `json:"/message/android/notification/importance"`
} `json:"HW"`
XM struct {
ChannelID string `json:"/extra.channel_id"`
} `json:""`
}
3 years ago
type PushResp struct {
}
func newGetuiClient() *Getui {
return &Getui{}
}
func (g *Getui) Push(userIDList []string, alert, detailContent, operationID string, opts push.PushOpts) (resp string, err error) {
3 years ago
token, err := db.DB.GetGetuiToken()
3 years ago
log.NewDebug(operationID, utils.GetSelfFuncName(), "token", token)
3 years ago
if err != nil {
3 years ago
log.NewError(operationID, utils.OperationIDGenerator(), "GetGetuiToken failed", err.Error())
3 years ago
}
if token == "" || err != nil {
3 years ago
token, err = g.getTokenAndSave2Redis(operationID)
3 years ago
if err != nil {
3 years ago
log.NewError(operationID, utils.GetSelfFuncName(), "getTokenAndSave2Redis failed", err.Error())
return "", utils.Wrap(err, "")
3 years ago
}
}
pushReq := PushReq{
RequestID: utils.OperationIDGenerator(),
Audience: struct {
3 years ago
Alias []string `json:"alias"`
}{Alias: []string{userIDList[0]}},
3 years ago
}
3 years ago
pushReq.PushMessage.Notification = Notification{
3 years ago
Title: alert,
Body: alert,
3 years ago
ClickType: "startapp",
3 years ago
}
3 years ago
pushReq.PushChannel.Ios.Aps.Sound = "default"
pushReq.PushChannel.Ios.Aps.Alert = Alert{
Title: alert,
Body: alert,
}
pushReq.PushChannel.Android.Ups.Notification = Notification{
Title: alert,
Body: alert,
ClickType: "startapp",
}
2 years ago
pushReq.PushChannel.Android.Ups.Options = Options{
HW: struct {
DefaultSound bool `json:"/message/android/notification/default_sound"`
ChannelID string `json:"/message/android/notification/channel_id"`
Sound string `json:"/message/android/notification/sound"`
Importance string `json:"/message/android/notification/importance"`
}{ChannelID: "RingRing4", Sound: "/raw/ring001", Importance: "importance"},
XM: struct {
ChannelID string `json:"/extra.channel_id"`
}{ChannelID: "Default"},
}
3 years ago
pushResp := PushResp{}
3 years ago
err = g.request(PushURL, pushReq, token, &pushResp, operationID)
3 years ago
switch err {
case TokenExpireError:
3 years ago
token, err = g.getTokenAndSave2Redis(operationID)
3 years ago
if err != nil {
log.NewError(operationID, utils.GetSelfFuncName(), "getTokenAndSave2Redis failed, ", err.Error())
3 years ago
} else {
log.NewInfo(operationID, utils.GetSelfFuncName(), "getTokenAndSave2Redis: ", token)
3 years ago
}
}
3 years ago
if err != nil {
3 years ago
return "", utils.Wrap(err, "push failed")
3 years ago
}
respBytes, err := json.Marshal(pushResp)
3 years ago
return string(respBytes), utils.Wrap(err, "")
}
3 years ago
func (g *Getui) Auth(operationID string, timeStamp int64) (token string, expireTime int64, err error) {
3 years ago
log.NewInfo(operationID, utils.GetSelfFuncName(), config.Config.Push.Getui.AppKey, timeStamp, config.Config.Push.Getui.MasterSecret)
3 years ago
h := sha256.New()
3 years ago
h.Write([]byte(config.Config.Push.Getui.AppKey + strconv.Itoa(int(timeStamp)) + config.Config.Push.Getui.MasterSecret))
sum := h.Sum(nil)
sign := hex.EncodeToString(sum)
log.NewInfo(operationID, utils.GetSelfFuncName(), "sha256 result", sign)
3 years ago
reqAuth := AuthReq{
Sign: sign,
Timestamp: strconv.Itoa(int(timeStamp)),
3 years ago
Appkey: config.Config.Push.Getui.AppKey,
3 years ago
}
respAuth := AuthResp{}
3 years ago
err = g.request(AuthURL, reqAuth, "", &respAuth, operationID)
3 years ago
if err != nil {
return "", 0, err
}
log.NewInfo(operationID, utils.GetSelfFuncName(), "result: ", respAuth)
3 years ago
expire, err := strconv.Atoi(respAuth.ExpireTime)
return respAuth.Token, int64(expire), err
}
3 years ago
func (g *Getui) request(url string, content interface{}, token string, returnStruct interface{}, operationID string) error {
3 years ago
con, err := json.Marshal(content)
if err != nil {
return err
}
client := &http.Client{}
2 years ago
log.Debug(operationID, utils.GetSelfFuncName(), "json:", string(con), "token:", token)
3 years ago
req, err := http.NewRequest("POST", config.Config.Push.Getui.PushUrl+url, bytes.NewBuffer(con))
3 years ago
if err != nil {
return err
}
3 years ago
if token != "" {
3 years ago
req.Header.Set("token", token)
3 years ago
}
3 years ago
req.Header.Set("content-type", "application/json")
resp, err := client.Do(req)
if err != nil {
return err
}
defer resp.Body.Close()
result, err := ioutil.ReadAll(resp.Body)
if err != nil {
return err
}
2 years ago
log.NewDebug(operationID, "getui", utils.GetSelfFuncName(), "resp, ", string(result))
3 years ago
commonResp := GetuiCommonResp{}
commonResp.Data = returnStruct
3 years ago
if err := json.Unmarshal(result, &commonResp); err != nil {
3 years ago
return err
}
3 years ago
if commonResp.Code == 10001 {
return TokenExpireError
}
3 years ago
return nil
}
3 years ago
func (g *Getui) getTokenAndSave2Redis(operationID string) (token string, err error) {
token, expireTime, err := g.Auth(operationID, time.Now().UnixNano()/1e6)
if err != nil {
return "", utils.Wrap(err, "Auth failed")
}
log.NewDebug(operationID, "getui", utils.GetSelfFuncName(), token, expireTime, err)
err = db.DB.SetGetuiToken(token, 60*60*23)
if err != nil {
return "", utils.Wrap(err, "Auth failed")
}
return token, nil
}