go-wxhelper/tasks/friends/friends.go

306 lines
8.5 KiB
Go
Raw Normal View History

package friends
2023-09-21 17:33:59 +08:00
import (
"encoding/json"
"github.com/go-resty/resty/v2"
"go-wechat/client"
"go-wechat/common/constant"
2024-07-16 11:36:10 +08:00
"go-wechat/common/types"
"go-wechat/config"
2024-07-05 09:32:39 +08:00
"go-wechat/model/dto"
"go-wechat/model/entity"
"go-wechat/utils"
"gorm.io/gorm"
2023-09-21 17:33:59 +08:00
"log"
"slices"
"strings"
2023-10-11 14:46:47 +08:00
"time"
2023-09-21 17:33:59 +08:00
)
// 同步群成员
2023-09-26 14:29:41 +08:00
// http客户端
var hc = resty.New()
// Sync
2023-09-21 17:33:59 +08:00
// @description: 同步好友列表
func Sync() {
2024-07-05 09:32:39 +08:00
var base dto.Response[[]dto.FriendItem]
2023-09-21 17:33:59 +08:00
2023-09-26 14:29:41 +08:00
resp, err := hc.R().
2023-09-21 17:33:59 +08:00
SetHeader("Content-Type", "application/json;chartset=utf-8").
SetResult(&base).
Post(config.Conf.Wechat.GetURL("/api/getContactList"))
2023-09-21 17:33:59 +08:00
if err != nil {
log.Printf("获取好友列表失败: %s", err.Error())
return
}
log.Printf("获取好友列表结果: %s", resp.String())
tx := client.MySQL.Begin()
defer tx.Commit()
nowIds := []string{}
2024-07-22 09:19:17 +08:00
// 取出已存在的成员信息
var oldData []entity.Friend
err = tx.Find(&oldData).Error
if err != nil {
log.Printf("查询好友列表失败: %s", err.Error())
return
}
// 将历史数据整理成map
oldMap := make(map[string]bool)
for _, item := range oldData {
oldMap[item.Wxid] = item.IsOk
}
2024-05-14 12:08:30 +08:00
// 新增的成员,用于通知给指定的人
var newItmes = make(map[string]string)
2023-09-21 17:33:59 +08:00
for _, friend := range base.Data {
if strings.Contains(friend.Wxid, "gh_") || strings.Contains(friend.Wxid, "@openim") {
continue
}
// 特殊Id跳过
if slices.Contains(constant.SpecialId, friend.Wxid) {
continue
}
2023-12-08 11:41:59 +08:00
//log.Printf("昵称: %s -> 类型: %d -> 微信号: %s -> 微信原始Id: %s", friend.Nickname, friend.Type, friend.CustomAccount, friend.Wxid)
nowIds = append(nowIds, friend.Wxid)
2023-09-21 17:33:59 +08:00
// 判断是否存在,不存在的话就新增,存在就修改一下名字
2024-07-22 09:19:17 +08:00
if _, e := oldMap[friend.Wxid]; !e {
// 新增
err = tx.Create(&entity.Friend{
2024-05-14 12:08:30 +08:00
CustomAccount: friend.CustomAccount,
Nickname: friend.Nickname,
Pinyin: friend.Pinyin,
PinyinAll: friend.PinyinAll,
Wxid: friend.Wxid,
IsOk: true,
EnableAi: config.Conf.System.DefaultRule.Ai,
EnableChatRank: config.Conf.System.DefaultRule.ChatRank,
EnableSummary: config.Conf.System.DefaultRule.Summary,
EnableWelcome: config.Conf.System.DefaultRule.Welcome,
2024-05-15 11:15:15 +08:00
EnableNews: config.Conf.System.DefaultRule.News,
EnableHotTop: config.Conf.System.DefaultRule.HotTop,
AiFreeLimit: config.Conf.System.DefaultRule.AiFreeLimit,
ClearMember: 0,
2024-07-16 11:36:10 +08:00
LastActive: types.DateTime(time.Now().Local()),
}).Error
if err != nil {
log.Printf("新增好友失败: %s", err.Error())
continue
}
2024-05-14 12:08:30 +08:00
newItmes[friend.Wxid] = friend.Nickname
2024-05-14 11:50:27 +08:00
if conf, ok := config.Conf.Resource["introduce"]; ok {
// 发送一条新消息
switch conf.Type {
case "text":
// 文字类型
utils.SendMessage(friend.Wxid, "", conf.Path, 0)
case "image":
// 图片类型
utils.SendImage(friend.Wxid, conf.Path, 0)
case "emotion":
// 表情类型
utils.SendEmotion(friend.Wxid, conf.Path, 0)
}
}
2024-07-16 11:36:10 +08:00
// 发送配置网页
if config.Conf.System.Domain != "" {
2024-08-19 11:44:59 +08:00
title := "欢迎使用微信机器人(切勿转发)"
2024-07-16 11:36:10 +08:00
desc := "点我可以配置功能喔,提示非微信官方网页,点击继续访问即可"
2024-08-19 11:44:59 +08:00
url := utils.GetManagerUrl(friend.Wxid)
2024-07-16 11:36:10 +08:00
utils.SendPublicMsg(friend.Wxid, title, desc, url, 0)
}
} else {
pm := map[string]any{
"nickname": friend.Nickname,
"custom_account": friend.CustomAccount,
"pinyin": friend.Pinyin,
"pinyin_all": friend.PinyinAll,
"is_ok": true,
}
err = tx.Model(&entity.Friend{}).Where("wxid = ?", friend.Wxid).Updates(pm).Error
if err != nil {
log.Printf("修改好友失败: %s", err.Error())
continue
}
2024-07-22 09:19:17 +08:00
// 如果已存在但是是已退出的群,也通知一下
if !oldMap[friend.Wxid] {
newItmes[friend.Wxid] = friend.Nickname
2024-07-22 09:19:38 +08:00
// 通知一下,初始化完成
if conf, ok := config.Conf.Resource["introduce"]; ok {
// 发送一条新消息
switch conf.Type {
case "text":
// 文字类型
utils.SendMessage(friend.Wxid, "", conf.Path, 0)
case "image":
// 图片类型
utils.SendImage(friend.Wxid, conf.Path, 0)
case "emotion":
// 表情类型
utils.SendEmotion(friend.Wxid, conf.Path, 0)
}
}
// 发送配置网页
if config.Conf.System.Domain != "" {
2024-08-19 11:44:59 +08:00
title := "欢迎使用微信机器人(切勿转发)"
2024-07-22 09:19:38 +08:00
desc := "点我可以配置功能喔,提示非微信官方网页,点击继续访问即可"
2024-08-19 11:44:59 +08:00
url := utils.GetManagerUrl(friend.Wxid)
2024-07-22 09:19:38 +08:00
utils.SendPublicMsg(friend.Wxid, title, desc, url, 0)
2024-07-22 09:19:17 +08:00
}
}
}
2023-09-21 17:33:59 +08:00
// 群成员,同步一下成员信息
if strings.Contains(friend.Wxid, "@chatroom") {
syncGroupUsers(tx, friend.Wxid)
2023-09-21 17:33:59 +08:00
}
}
2024-05-14 12:08:30 +08:00
// 通知有新成员
if len(newItmes) > 0 && config.Conf.System.NewFriendNotify.Enable {
// 组装成一句话
msg := []string{"#新好友通知\n"}
for wxId, nickname := range newItmes {
2024-05-29 14:12:26 +08:00
msg = append(msg, "微信Id: "+wxId+"\n昵称: "+nickname)
2024-05-14 12:08:30 +08:00
}
for _, user := range config.Conf.System.NewFriendNotify.ToUser {
if user != "" {
// 发送一条新消息
2024-05-29 14:12:26 +08:00
utils.SendMessage(user, "", strings.Join(msg, "\n-------\n"), 0)
2024-05-14 12:08:30 +08:00
}
}
}
// 清理不在列表中的好友
clearPm := map[string]any{
2024-07-15 15:07:53 +08:00
"is_ok": false,
}
err = tx.Model(&entity.Friend{}).Where("wxid NOT IN (?)", nowIds).Updates(clearPm).Error
if err != nil {
log.Printf("清理好友失败: %s", err.Error())
}
log.Println("同步好友列表完成")
2023-09-21 17:33:59 +08:00
}
// syncGroupUsers
// @description: 同步群成员
// @param gid
func syncGroupUsers(tx *gorm.DB, gid string) {
2024-07-05 09:32:39 +08:00
var baseResp dto.Response[dto.GroupUser]
2023-09-21 17:33:59 +08:00
// 组装参数
param := map[string]any{
"chatRoomId": gid, // 群Id
}
pbs, _ := json.Marshal(param)
2023-09-26 14:29:41 +08:00
_, err := hc.R().
2023-09-21 17:33:59 +08:00
SetHeader("Content-Type", "application/json;chartset=utf-8").
SetBody(string(pbs)).
SetResult(&baseResp).
Post(config.Conf.Wechat.GetURL("/api/getMemberFromChatRoom"))
2023-09-21 17:33:59 +08:00
if err != nil {
log.Printf("获取群成员信息失败: %s", err.Error())
return
}
// 昵称Id
wxIds := strings.Split(baseResp.Data.Members, "^G")
2023-12-08 11:41:59 +08:00
//log.Printf(" 群成员数: %d", len(wxIds))
// 修改不在数组的群成员状态为不在
2023-10-11 14:46:47 +08:00
pm := map[string]any{
"is_member": false,
"leave_time": time.Now().Local(),
}
2023-10-12 14:12:33 +08:00
err = tx.Model(&entity.GroupUser{}).Where("group_id = ?", gid).Where("is_member IS TRUE").Where("wxid NOT IN (?)", wxIds).Updates(pm).Error
if err != nil {
log.Printf("修改群成员状态失败: %s", err.Error())
return
}
2023-09-21 17:33:59 +08:00
for _, wxid := range wxIds {
// 获取成员信息
cp, _ := getContactProfile(wxid)
if cp.Wxid != "" {
2023-12-08 11:41:59 +08:00
//log.Printf(" 微信Id: %s -> 昵称: %s -> 微信号: %s", wxid, cp.Nickname, cp.Account)
// 查询成员是否存在,不在就新增,否则修改
var count int64
err = tx.Model(&entity.GroupUser{}).Where("group_id = ?", gid).Where("wxid = ?", wxid).Count(&count).Error
if err != nil {
log.Printf("查询群成员失败: %s", err.Error())
continue
}
if count == 0 {
// 新增
err = tx.Create(&entity.GroupUser{
2024-01-26 12:00:29 +08:00
GroupId: gid,
Account: cp.Account,
HeadImage: cp.HeadImage,
Nickname: cp.Nickname,
Wxid: cp.Wxid,
IsMember: true,
IsAdmin: wxid == baseResp.Data.Admin,
JoinTime: time.Now().Local(),
LastActive: time.Now().Local(),
}).Error
if err != nil {
log.Printf("新增群成员失败: %s", err.Error())
continue
}
} else {
// 修改
pm := map[string]any{
"account": cp.Account,
"head_image": cp.HeadImage,
"nickname": cp.Nickname,
2023-10-16 08:45:00 +08:00
"is_member": true,
2024-01-12 22:26:06 +08:00
"is_admin": wxid == baseResp.Data.Admin,
2023-10-16 08:45:00 +08:00
"leave_time": nil,
}
err = tx.Model(&entity.GroupUser{}).Where("group_id = ?", gid).Where("wxid = ?", wxid).Updates(pm).Error
if err != nil {
log.Printf("修改群成员失败: %s", err.Error())
continue
}
}
2023-09-21 17:33:59 +08:00
}
}
}
// getContactProfile
// @description: 获取成员详情
// @param wxid
// @return ent
// @return err
2024-07-05 09:32:39 +08:00
func getContactProfile(wxid string) (ent dto.ContactProfile, err error) {
var baseResp dto.Response[dto.ContactProfile]
2023-09-21 17:33:59 +08:00
// 组装参数
param := map[string]any{
"wxid": wxid, // 群Id
}
pbs, _ := json.Marshal(param)
2023-09-26 14:29:41 +08:00
_, err = hc.R().
2023-09-21 17:33:59 +08:00
SetHeader("Content-Type", "application/json;chartset=utf-8").
SetBody(string(pbs)).
SetResult(&baseResp).
Post(config.Conf.Wechat.GetURL("/api/getContactProfile"))
2023-09-21 17:33:59 +08:00
if err != nil {
log.Printf("获取成员详情失败: %s", err.Error())
return
}
ent = baseResp.Data
return
}