mirror of
				https://github.com/openimsdk/open-im-server.git
				synced 2025-10-26 05:02:11 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			486 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			486 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright © 2023 OpenIM. All rights reserved.
 | |
| //
 | |
| // Licensed under the Apache License, Version 2.0 (the "License");
 | |
| // you may not use this file except in compliance with the License.
 | |
| // You may obtain a copy of the License at
 | |
| //
 | |
| //     http://www.apache.org/licenses/LICENSE-2.0
 | |
| //
 | |
| // Unless required by applicable law or agreed to in writing, software
 | |
| // distributed under the License is distributed on an "AS IS" BASIS,
 | |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | |
| // See the License for the specific language governing permissions and
 | |
| // limitations under the License.
 | |
| 
 | |
| package api
 | |
| 
 | |
| import (
 | |
| 	"encoding/base64"
 | |
| 	"encoding/json"
 | |
| 
 | |
| 	"github.com/gin-gonic/gin"
 | |
| 	"github.com/go-playground/validator/v10"
 | |
| 	"github.com/mitchellh/mapstructure"
 | |
| 
 | |
| 	"github.com/openimsdk/open-im-server/v3/pkg/apistruct"
 | |
| 	"github.com/openimsdk/open-im-server/v3/pkg/authverify"
 | |
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/config"
 | |
| 	"github.com/openimsdk/open-im-server/v3/pkg/common/webhook"
 | |
| 	"github.com/openimsdk/open-im-server/v3/pkg/rpcli"
 | |
| 	"github.com/openimsdk/protocol/constant"
 | |
| 	"github.com/openimsdk/protocol/msg"
 | |
| 	"github.com/openimsdk/protocol/sdkws"
 | |
| 	"github.com/openimsdk/tools/a2r"
 | |
| 	"github.com/openimsdk/tools/apiresp"
 | |
| 	"github.com/openimsdk/tools/errs"
 | |
| 	"github.com/openimsdk/tools/log"
 | |
| 	"github.com/openimsdk/tools/mcontext"
 | |
| 	"github.com/openimsdk/tools/utils/datautil"
 | |
| 	"github.com/openimsdk/tools/utils/idutil"
 | |
| 	"github.com/openimsdk/tools/utils/jsonutil"
 | |
| 	"github.com/openimsdk/tools/utils/timeutil"
 | |
| )
 | |
| 
 | |
| type MessageApi struct {
 | |
| 	Client        msg.MsgClient
 | |
| 	userClient    *rpcli.UserClient
 | |
| 	imAdminUserID []string
 | |
| 	validate      *validator.Validate
 | |
| }
 | |
| 
 | |
| func NewMessageApi(client msg.MsgClient, userClient *rpcli.UserClient, imAdminUserID []string) MessageApi {
 | |
| 	return MessageApi{Client: client, userClient: userClient, imAdminUserID: imAdminUserID, validate: validator.New()}
 | |
| }
 | |
| 
 | |
| func (*MessageApi) SetOptions(options map[string]bool, value bool) {
 | |
| 	datautil.SetSwitchFromOptions(options, constant.IsHistory, value)
 | |
| 	datautil.SetSwitchFromOptions(options, constant.IsPersistent, value)
 | |
| 	datautil.SetSwitchFromOptions(options, constant.IsSenderSync, value)
 | |
| 	datautil.SetSwitchFromOptions(options, constant.IsConversationUpdate, value)
 | |
| }
 | |
| 
 | |
