Compare commits
	
		
			3 Commits
		
	
	
		
			f60f7c4a6a
			...
			a295762f37
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| a295762f37 | |||
| 2151a8e42b | |||
| 4269f34670 | 
| @ -96,7 +96,6 @@ func (o *ChatRoom) Run() { | ||||
| 		// 注册事件
 | ||||
| 		case newClient := <-o.register: | ||||
| 			o.pushEvent(EventUserJoin, EventProgressBefore, newClient) | ||||
| 			defer o.pushEvent(EventUserJoin, EventProgressAfter, newClient) | ||||
| 
 | ||||
| 			o.clientsRwLocker.Lock() | ||||
| 			//添加到客户端集合中
 | ||||
| @ -139,11 +138,11 @@ func (o *ChatRoom) Run() { | ||||
| 				//再把自己的客户端加入会话
 | ||||
| 				o.Session[newClient.SessionId] = append(o.Session[newClient.SessionId], newClient) | ||||
| 			} | ||||
| 			o.pushEvent(EventUserJoin, EventProgressAfter, newClient) | ||||
| 			o.clientsRwLocker.Unlock() | ||||
| 		//注销事件
 | ||||
| 		case client := <-o.UnRegister: | ||||
| 			o.pushEvent(EventUserLeave, EventProgressBefore, client) | ||||
| 			defer o.pushEvent(EventUserLeave, EventProgressAfter, client) | ||||
| 			//panic 恢复
 | ||||
| 			defer func() { | ||||
| 				if r := recover(); r != "" { | ||||
| @ -168,6 +167,7 @@ func (o *ChatRoom) Run() { | ||||
| 				delete(o.clients[client.UserId], client.ClientId) | ||||
| 				fmt.Printf("ws客户端%s 被注销\n", client.ClientId) | ||||
| 			} | ||||
| 			o.pushEvent(EventUserLeave, EventProgressAfter, client) | ||||
| 		// 消息群发事件
 | ||||
| 		case messageInfo := <-o.broadcast: | ||||
| 			o.Broadcast(messageInfo.message, messageInfo.UserIds...) | ||||
|  | ||||
| @ -197,15 +197,15 @@ func (a *ChatAutoReplyRulerHandler) FieeLoginDemo(c *gin.Context) { | ||||
| 		return | ||||
| 	} | ||||
| 
 | ||||
| 	departmentName := "" | ||||
| 	if loginRes.AccountInfo != nil && len(loginRes.AccountInfo.Departments) > 0 { | ||||
| 		departmentName = loginRes.AccountInfo.Departments[0].Name | ||||
| 	} | ||||
| 	loginRes.Token, err = secret.CombineSecret("xxx", departmentName, loginRes.Token) | ||||
| 	if err != nil { | ||||
| 		service.Error(c, err) | ||||
| 		return | ||||
| 	} | ||||
| 	//departmentName := ""
 | ||||
| 	//if loginRes.AccountInfo != nil && len(loginRes.AccountInfo.Departments) > 0 {
 | ||||
| 	//	departmentName = loginRes.AccountInfo.Departments[0].Name
 | ||||
| 	//}
 | ||||
| 	//loginRes.Token, err = secret.CombineSecret("xxx", departmentName, loginRes.Token)
 | ||||
| 	//if err != nil {
 | ||||
| 	//	service.Error(c, err)
 | ||||
| 	//	return
 | ||||
| 	//}
 | ||||
| 	service.Success(c, loginRes) | ||||
| } | ||||
| func (a *ChatAutoReplyRulerHandler) Test(c *gin.Context) { | ||||
|  | ||||
| @ -183,5 +183,17 @@ type AutoReplyRule struct { | ||||
| } | ||||
| 
 | ||||
| type UserDetailReq struct { | ||||
| 	UserId uint64 `json:"userId"` | ||||
| 	ChatUserId int64 `json:"chatUserId"` //注意是聊天用户ID(chat_user表),不是账号服务ID
 | ||||
| } | ||||
| type UserDetailResp struct { | ||||
| 	UserId      uint64 `json:"userId"` | ||||
| 	ChatUserId  int64  `json:"chatUserId"` | ||||
| 	RnStatus    int32  `json:"rnStatus"` //状态 1:未实名 2:审核中 3:审核失败 4:审核通过
 | ||||
| 	SubNum      string `json:"subNum"` | ||||
| 	RealName    string `json:"realName"` | ||||
| 	Age         string `json:"age"` | ||||
| 	Gender      string `json:"gender"` | ||||
| 	NativePlace string `json:"nativePlace"` | ||||
| 	Phone       string `json:"phone"` | ||||
| 	GroupPhoto  string `json:"groupPhoto"` | ||||
| } | ||||
|  | ||||
| @ -541,16 +541,43 @@ func (cr ChatHandler) UserDetail(c *gin.Context) { | ||||
| 		service.Error(c, err) | ||||
| 		return | ||||
| 	} | ||||
| 	if req.UserId == 0 { | ||||
| 		service.Success(c, dto.ArtistInfo{}) | ||||
| 	var chatUser *accountFiee.ChatUserData | ||||
| 	if req.ChatUserId == 0 { //不传ChatUserId则从token获取
 | ||||
| 		var code e.ErrorCodeType | ||||
| 		chatUser, code = jwt.ParseToChatUser(c) | ||||
| 		if code != 0 { | ||||
| 			service.ErrWithCode(c, code) | ||||
| 			return | ||||
| 		} | ||||
| 	resp, err := service.AccountFieeProvider.Info(c, &accountFiee.InfoRequest{ID: req.UserId, Domain: config.AppConfig.System.Domain}) | ||||
| 	} else { | ||||
| 		var err error | ||||
| 		chatUser, err = service.AccountFieeProvider.GetChatUserDetail(c, &accountFiee.GetChatUserByIdRequest{ | ||||
| 			Id: req.ChatUserId, | ||||
| 		}) | ||||
| 		if err != nil { | ||||
| 			service.Error(c, err) | ||||
| 			return | ||||
| 		} | ||||
| 	service.Success(c, resp) | ||||
| 	} | ||||
| 	resp, err := service.AccountFieeProvider.Info(c, &accountFiee.InfoRequest{ID: uint64(chatUser.OriginId), Domain: config.AppConfig.System.Domain}) | ||||
| 	if err != nil { | ||||
| 		service.Error(c, err) | ||||
| 		return | ||||
| 	} | ||||
| 	var detail = dto.UserDetailResp{ | ||||
| 		UserId:      resp.Id, | ||||
| 		ChatUserId:  chatUser.ID, | ||||
| 		RnStatus:    resp.Status, | ||||
| 		SubNum:      resp.SubNum, | ||||
| 		RealName:    resp.Name, | ||||
| 		Age:         "∞", | ||||
| 		Gender:      resp.Sex, | ||||
| 		NativePlace: resp.PlaceOfResidence, | ||||
| 		Phone:       resp.TelNum, | ||||
| 		GroupPhoto:  resp.GroupPhoto, | ||||
| 	} | ||||
| 
 | ||||
| 	service.Success(c, detail) | ||||
| } | ||||
| 
 | ||||
| // 对没有名字的name进行优化
 | ||||
|  | ||||
| @ -30,7 +30,7 @@ func NewMessage(ctx context.Context, cache *chatCache.ChatCache, chatUser *accou | ||||
| 	fmt.Println("NewMessage 1111111111111111111111111111111") | ||||
| 	fmt.Println("NewMessage 22222222222222222222222222222222222") | ||||
| 	//存储入库
 | ||||
| 	if chatUser.NickName != "" { | ||||
| 	if chatUser.NickName == "" { | ||||
| 		chatUser.NickName = fmt.Sprintf("未知用户(%d)", chatUser.ID) | ||||
| 	} | ||||
| 	fmt.Println("NewMessage 3333333333333333333333333333333333") | ||||
|  | ||||
| @ -22,23 +22,37 @@ type Reply struct { | ||||
| 	Rules    []IRule | ||||
| } | ||||
| 
 | ||||
| func (r *Reply) Hit(eventType ws.EventType, msg *accountFiee.ChatRecordData, robotId int64) (hit bool, runTime time.Time, logic func() error) { | ||||
| 	for _, rule := range r.Rules { | ||||
| 		hit, runTime, logic = rule.Hit(eventType, msg, robotId) | ||||
| 		if hit { | ||||
| 			return | ||||
| 		} | ||||
| 	} | ||||
| 	return | ||||
| } | ||||
| 
 | ||||
| type IRule interface { | ||||
| 	Hit(msg *accountFiee.ChatRecordData) (hit bool, runTime time.Time, logic func(robotId int64, response string) error) | ||||
| 	Hit(eventType ws.EventType, msg *accountFiee.ChatRecordData, robotId int64) (hit bool, runTime time.Time, logic func() error) | ||||
| } | ||||
| 
 | ||||
| func NewReplyWhenHitKeywords(keywords []string) IRule { | ||||
| 	return &ReplyWhenHitKeywords{Keywords: keywords} | ||||
| } | ||||
| 
 | ||||
| // KeywordsRuleChecker 关键字回复
 | ||||
| type KeywordsRuleChecker struct { | ||||
| type ReplyWhenHitKeywords struct { | ||||
| 	Keywords []string `json:"keywords"` | ||||
| } | ||||
| 
 | ||||
| func (k KeywordsRuleChecker) Hit(record *accountFiee.ChatRecordData) (hit bool, runTime time.Time, logic func(robotId int64, response string) error) { | ||||
| func (k ReplyWhenHitKeywords) Hit(eventType ws.EventType, record *accountFiee.ChatRecordData, robotId int64) (hit bool, runTime time.Time, logic func() error) { | ||||
| 	for _, v := range k.Keywords { | ||||
| 		if strings.Contains(record.Content, v) { | ||||
| 			hit = true | ||||
| 			break | ||||
| 		} | ||||
| 	} | ||||
| 	logic = func(robotId int64, response string) error { | ||||
| 	logic = func() error { | ||||
| 		var notice = dto.MessageListType{} | ||||
| 		notice.BuildMessage(record) | ||||
| 		_, err := consts.ChatRoom.SendSessionMessage(robotId, record.SessionId, ws.NewChatMsgType, notice) | ||||
| @ -48,10 +62,17 @@ func (k KeywordsRuleChecker) Hit(record *accountFiee.ChatRecordData) (hit bool, | ||||
| } | ||||
| 
 | ||||
| // 用户打开聊天会话直接发送
 | ||||
| func NewReplyWhenUserJoinSession() IRule { | ||||
| 	return &ReplyWhenUserJoinSession{} | ||||
| } | ||||
| 
 | ||||
| type ReplyWhenUserJoinSession struct { | ||||
| } | ||||
| 
 | ||||
| func (k ReplyWhenUserJoinSession) Hit(record *accountFiee.ChatRecordData, robotId int64) (hit bool, runTime time.Time, logic func(robotId int64, response string) error) { | ||||
| func (k ReplyWhenUserJoinSession) Hit(eventType ws.EventType, record *accountFiee.ChatRecordData, robotId int64) (hit bool, runTime time.Time, logic func() error) { | ||||
| 	if eventType != ws.EventUserJoin { | ||||
| 		return | ||||
| 	} | ||||
| 	queryRes, err := service.AccountFieeProvider.GetChatRecordList(context.Background(), &accountFiee.GetChatRecordListRequest{ | ||||
| 		Query: &accountFiee.ChatRecordData{ | ||||
| 			SessionId: record.SessionId, | ||||
| @ -74,7 +95,7 @@ func (k ReplyWhenUserJoinSession) Hit(record *accountFiee.ChatRecordData, robotI | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	logic = func(robotId int64, response string) error { | ||||
| 	logic = func() error { | ||||
| 		var notice = dto.MessageListType{} | ||||
| 		notice.BuildMessage(record) | ||||
| 		_, err = consts.ChatRoom.SendSessionMessage(robotId, record.SessionId, ws.NewChatMsgType, notice) | ||||
| @ -82,3 +103,23 @@ func (k ReplyWhenUserJoinSession) Hit(record *accountFiee.ChatRecordData, robotI | ||||
| 	} | ||||
| 	return | ||||
| } | ||||
| 
 | ||||
| // 客服
 | ||||
| func NewReplyWhenWaiterNoAction() *ReplyWhenWaiterNoAction { | ||||
| 	return &ReplyWhenWaiterNoAction{} | ||||
| } | ||||
| 
 | ||||
| type ReplyWhenWaiterNoAction struct { | ||||
| } | ||||
| 
 | ||||
| func (k ReplyWhenWaiterNoAction) Hit(eventType ws.EventType, record *accountFiee.ChatRecordData, robotId int64) (hit bool, runTime time.Time, logic func() error) { | ||||
| 
 | ||||
| 	logic = func() error { | ||||
| 		var notice = dto.MessageListType{} | ||||
| 		notice.BuildMessage(record) | ||||
| 		_, err := consts.ChatRoom.SendSessionMessage(robotId, record.SessionId, ws.NewChatMsgType, notice) | ||||
| 		return err | ||||
| 	} | ||||
| 	return | ||||
| 
 | ||||
| } | ||||
|  | ||||
| @ -13,6 +13,7 @@ import ( | ||||
| 	"fonchain-fiee/pkg/common/ws" | ||||
| 	"fonchain-fiee/pkg/service" | ||||
| 	"fonchain-fiee/pkg/service/asChat/consts" | ||||
| 	"fonchain-fiee/pkg/service/asChat/dto" | ||||
| 	"log" | ||||
| 	"sync" | ||||
| 	"time" | ||||
| @ -47,11 +48,31 @@ func NewRobot() *Robot { | ||||
| 		EventListener: &ws.EventListener{ | ||||
| 			Name: "robot1", | ||||
| 			ListenEvents: []ws.ListenEvent{ //只监听消息推送事件
 | ||||
| 				{ws.EventUserJoin, ws.EventProgressAfter}, | ||||
| 				{ws.EventChatMessage, ws.EventProgressAfter}, | ||||
| 			}, | ||||
| 			Chan: make(ws.ListenEventChan), | ||||
| 		}, | ||||
| 	} | ||||
| 	ruleListRes, err := service.AccountFieeProvider.GetChatAutoReplyRulerList(ctx, &accountFiee.GetChatAutoReplyRulerListRequest{ | ||||
| 		Query:    &accountFiee.ChatAutoReplyRulerData{Status: 1}, | ||||
| 		Page:     1, | ||||
| 		PageSize: -1, | ||||
| 	}) | ||||
| 	if err != nil { | ||||
| 		log.Printf("robot 查询回复规则失败:%v\n", err) | ||||
| 	} else { | ||||
| 		var data []*dto.ChatAutoReplyData | ||||
| 		for _, v := range ruleListRes.List { | ||||
| 			tmp := dto.ChatAutoReplyData{} | ||||
| 			tmp.Parse(v) | ||||
| 			data = append(data, &tmp) | ||||
| 		} | ||||
| 		for _, v := range data { | ||||
| 			reply := ParseReplyRule(v) | ||||
| 			r.Rules = append(r.Rules, reply) | ||||
| 		} | ||||
| 	} | ||||
| 	consts.ChatRoom.RegisterEventListener(r.EventListener) | ||||
| 	go r.Run() | ||||
| 	return r | ||||
| @ -68,36 +89,36 @@ type Robot struct { | ||||
| 	*ws.EventListener | ||||
| } | ||||
| 
 | ||||
| func (r *Robot) Listen(record *accountFiee.ChatRecordData) { | ||||
| 	for _, replyRules := range r.Rules { | ||||
| 		for _, rule := range replyRules.Rules { | ||||
| 			hit, runTime, function := rule.Hit(record) | ||||
| 			if hit && function != nil { | ||||
| 				if runTime.IsZero() { | ||||
| 					go func() { | ||||
| 						err := function(r.Info.ID, replyRules.Response) | ||||
| 						if err != nil { | ||||
| 							log.Printf("聊天机器人[%d]回复消息失败:%v", r.Info.ID, err) | ||||
| 						} | ||||
| 					}() | ||||
| 				} else { | ||||
| 					r.mu.Lock() | ||||
| 					r.DelayTask = append(r.DelayTask, RobotTask{ | ||||
| 						RunTime:  runTime, | ||||
| 						Run:      function, | ||||
| 						Response: replyRules.Response, | ||||
| 					}) | ||||
| 					r.mu.Unlock() | ||||
| 					// 添加任务后启动定时任务(如果未运行)
 | ||||
| 					if !r.isRunning { | ||||
| 						go r.Run() | ||||
| 					} | ||||
| 				} | ||||
| 				break | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| //func (r *Robot) Listen(record *accountFiee.ChatRecordData) {
 | ||||
| //	for _, replyRules := range r.Rules {
 | ||||
| //		for _, rule := range replyRules.Rules {
 | ||||
| //			hit, runTime, function := rule.Hit(record)
 | ||||
| //			if hit && function != nil {
 | ||||
| //				if runTime.IsZero() {
 | ||||
| //					go func() {
 | ||||
| //						err := function(r.Info.ID, replyRules.Response)
 | ||||
| //						if err != nil {
 | ||||
| //							log.Printf("聊天机器人[%d]回复消息失败:%v", r.Info.ID, err)
 | ||||
| //						}
 | ||||
| //					}()
 | ||||
| //				} else {
 | ||||
| //					r.mu.Lock()
 | ||||
| //					r.DelayTask = append(r.DelayTask, RobotTask{
 | ||||
| //						RunTime:  runTime,
 | ||||
| //						Run:      function,
 | ||||
| //						Response: replyRules.Response,
 | ||||
| //					})
 | ||||
| //					r.mu.Unlock()
 | ||||
| //					// 添加任务后启动定时任务(如果未运行)
 | ||||
| //					if !r.isRunning {
 | ||||
| //						go r.Run()
 | ||||
| //					}
 | ||||
| //				}
 | ||||
| //				break
 | ||||
| //			}
 | ||||
| //		}
 | ||||
| //	}
 | ||||
| //}
 | ||||
| 
 | ||||
| func (r *Robot) Run() { | ||||
| 	r.mu.Lock() | ||||
| @ -151,8 +172,35 @@ func (r *Robot) Run() { | ||||
| 		case <-r.stopChan: | ||||
| 			return | ||||
| 		case event := <-r.EventListener.Chan: | ||||
| 			fmt.Sprintf("listen event:%#v\n", event) | ||||
| 
 | ||||
| 			fmt.Printf("listen event:%#v\n", event) | ||||
| 			switch event.EventType { | ||||
| 			case ws.EventUserJoin: //用户加入聊天室
 | ||||
| 				for _, ruleResponse := range r.Rules { | ||||
| 					hit, runtime, logic := ruleResponse.Hit(ws.EventUserJoin, nil, r.Info.ID) | ||||
| 					if hit { | ||||
| 						if !runtime.IsZero() { | ||||
| 							err := logic() | ||||
| 							if err != nil { | ||||
| 								log.Printf("robot 执行任务失败:%v\n", err) | ||||
| 							} | ||||
| 						} | ||||
| 						break | ||||
| 					} | ||||
| 				} | ||||
| 			case ws.EventChatMessage: | ||||
| 				for _, ruleResponse := range r.Rules { | ||||
| 					hit, runtime, logic := ruleResponse.Hit(ws.EventChatMessage, nil, r.Info.ID) | ||||
| 					if hit { | ||||
| 						if !runtime.IsZero() { | ||||
| 							err := logic() | ||||
| 							if err != nil { | ||||
| 								log.Printf("robot 执行任务失败:%v\n", err) | ||||
| 							} | ||||
| 						} | ||||
| 						break | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| @ -5,3 +5,32 @@ | ||||
| // @time      : 2025/6/13 16:16
 | ||||
| // -------------------------------------------
 | ||||
| package robot | ||||
| 
 | ||||
| import ( | ||||
| 	"fonchain-fiee/pkg/service/asChat/dto" | ||||
| 	"strings" | ||||
| ) | ||||
| 
 | ||||
| func ParseReplyRule(data *dto.ChatAutoReplyData) (r Reply) { | ||||
| 	r.Response = data.Response | ||||
| 	for ruleName, v := range data.Rules { | ||||
| 		if !v.Enable { | ||||
| 			continue | ||||
| 		} | ||||
| 		switch ruleName { | ||||
| 		case "keywords": //关键字回复
 | ||||
| 			var keywords []string | ||||
| 			if v.Content != "" { | ||||
| 				continue | ||||
| 			} else { | ||||
| 				keywords = strings.Split(v.Content, ",") | ||||
| 			} | ||||
| 			r.Rules = append(r.Rules, NewReplyWhenHitKeywords(keywords)) | ||||
| 		case "joinSession": //加入聊天后回复
 | ||||
| 			r.Rules = append(r.Rules, NewReplyWhenUserJoinSession()) | ||||
| 		case "noReplyAfter": //指定时间没有回复则自动回复
 | ||||
| 			r.Rules = append(r.Rules, NewReplyWhenWaiterNoAction()) | ||||
| 		} | ||||
| 	} | ||||
| 	return | ||||
| } | ||||
|  | ||||
		Loading…
	
		Reference in New Issue
	
	Block a user