| func (m *MessageApi) newUserSendMsgReq(_ *gin.Context, params *apistruct.SendMsg) *msg.SendMsgReq {
 | |
| 	var newContent string
 | |
| 	options := make(map[string]bool, 5)
 | |
| 	switch params.ContentType {
 | |
| 	case constant.OANotification:
 | |
| 		notification := sdkws.NotificationElem{}
 | |
| 		notification.Detail = jsonutil.StructToJsonString(params.Content)
 | |
| 		newContent = jsonutil.StructToJsonString(¬ification)
 | |
| 	case constant.Text:
 | |
| 		fallthrough
 | |
| 	case constant.Picture:
 | |
| 		fallthrough
 | |
| 	case constant.Custom:
 | |
| 		fallthrough
 | |
| 	case constant.Voice:
 | |
| 		fallthrough
 | |
| 	case constant.Video:
 | |
| 		fallthrough
 | |
| 	case constant.File:
 | |
| 		fallthrough
 | |
| 	default:
 | |
| 		newContent = jsonutil.StructToJsonString(params.Content)
 | |
| 	}
 | |
| 	if params.IsOnlineOnly {
 | |
| 		m.SetOptions(options, false)
 | |
| 	}
 | |
| 	if params.NotOfflinePush {
 | |
| 		datautil.SetSwitchFromOptions(options, constant.IsOfflinePush, false)
 | |
| 	}
 | |
| 	pbData := msg.SendMsgReq{
 | |
| 		MsgData: &sdkws.MsgData{
 | |
| 			SendID:           params.SendID,
 | |
| 			GroupID:          params.GroupID,
 | |
| 			ClientMsgID:      idutil.GetMsgIDByMD5(params.SendID),
 | |
| 			SenderPlatformID: params.SenderPlatformID,
 | |
| 			SenderNickname:   params.SenderNickname,
 | |
| 			SenderFaceURL:    params.SenderFaceURL,
 | |
| 			SessionType:      params.SessionType,
 | |
| 			MsgFrom:          constant.SysMsgType,
 | |
| 			ContentType:      params.ContentType,
 | |
| 			Content:          []byte(newContent),
 | |
| 			CreateTime:       timeutil.GetCurrentTimestampByMill(),
 | |
| 			SendTime:         params.SendTime,
 | |
| 			Options:          options,
 | |
| 			OfflinePushInfo:  params.OfflinePushInfo,
 | |
| 			Ex:               params.Ex,
 | |
| 		},
 | |
| 	}
 | |
| 	return &pbData
 | |
| }
 | |
| 
 | |
| func (m *MessageApi) GetSeq(c *gin.Context) {
 | |
| 	a2r.Call(c, msg.MsgClient.GetMaxSeq, m.Client)
 | |
| }
 | |
| 
 | |
| func (m *MessageApi) PullMsgBySeqs(c *gin.Context) {
 | |
| 	a2r.Call(c, msg.MsgClient.PullMessageBySeqs, m.Client)
 | |
| }
 | |
| 
 | |
| func (m *MessageApi) RevokeMsg(c *gin.Context) {
 | |
| 	a2r.Call(c, msg.MsgClient.RevokeMsg, m.Client)
 | |
| }
 | |
| 
 | |
| func (m *MessageApi) MarkMsgsAsRead(c *gin.Context) {
 | |
| 	a2r.Call(c, msg.MsgClient.MarkMsgsAsRead, m.Client)
 | |
| }
 | |
| 
 | |
| func (m *MessageApi) MarkConversationAsRead(c *gin.Context) {
 | |
| 	a2r.Call(c, msg.MsgClient.MarkConversationAsRead, m.Client)
 | |
| }
 | |
| 
 | |
| func (m *MessageApi) GetConversationsHasReadAndMaxSeq(c *gin.Context) {
 | |
| 	a2r.Call(c, msg.MsgClient.GetConversationsHasReadAndMaxSeq, m.Client)
 | |
| }
 | |
| 
 | |
| func (m *MessageApi) SetConversationHasReadSeq(c *gin.Context) {
 | |
| 	a2r.Call(c, msg.MsgClient.SetConversationHasReadSeq, m.Client)
 | |
| }
 | |
| 
 | |
| func (m *MessageApi) ClearConversationsMsg(c *gin.Context) {
 | |
| 	a2r.Call(c, msg.MsgClient.ClearConversationsMsg, m.Client)
 | |
| }
 | |
| 
 | |
| func (m *MessageApi) UserClearAllMsg(c *gin.Context) {
 | |
| 	a2r.Call(c, msg.MsgClient.UserClearAllMsg, m.Client)
 | |
| }
 | |
| 
 | |
| func (m *MessageApi) DeleteMsgs(c *gin.Context) {
 | |
| 	a2r.Call(c, msg.MsgClient.DeleteMsgs, m.Client)
 | |
| }
 | |
| 
 | |
| func (m *MessageApi) DeleteMsgPhysicalBySeq(c *gin.Context) {
 | |
| 	a2r.Call(c, msg.MsgClient.DeleteMsgPhysicalBySeq, m.Client)
 | |
| }
 | |
| 
 | |
| func (m *MessageApi) DeleteMsgPhysical(c *gin.Context) {
 | |
| 	a2r.Call(c, msg.MsgClient.DeleteMsgPhysical, m.Client)
 | |
| }
 | |
| 
 | |
| func (m *MessageApi) getSendMsgReq(c *gin.Context, req apistruct.SendMsg) (sendMsgReq *msg.SendMsgReq, err error) {
 | |
| 	var data any
 | |
| 	log.ZDebug(c, "getSendMsgReq", "req", req.Content)
 | |
| 	switch req.ContentType {
 | |
| 	case constant.Text:
 | |
| 		data = apistruct.TextElem{}
 | |
| 	case constant.Picture:
 | |
| 		data = apistruct.PictureElem{}
 | |
| 	case constant.Voice:
 | |
| 		data = apistruct.SoundElem{}
 | |
| 	case constant.Video:
 | |
| 		data = apistruct.VideoElem{}
 | |
| 	case constant.File:
 | |
| 		data = apistruct.FileElem{}
 | |
| 	case constant.AtText:
 | |
| 		data = apistruct.AtElem{}
 | |
| 	case constant.Custom:
 | |
| 		data = apistruct.CustomElem{}
 | |
| 	case constant.MarkdownText:
 | |
| 		data = apistruct.MarkdownTextElem{}
 | |
| 	case constant.OANotification:
 | |
| 		data = apistruct.OANotificationElem{}
 | |
| 		req.SessionType = constant.NotificationChatType
 | |
| 		if err = m.userClient.GetNotificationByID(c, req.SendID); err != nil {
 | |
| 			return nil, err
 | |
| 		}
 | |
| 	default:
 | |
| 		return nil, errs.WrapMsg(errs.ErrArgs, "unsupported content type", "contentType", req.ContentType)
 | |
| 	}
 | |
| 	if err := mapstructure.WeakDecode(req.Content, &data); err != nil {
 | |
| 		return nil, errs.WrapMsg(err, "failed to decode message content")
 | |
| 	}
 | |
| 	log.ZDebug(c, "getSendMsgReq", "decodedContent", data)
 | |
| 	if err := m.validate.Struct(data); err != nil {
 | |
| 		return nil, errs.WrapMsg(err, "validation error")
 | |
| 	}
 | |
| 	return m.newUserSendMsgReq(c, &req), nil
 | |
| }
 | |
| 
 | |
| // SendMessage handles the sending of a message. It's an HTTP handler function to be used with Gin framework.
 | |
| func (m *MessageApi) SendMessage(c *gin.Context) {
 | |
| 	// Initialize a request struct for sending a message.
 | |
| 	req := apistruct.SendMsgReq{}
 | |
| 
 | |
| 	// Bind the JSON request body to the request struct.
 | |
| 	if err := c.BindJSON(&req); err != nil {
 | |
| 		// Respond with an error if request body binding fails.
 | |
| 		apiresp.GinError(c, errs.ErrArgs.WithDetail(err.Error()).Wrap())
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	// Check if the user has the app manager role.
 | |
| 	if !authverify.IsAppManagerUid(c, m.imAdminUserID) {
 | |
| 		// Respond with a permission error if the user is not an app manager.
 | |
| 		apiresp.GinError(c, errs.ErrNoPermission.WrapMsg("only app manager can send message"))
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	// Prepare the message request with additional required data.
 | |
| 	sendMsgReq, err := m.getSendMsgReq(c, req.SendMsg)
 | |
| 	if err != nil {
 | |
| 		// Log and respond with an error if preparation fails.
 | |
| 		apiresp.GinError(c, err)
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	// Set the receiver ID in the message data.
 | |
| 	sendMsgReq.MsgData.RecvID = req.RecvID
 | |
| 
 | |
| 	// Attempt to send the message using the client.
 | |
| 	respPb, err := m.Client.SendMsg(c, sendMsgReq)
 | |
| 	if err != nil {
 | |
| 		// Set the status to failed and respond with an error if sending fails.
 | |
| 		apiresp.GinError(c, err)
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	// Set the status to successful if the message is sent.
 | |
| 	var status = constant.MsgSendSuccessed
 | |
| 
 | |
| 	// Attempt to update the message sending status in the system.
 | |
| 	_, err = m.Client.SetSendMsgStatus(c, &msg.SetSendMsgStatusReq{
 | |
| 		Status: int32(status),
 | |
| 	})
 | |
| 
 | |
| 	if err != nil {
 | |
| 		// Log the error if updating the status fails.
 | |
| 		apiresp.GinError(c, err)
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	// Respond with a success message and the response payload.
 | |
| 	apiresp.GinSuccess(c, respPb)
 | |
| }
 | |
| 
 | |
| func (m *MessageApi) SendBusinessNotification(c *gin.Context) {
 | |
| 	req := struct {
 | |
| 		Key              string `json:"key"`
 | |
| 		Data             string `json:"data"`
 | |
| 		SendUserID       string `json:"sendUserID" binding:"required"`
 | |
| 		RecvUserID       string `json:"recvUserID"`
 | |
| 		RecvGroupID      string `json:"recvGroupID"`
 | |
| 		SendMsg          bool   `json:"sendMsg"`
 | |
| 		ReliabilityLevel *int   `json:"reliabilityLevel"`
 | |
| 	}{}
 | |
| 	if err := c.BindJSON(&req); err != nil {
 | |
| 		apiresp.GinError(c, errs.ErrArgs.WithDetail(err.Error()).Wrap())
 | |
| 		return
 | |
| 	}
 | |
| 	if req.RecvUserID == "" && req.RecvGroupID == "" {
 | |
| 		apiresp.GinError(c, errs.ErrArgs.WrapMsg("recvUserID and recvGroupID cannot be empty at the same time"))
 | |
| 		return
 | |
| 	}
 | |
| 	if req.RecvUserID != "" && req.RecvGroupID != "" {
 | |
| 		apiresp.GinError(c, errs.ErrArgs.WrapMsg("recvUserID and recvGroupID cannot be set at the same time"))
 | |
| 		return
 | |
| 	}
 | |
| 	var sessionType int32
 | |
| 	if req.RecvUserID != "" {
 | |
| 		sessionType = constant.SingleChatType
 | |
| 	} else {
 | |
| 		sessionType = constant.ReadGroupChatType
 | |
| 	}
 | |
| 	if req.ReliabilityLevel == nil {
 | |
| 		req.ReliabilityLevel = datautil.ToPtr(1)
 | |
| 	}
 | |
| 	if !authverify.IsAppManagerUid(c, m.imAdminUserID) {
 | |
| 		apiresp.GinError(c, errs.ErrNoPermission.WrapMsg("only app manager can send message"))
 | |
| 		return
 | |
| 	}
 | |
| 	sendMsgReq := msg.SendMsgReq{
 | |
| 		MsgData: &sdkws.MsgData{
 | |
| 			SendID:  req.SendUserID,
 | |
| 			RecvID:  req.RecvUserID,
 | |
| 			GroupID: req.RecvGroupID,
 | |
| 			Content: []byte(jsonutil.StructToJsonString(&sdkws.NotificationElem{
 | |
| 				Detail: jsonutil.StructToJsonString(&struct {
 | |
| 					Key  string `json:"key"`
 | |
| 					Data string `json:"data"`
 | |
| 				}{Key: req.Key, Data: req.Data}),
 | |
| 			})),
 | |
| 			MsgFrom:     constant.SysMsgType,
 | |
| 			ContentType: constant.BusinessNotification,
 | |
| 			SessionType: sessionType,
 | |
| 			CreateTime:  timeutil.GetCurrentTimestampByMill(),
 | |
| 			ClientMsgID: idutil.GetMsgIDByMD5(mcontext.GetOpUserID(c)),
 | |
| 			Options: config.GetOptionsByNotification(config.NotificationConfig{
 | |
| 				IsSendMsg:        req.SendMsg,
 | |
| 				ReliabilityLevel: *req.ReliabilityLevel,
 | |
| 				UnreadCount:      false,
 | |
| 			}),
 | |
| 		},
 | |
| 	}
 | |
| 	respPb, err := m.Client.SendMsg(c, &sendMsgReq)
 | |
| 	if err != nil {
 | |
| 		apiresp.GinError(c, err)
 | |
| 		return
 | |
| 	}
 | |
| 	apiresp.GinSuccess(c, respPb)
 | |
| }
 | |
| 
 | |
| func (m *MessageApi) BatchSendMsg(c *gin.Context) {
 | |
| 	var (
 | |
| 		req  apistruct.BatchSendMsgReq
 | |
| 		resp apistruct.BatchSendMsgResp
 | |
| 	)
 | |
| 	if err := c.BindJSON(&req); err != nil {
 | |
| 		apiresp.GinError(c, errs.ErrArgs.WithDetail(err.Error()).Wrap())
 | |
| 		return
 | |
| 	}
 | |
| 	if err := authverify.CheckAdmin(c, m.imAdminUserID); err != nil {
 | |
| 		apiresp.GinError(c, errs.ErrNoPermission.WrapMsg("only app manager can send message"))
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	var recvIDs []string
 | |
| 	if req.IsSendAll {
 | |
| 		var pageNumber int32 = 1
 | |
| 		const showNumber = 500
 | |
| 		for {
 | |
| 			recvIDsPart, err := m.userClient.GetAllUserIDs(c, pageNumber, showNumber)
 | |
| 			if err != nil {
 | |
| 				apiresp.GinError(c, err)
 | |
| 				return
 | |
| 			}
 | |
| 			recvIDs = append(recvIDs, recvIDsPart...)
 | |
| 			if len(recvIDsPart) < showNumber {
 | |
| 				break
 | |
| 			}
 | |
| 			pageNumber++
 | |
| 		}
 | |
| 	} else {
 | |
| 		recvIDs = req.RecvIDs
 | |
| 	}
 | |
| 	log.ZDebug(c, "BatchSendMsg nums", "nums ", len(recvIDs))
 | |
| 	sendMsgReq, err := m.getSendMsgReq(c, req.SendMsg)
 | |
| 	if err != nil {
 | |
| 		apiresp.GinError(c, err)
 | |
| 		return
 | |
| 	}
 | |
| 	for _, recvID := range recvIDs {
 | |
| 		sendMsgReq.MsgData.RecvID = recvID
 | |
| 		rpcResp, err := m.Client.SendMsg(c, sendMsgReq)
 | |
| 		if err != nil {
 | |
| 			resp.FailedIDs = append(resp.FailedIDs, recvID)
 | |
| 			continue
 | |
| 		}
 | |
| 		resp.Results = append(resp.Results, &apistruct.SingleReturnResult{
 | |
| 			ServerMsgID: rpcResp.ServerMsgID,
 | |
| 			ClientMsgID: rpcResp.ClientMsgID,
 | |
| 			SendTime:    rpcResp.SendTime,
 | |
| 			RecvID:      recvID,
 | |
| 		})
 | |
| 	}
 | |
| 	apiresp.GinSuccess(c, resp)
 | |
| }
 | |
| 
 | |
| func (m *MessageApi) SendSimpleMessage(c *gin.Context) {
 | |
| 	encodedKey, ok := c.GetQuery(webhook.Key)
 | |
| 	if !ok {
 | |
| 		apiresp.GinError(c, errs.ErrArgs.WithDetail("missing key in query").Wrap())
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	decodedData, err := base64.StdEncoding.DecodeString(encodedKey)
 | |
| 	if err != nil {
 | |
| 		apiresp.GinError(c, errs.ErrArgs.WithDetail(err.Error()).Wrap())
 | |
| 		return
 | |
| 	}
 | |
| 	var (
 | |
| 		req        apistruct.SendSingleMsgReq
 | |
| 		keyMsgData apistruct.KeyMsgData
 | |
| 
 | |
| 		sendID      string
 | |
| 		sessionType int32
 | |
| 		recvID      string
 | |
| 	)
 | |
| 	err = json.Unmarshal(decodedData, &keyMsgData)
 | |
| 	if err != nil {
 | |
| 		apiresp.GinError(c, errs.ErrArgs.WithDetail(err.Error()).Wrap())
 | |
| 		return
 | |
| 	}
 | |
| 	if keyMsgData.GroupID != "" {
 | |
| 		sessionType = constant.ReadGroupChatType
 | |
| 		sendID = req.SendID
 | |
| 	} else {
 | |
| 		sessionType = constant.SingleChatType
 | |
| 		sendID = keyMsgData.RecvID
 | |
| 		recvID = keyMsgData.SendID
 | |
| 	}
 | |
| 	// check param
 | |
| 	if keyMsgData.SendID == "" {
 | |
| 		apiresp.GinError(c, errs.ErrArgs.WithDetail("missing recvID or GroupID").Wrap())
 | |
| 		return
 | |
| 	}
 | |
| 	if sendID == "" {
 | |
| 		apiresp.GinError(c, errs.ErrArgs.WithDetail("missing sendID").Wrap())
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	msgData := &sdkws.MsgData{
 | |
| 		SendID:           sendID,
 | |
| 		RecvID:           recvID,
 | |
| 		GroupID:          keyMsgData.GroupID,
 | |
| 		ClientMsgID:      idutil.GetMsgIDByMD5(sendID),
 | |
| 		SenderPlatformID: constant.AdminPlatformID,
 | |
| 		SessionType:      sessionType,
 | |
| 		MsgFrom:          constant.UserMsgType,
 | |
| 		ContentType:      constant.Text,
 | |
| 		Content:          []byte(req.Content),
 | |
| 		OfflinePushInfo:  req.OfflinePushInfo,
 | |
| 		Ex:               req.Ex,
 | |
| 	}
 | |
| 
 | |
| 	respPb, err := m.Client.SendMsg(c, &msg.SendMsgReq{MsgData: msgData})
 | |
| 	if err != nil {
 | |
| 		apiresp.GinError(c, err)
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	var status = constant.MsgSendSuccessed
 | |
| 
 | |
| 	_, err = m.Client.SetSendMsgStatus(c, &msg.SetSendMsgStatusReq{
 | |
| 		Status: int32(status),
 | |
| 	})
 | |
| 
 | |
| 	if err != nil {
 | |
| 		apiresp.GinError(c, err)
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	apiresp.GinSuccess(c, respPb)
 | |
| }
 | |
| 
 | |
| func (m *MessageApi) CheckMsgIsSendSuccess(c *gin.Context) {
 | |
| 	a2r.Call(c, msg.MsgClient.GetSendMsgStatus, m.Client)
 | |
| }
 | |
| 
 | |
| func (m *MessageApi) GetUsersOnlineStatus(c *gin.Context) {
 | |
| 	a2r.Call(c, msg.MsgClient.GetSendMsgStatus, m.Client)
 | |
| }
 | |
| 
 | |
| func (m *MessageApi) GetActiveUser(c *gin.Context) {
 | |
| 	a2r.Call(c, msg.MsgClient.GetActiveUser, m.Client)
 | |
| }
 | |
| 
 | |
| func (m *MessageApi) GetActiveGroup(c *gin.Context) {
 | |
| 	a2r.Call(c, msg.MsgClient.GetActiveGroup, m.Client)
 | |
| }
 | |
| 
 | |
| func (m *MessageApi) SearchMsg(c *gin.Context) {
 | |
| 	a2r.Call(c, msg.MsgClient.SearchMessage, m.Client)
 | |
| }
 | |
| 
 | |
| func (m *MessageApi) GetServerTime(c *gin.Context) {
 | |
| 	a2r.Call(c, msg.MsgClient.GetServerTime, m.Client)
 | |
| }
 | |
| 
 | |
| func (m *MessageApi) GetStreamMsg(c *gin.Context) {
 | |
| 	a2r.Call(c, msg.MsgClient.GetServerTime, m.Client)
 | |
| }
 | |
| 
 | |
| func (m *MessageApi) AppendStreamMsg(c *gin.Context) {
 | |
| 	a2r.Call(c, msg.MsgClient.GetServerTime, m.Client)
 | |
| }
 |