Compare commits

...

12 Commits

Author SHA1 Message Date
李寻欢
be15d42d93 Merge pull request 'dev' (#12) from dev into main
Reviewed-on: https://gitee.ltd/lxh/go-wxhelper/pulls/12
2024-01-12 22:54:50 +08:00
李寻欢
6a879cbef6 🆕 群成员新增是否群主 2024-01-12 22:54:27 +08:00
李寻欢
01051ff606 🆕 网页新增控制是否打开指令功能 2024-01-12 22:20:14 +08:00
李寻欢
450857dee8 Merge pull request '🐛 Fix a bug. #7' (#11) from dev into main
Reviewed-on: https://gitee.ltd/lxh/go-wxhelper/pulls/11
2024-01-12 21:51:25 +08:00
李寻欢
77c5a96c76 🐛 Fix a bug. #7 2024-01-12 21:50:56 +08:00
李寻欢
0c2010c348 Merge pull request '🐛 修复AI插件会响应@所有人消息事件 #7' (#9) from dev into main
Reviewed-on: https://gitee.ltd/lxh/go-wxhelper/pulls/9
2024-01-09 11:12:52 +08:00
李寻欢
2918924107 🐛 修复AI插件会响应@所有人消息事件 2024-01-09 11:11:09 +08:00
李寻欢
f796e7b3a8 Merge pull request 'dev' (#8) from dev into main
Reviewed-on: https://gitee.ltd/lxh/go-wxhelper/pulls/8
2024-01-09 10:49:21 +08:00
李寻欢
83efe20ddf Merge remote-tracking branch 'origin/dev' into dev 2024-01-09 10:48:30 +08:00
李寻欢
f0bb46b9ab 🆕 新增显示vnc页面 2024-01-09 10:48:21 +08:00
李寻欢
68ae670429 🐛 Fix a bug. 2024-01-01 09:38:55 +08:00
李寻欢
1d14b036ed 🐛 Fix a bug. 2024-01-01 09:28:11 +08:00
15 changed files with 134 additions and 45 deletions

View File

@@ -85,6 +85,29 @@ func ChangeEnableWelcomeStatus(ctx *gin.Context) {
ctx.String(http.StatusOK, "操作成功") ctx.String(http.StatusOK, "操作成功")
} }
// ChangeEnableCommandStatus
// @description: 修改是否开启指令
// @param ctx
func ChangeEnableCommandStatus(ctx *gin.Context) {
var p changeStatusParam
if err := ctx.ShouldBindJSON(&p); err != nil {
ctx.String(http.StatusBadRequest, "参数错误")
return
}
log.Printf("待修改的群Id%s", p.WxId)
err := client.MySQL.Model(&entity.Friend{}).
Where("wxid = ?", p.WxId).
Update("`enable_command`", gorm.Expr(" !`enable_command`")).Error
if err != nil {
log.Printf("修改指令启用状态失败:%s", err)
ctx.String(http.StatusInternalServerError, "操作失败: %s", err)
return
}
ctx.String(http.StatusOK, "操作成功")
}
// ChangeSkipGroupRankStatus // ChangeSkipGroupRankStatus
// @description: 修改是否跳过水群排行榜 // @description: 修改是否跳过水群排行榜
// @param ctx // @param ctx

View File

@@ -3,6 +3,7 @@ package app
import ( import (
"fmt" "fmt"
"github.com/gin-gonic/gin" "github.com/gin-gonic/gin"
"go-wechat/config"
"go-wechat/service" "go-wechat/service"
"net/http" "net/http"
) )
@@ -21,6 +22,7 @@ func Index(ctx *gin.Context) {
} }
result["friends"] = friends result["friends"] = friends
result["groups"] = groups result["groups"] = groups
result["vnc"] = config.Conf.Wechat.VncUrl
// 渲染页面 // 渲染页面
ctx.HTML(http.StatusOK, "index.html", result) ctx.HTML(http.StatusOK, "index.html", result)
} }

View File

@@ -2,6 +2,8 @@
wechat: wechat:
# 微信HOOK接口地址 # 微信HOOK接口地址
host: 10.0.0.73:19088 host: 10.0.0.73:19088
# 微信容器映射出来的vnc页面地址没有就不填
vncUrl: http://192.168.1.175:19087/vnc_lite.html
# 是否在启动的时候自动设置hook服务的回调 # 是否在启动的时候自动设置hook服务的回调
autoSetCallback: false autoSetCallback: false
# 回调IP如果是Docker运行本参数必填(填auto表示自动不适用于 docker 环境)如果Docker修改了映射格式为 ip:port # 回调IP如果是Docker运行本参数必填(填auto表示自动不适用于 docker 环境)如果Docker修改了映射格式为 ip:port
@@ -29,7 +31,7 @@ task:
yesterday: '30 9 * * *' # 每天9:30 yesterday: '30 9 * * *' # 每天9:30
week: '30 9 * * 1' # 每周一9:30 week: '30 9 * * 1' # 每周一9:30
month: '30 9 1 * *' # 每月1号9:30 month: '30 9 1 * *' # 每月1号9:30
year: '55 0 1 1 *' # 每年1月1号9:30 year: '0 9 1 1 *' # 每年1月1号9:30
# AI回复 # AI回复
ai: ai:

View File

@@ -6,6 +6,7 @@ import "strings"
// @description: 微信助手 // @description: 微信助手
type wechat struct { type wechat struct {
Host string `json:"host" yaml:"host"` // 接口地址 Host string `json:"host" yaml:"host"` // 接口地址
VncUrl string `json:"vncUrl" yaml:"vncUrl"` // vnc页面地址
AutoSetCallback bool `json:"autoSetCallback" yaml:"autoSetCallback"` // 是否自动设置回调地址 AutoSetCallback bool `json:"autoSetCallback" yaml:"autoSetCallback"` // 是否自动设置回调地址
Callback string `json:"callback" yaml:"callback"` // 回调地址 Callback string `json:"callback" yaml:"callback"` // 回调地址
Forward []string `json:"forward" yaml:"forward"` // 转发地址 Forward []string `json:"forward" yaml:"forward"` // 转发地址

View File

@@ -2,7 +2,7 @@ version: '3.9'
services: services:
wechat: wechat:
image: lxh01/wxhelper-docker:3.9.5.81 image: lxh01/wxhelper-docker:3.9.5.81-v11
container_name: gw-wechat container_name: gw-wechat
restart: unless-stopped restart: unless-stopped
environment: environment:
@@ -10,7 +10,7 @@ services:
volumes: volumes:
- ./data/wechat:/home/app/.wine/drive_c/users/app/Documents/WeChat\ Files - ./data/wechat:/home/app/.wine/drive_c/users/app/Documents/WeChat\ Files
ports: ports:
- "8080:8080" - "19087:8080"
- "19088:19088" - "19088:19088"
healthcheck: healthcheck:
test: ["CMD", "curl", "-f", "http://localhost:19088/api/checkLogin"] test: ["CMD", "curl", "-f", "http://localhost:19088/api/checkLogin"]

View File

@@ -31,6 +31,7 @@ type GroupUser struct {
HeadImage string `json:"headImage"` // 头像 HeadImage string `json:"headImage"` // 头像
Nickname string `json:"nickname"` // 昵称 Nickname string `json:"nickname"` // 昵称
IsMember bool `json:"isMember" gorm:"type:tinyint(1) default 0 not null"` // 是否群成员 IsMember bool `json:"isMember" gorm:"type:tinyint(1) default 0 not null"` // 是否群成员
IsAdmin bool `json:"isAdmin" gorm:"type:tinyint(1) default 0 not null"` // 是否群主
JoinTime time.Time `json:"joinTime"` // 加入时间 JoinTime time.Time `json:"joinTime"` // 加入时间
LeaveTime *time.Time `json:"leaveTime"` // 离开时间 LeaveTime *time.Time `json:"leaveTime"` // 离开时间
SkipChatRank bool `json:"skipChatRank" gorm:"type:tinyint(1) default 0 not null"` // 是否跳过聊天排行 SkipChatRank bool `json:"skipChatRank" gorm:"type:tinyint(1) default 0 not null"` // 是否跳过聊天排行

View File

@@ -26,13 +26,14 @@ func Plugin() {
// 私聊指令消息 // 私聊指令消息
dispatcher.RegisterHandler(func(m *model.Message) bool { dispatcher.RegisterHandler(func(m *model.Message) bool {
// 私聊消息 或 群聊艾特机器人并且以/开头的消息 // 私聊消息 或 群聊艾特机器人并且以/开头的消息
return (m.IsPrivateText() || (m.IsAt() && m.CleanContentStartWith("/"))) && service.CheckIsEnableCommand(m.FromUser) isGroupAt := m.IsAt() && !m.IsAtAll()
return (m.IsPrivateText() || isGroupAt) && m.CleanContentStartWith("/") && service.CheckIsEnableCommand(m.FromUser)
}, plugins.Command) }, plugins.Command)
// AI消息插件 // AI消息插件
dispatcher.RegisterHandler(func(m *model.Message) bool { dispatcher.RegisterHandler(func(m *model.Message) bool {
// 群内@或者私聊文字消息 // 群内@或者私聊文字消息
return m.IsAt() || m.IsPrivateText() return (m.IsAt() && !m.IsAtAll()) || m.IsPrivateText()
}, plugins.AI) }, plugins.AI)
// 欢迎新成员 // 欢迎新成员

View File

@@ -2,6 +2,7 @@ package model
import ( import (
"encoding/xml" "encoding/xml"
"github.com/duke-git/lancet/v2/slice"
"go-wechat/types" "go-wechat/types"
"regexp" "regexp"
"strings" "strings"
@@ -31,6 +32,21 @@ type systemMsgDataXml struct {
Type string `xml:"type,attr"` Type string `xml:"type,attr"`
} }
// atMsgDataXml
// @description: 微信@消息的xml结构
type atMsgDataXml struct {
XMLName xml.Name `xml:"msgsource"`
Text string `xml:",chardata"`
AtUserList string `xml:"atuserlist"`
Silence string `xml:"silence"`
MemberCount string `xml:"membercount"`
Signature string `xml:"signature"`
TmpNode struct {
Text string `xml:",chardata"`
PublisherID string `xml:"publisher-id"`
} `xml:"tmp_node"`
}
// sysMsg // sysMsg
// @description: 消息主体 // @description: 消息主体
type sysMsg struct{} type sysMsg struct{}
@@ -92,6 +108,22 @@ func (m Message) IsAt() bool {
return strings.HasSuffix(m.DisplayFullContent, "在群聊中@了你") return strings.HasSuffix(m.DisplayFullContent, "在群聊中@了你")
} }
// IsAtAll
// @description: 是否是At所有人的消息
// @receiver m
// @return bool
func (m Message) IsAtAll() bool {
// 解析raw里面的xml
var d atMsgDataXml
if err := xml.Unmarshal([]byte(m.Raw), &d); err != nil {
return false
}
// 转换@用户列表为数组
atUserList := strings.Split(d.AtUserList, ",")
// 判断是否包含@所有人
return slice.Contain(atUserList, "notify@all")
}
// IsPrivateText // IsPrivateText
// @description: 是否是私聊消息 // @description: 是否是私聊消息
// @receiver m // @receiver m

View File

@@ -14,6 +14,8 @@ vim config.yaml # 编辑配置文件,内容如下,最新配置请参考项
wechat: wechat:
# 微信HOOK接口地址 # 微信HOOK接口地址
host: wechat:19088 host: wechat:19088
# 微信容器映射出来的vnc页面地址没有就不填
vncUrl: http://192.168.1.175:19087/vnc_lite.html
# 是否在启动的时候自动设置hook服务的回调 # 是否在启动的时候自动设置hook服务的回调
autoSetCallback: true autoSetCallback: true
# 回调IP如果是Docker运行本参数必填如果Docker修改了映射格式为 ip:port如果使用项目提供的docker-compsoe.yaml文件启动可以填`auto` # 回调IP如果是Docker运行本参数必填如果Docker修改了映射格式为 ip:port如果使用项目提供的docker-compsoe.yaml文件启动可以填`auto`
@@ -46,7 +48,7 @@ version: '3.9'
services: services:
wechat: wechat:
image: lxh01/wxhelper-docker:3.9.5.81 image: lxh01/wxhelper-docker:3.9.5.81-v11
container_name: gw-wechat container_name: gw-wechat
restart: unless-stopped restart: unless-stopped
environment: environment:
@@ -54,7 +56,7 @@ services:
volumes: volumes:
- ./data/wechat:/home/app/.wine/drive_c/users/app/Documents/WeChat\ Files - ./data/wechat:/home/app/.wine/drive_c/users/app/Documents/WeChat\ Files
ports: ports:
- "8080:8080" - "19087:8080"
- "19088:19088" - "19088:19088"
healthcheck: healthcheck:
test: ["CMD", "curl", "-f", "http://localhost:19088/api/checkLogin"] test: ["CMD", "curl", "-f", "http://localhost:19088/api/checkLogin"]

View File

@@ -23,6 +23,7 @@ func Init(g *gin.Engine) {
api := g.Group("/api") api := g.Group("/api")
api.PUT("/ai/status", app.ChangeEnableAiStatus) // 修改是否开启AI状态 api.PUT("/ai/status", app.ChangeEnableAiStatus) // 修改是否开启AI状态
api.PUT("/welcome/status", app.ChangeEnableWelcomeStatus) // 修改是否开启迎新状态 api.PUT("/welcome/status", app.ChangeEnableWelcomeStatus) // 修改是否开启迎新状态
api.PUT("/command/status", app.ChangeEnableCommandStatus) // 修改是否开启指令状态
api.PUT("/grouprank/status", app.ChangeEnableGroupRankStatus) // 修改是否开启水群排行榜状态 api.PUT("/grouprank/status", app.ChangeEnableGroupRankStatus) // 修改是否开启水群排行榜状态
api.PUT("/grouprank/skip", app.ChangeSkipGroupRankStatus) // 修改是否跳过水群排行榜状态 api.PUT("/grouprank/skip", app.ChangeSkipGroupRankStatus) // 修改是否跳过水群排行榜状态
api.GET("/group/users", app.GetGroupUsers) // 获取群成员列表 api.GET("/group/users", app.GetGroupUsers) // 获取群成员列表

View File

@@ -155,6 +155,7 @@ func syncGroupUsers(tx *gorm.DB, gid string) {
Nickname: cp.Nickname, Nickname: cp.Nickname,
Wxid: cp.Wxid, Wxid: cp.Wxid,
IsMember: true, IsMember: true,
IsAdmin: wxid == baseResp.Data.Admin,
JoinTime: time.Now().Local(), JoinTime: time.Now().Local(),
}).Error }).Error
if err != nil { if err != nil {
@@ -168,6 +169,7 @@ func syncGroupUsers(tx *gorm.DB, gid string) {
"head_image": cp.HeadImage, "head_image": cp.HeadImage,
"nickname": cp.Nickname, "nickname": cp.Nickname,
"is_member": true, "is_member": true,
"is_admin": wxid == baseResp.Data.Admin,
"leave_time": nil, "leave_time": nil,
} }
err = tx.Model(&entity.GroupUser{}).Where("group_id = ?", gid).Where("wxid = ?", wxid).Updates(pm).Error err = tx.Model(&entity.GroupUser{}).Where("group_id = ?", gid).Where("wxid = ?", wxid).Updates(pm).Error

View File

@@ -67,8 +67,7 @@ func dealYear(gid string) {
notifyMsgs = append(notifyMsgs, "对于那些未上榜的朋友们,我要说,你们也是我们群聊中非常重要的一部分。你们或许没有在排行榜上留下痕迹,但你们的存在和参与同样不可或缺。你们为群聊注入了新的思维和观点,为我们提供了不同的视角和见解。") notifyMsgs = append(notifyMsgs, "对于那些未上榜的朋友们,我要说,你们也是我们群聊中非常重要的一部分。你们或许没有在排行榜上留下痕迹,但你们的存在和参与同样不可或缺。你们为群聊注入了新的思维和观点,为我们提供了不同的视角和见解。")
notifyMsgs = append(notifyMsgs, "因此,我想特别鼓励未上榜的朋友们,继续发扬你们的热情和积极性。无论是在分享知识、讨论问题、还是互相支持鼓励,你们的贡献都是宝贵的。让我们共同创造一个更加活跃和有意义的群聊环境。") notifyMsgs = append(notifyMsgs, "因此,我想特别鼓励未上榜的朋友们,继续发扬你们的热情和积极性。无论是在分享知识、讨论问题、还是互相支持鼓励,你们的贡献都是宝贵的。让我们共同创造一个更加活跃和有意义的群聊环境。")
notifyMsgs = append(notifyMsgs, "最后,让我们一起迈向新的一年,相信自己的潜力和可能性,用我们的友谊和互助支持彼此。愿新的一年给我们带来更多的快乐、成功和成长。") notifyMsgs = append(notifyMsgs, "最后,让我们一起迈向新的一年,相信自己的潜力和可能性,用我们的友谊和互助支持彼此。愿新的一年给我们带来更多的快乐、成功和成长。")
notifyMsgs = append(notifyMsgs, "祝福你们新年快乐!让我们一起迎接2022年的到来!") notifyMsgs = append(notifyMsgs, fmt.Sprintf("祝福你们新年快乐!让我们一起迎接%d年的到来!", time.Now().Local().Year()))
notifyMsgs = append(notifyMsgs, " -- OpnWechat")
notifyMsgs = append(notifyMsgs, " ") notifyMsgs = append(notifyMsgs, " ")
notifyMsgs = append(notifyMsgs, fmt.Sprintf("🗣️ 去年本群 %d 位朋友共产生 %d 条发言", len(records), msgCount)) notifyMsgs = append(notifyMsgs, fmt.Sprintf("🗣️ 去年本群 %d 位朋友共产生 %d 条发言", len(records), msgCount))
notifyMsgs = append(notifyMsgs, "\n🏵 活跃用户排行榜 🏵") notifyMsgs = append(notifyMsgs, "\n🏵 活跃用户排行榜 🏵")

View File

@@ -37,6 +37,7 @@
<th>最后活跃时间</th> <th>最后活跃时间</th>
<th>是否在通讯录</th> <th>是否在通讯录</th>
<th>是否启用AI</th> <th>是否启用AI</th>
<th>是否启用指令</th>
</tr> </tr>
</thead> </thead>
<tbody> <tbody>
@@ -67,6 +68,14 @@
<label class="swap swap-flip {{ checkSwap .EnableAi }}"> <label class="swap swap-flip {{ checkSwap .EnableAi }}">
<input type="checkbox" onclick="changeAiEnableStatus({{.Wxid}})"/> <input type="checkbox" onclick="changeAiEnableStatus({{.Wxid}})"/>
<div class="swap-on">✔️已启用</div>
<div class="swap-off">❌已禁用</div>
</label>
</td>
<td>
<label class="swap swap-flip {{ checkSwap .EnableCommand }}">
<input type="checkbox" onclick="changeCommandEnableStatus({{.Wxid}})"/>
<div class="swap-on">✔️已启用</div> <div class="swap-on">✔️已启用</div>
<div class="swap-off">❌已禁用</div> <div class="swap-off">❌已禁用</div>
</label> </label>
@@ -91,6 +100,7 @@
<th>是否启用AI</th> <th>是否启用AI</th>
<th>是否启用水群排行榜</th> <th>是否启用水群排行榜</th>
<th>是否启用迎新</th> <th>是否启用迎新</th>
<th>是否启用指令</th>
<th>操作</th> <th>操作</th>
</tr> </tr>
</thead> </thead>
@@ -143,6 +153,14 @@
<div class="swap-off">❌已禁用</div> <div class="swap-off">❌已禁用</div>
</label> </label>
</td> </td>
<td>
<label class="swap swap-flip {{ checkSwap .EnableCommand }}">
<input type="checkbox" onclick="changeCommandEnableStatus({{.Wxid}})"/>
<div class="swap-on">✔️已启用</div>
<div class="swap-off">❌已禁用</div>
</label>
</td>
<td> <td>
<button class="btn btn-link" onclick="getGroupUsers({{.Wxid}}, {{.Nickname}})">查看群成员</button> <button class="btn btn-link" onclick="getGroupUsers({{.Wxid}}, {{.Nickname}})">查看群成员</button>
</td> </td>
@@ -151,6 +169,15 @@
</tbody> </tbody>
</table> </table>
</div> </div>
{{ if ne .vnc "" }}
<input type="radio" name="friend_tab" role="tab" class="tab" aria-label="运行状态"/>
<div role="tabpanel" class="tab-content p-6">
<div style="height: 747px;width: 1280px;overflow: hidden !important;">
<iframe src="{{ .vnc }}" frameborder="0" style="width: 100%;height: 100%;pointer-events: none;"></iframe>
</div>
</div>
{{ end }}
</div> </div>
@@ -174,6 +201,7 @@
<th>微信Id</th> <th>微信Id</th>
<th>昵称</th> <th>昵称</th>
<th>是否群成员</th> <th>是否群成员</th>
<th>是否群主</th>
<th>加群时间</th> <th>加群时间</th>
<th>最后活跃时间</th> <th>最后活跃时间</th>
<th>退群时间</th> <th>退群时间</th>

View File

@@ -51,6 +51,22 @@ function changeWelcomeEnableStatus(wxId) {
}) })
} }
// 修改指令权限启用状态
function changeCommandEnableStatus(wxId) {
axios({
method: 'put',
url: '/api/command/status',
data: {
wxId: wxId
}
}).then(function (response) {
console.log(`返回结果: ${JSON.stringify(response)}`);
}).catch(function (error) {
console.log(`错误信息: ${error}`);
alert("修改失败")
})
}
// 修改群成员是否参与排行榜状态 // 修改群成员是否参与排行榜状态
function changeUserGroupRankSkipStatus(groupId, userId) { function changeUserGroupRankSkipStatus(groupId, userId) {
console.log("修改水群排行榜开启状态: ", groupId, userId) console.log("修改水群排行榜开启状态: ", groupId, userId)
@@ -96,43 +112,20 @@ function getGroupUsers(groupId, groupName) {
// 渲染群成员列表 // 渲染群成员列表
const groupUsers = response.data const groupUsers = response.data
// 循环渲染数据 // 循环渲染数据
for (let i = 0; i < groupUsers.length; i++) { groupUsers.forEach((groupUser, i) => {
const groupUser = groupUsers[i] const { wxid, nickname, isMember, isAdmin, joinTime, lastActiveTime, leaveTime, skipChatRank } = groupUser;
let row = tbody.insertRow(i); // 插入新行 let row = tbody.insertRow(i);
// Insert data into cells
// 微信Id row.insertCell(0).innerHTML = wxid;
let wxId = row.insertCell(0); row.insertCell(1).innerHTML = nickname;
wxId.innerHTML = groupUser.wxid; row.insertCell(2).innerHTML = `<div class="badge badge-${isMember ? 'info' : 'error'} gap-2">${isMember ? '是' : '否'}</div>`;
row.insertCell(3).innerHTML = `<div class="badge badge-${isAdmin ? 'info' : 'error'} gap-2">${isAdmin ? '是' : '否'}</div>`;
// 昵称 row.insertCell(4).innerHTML = joinTime;
let nickname = row.insertCell(1); row.insertCell(5).innerHTML = lastActiveTime;
nickname.innerHTML = groupUser.nickname; row.insertCell(6).innerHTML = leaveTime;
row.insertCell(7).innerHTML = `<input type="checkbox" class="toggle toggle-error" ${skipChatRank ? 'checked' : ''} onclick="changeUserGroupRankSkipStatus('${groupId}', '${wxid}')" />`;
// 是否群成员 });
let isMember = row.insertCell(2);
if (groupUser.isMember) {
isMember.innerHTML = '<div class="badge badge-info gap-2">是</div>';
} else {
isMember.innerHTML = '<div class="badge badge-error gap-2">否</div>';
}
// 加群时间
let joinTime = row.insertCell(3);
joinTime.innerHTML = groupUser.joinTime;
// 最后活跃时间
let lastActiveTime = row.insertCell(4);
lastActiveTime.innerHTML = groupUser.lastActiveTime;
// 退群时间
let leaveTime = row.insertCell(5);
leaveTime.innerHTML = groupUser.leaveTime;
// 是否跳过水群排行榜
let skipChatRank = row.insertCell(6);
skipChatRank.innerHTML = `<input type="checkbox" class="toggle toggle-error" ${groupUser.skipChatRank ? 'checked' : ''} onclick="changeUserGroupRankSkipStatus(\'${groupId}\', \'${groupUser.wxid}\')" />`;
}
}).catch(function (error) { }).catch(function (error) {
console.log(`错误信息: ${error}`); console.log(`错误信息: ${error}`);
}).finally(function () { }).finally(function () {

View File

@@ -15,6 +15,7 @@ type FriendItem struct {
EnableAi bool // 是否使用AI EnableAi bool // 是否使用AI
EnableChatRank bool // 是否使用聊天排行 EnableChatRank bool // 是否使用聊天排行
EnableWelcome bool // 是否使用迎新 EnableWelcome bool // 是否使用迎新
EnableCommand bool // 是否启用指令
IsOk bool // 是否还在通讯库(群聊是要还在群里也算) IsOk bool // 是否还在通讯库(群聊是要还在群里也算)
LastActiveTime types.DateTime // 最后活跃时间 LastActiveTime types.DateTime // 最后活跃时间
} }
@@ -27,6 +28,7 @@ type GroupUserItem struct {
HeadImage string `json:"headImage"` // 头像 HeadImage string `json:"headImage"` // 头像
Nickname string `json:"nickname"` // 昵称 Nickname string `json:"nickname"` // 昵称
IsMember bool `json:"isMember" ` // 是否群成员 IsMember bool `json:"isMember" ` // 是否群成员
IsAdmin bool `json:"isAdmin"` // 是否群主
JoinTime types.DateTime `json:"joinTime"` // 加入时间 JoinTime types.DateTime `json:"joinTime"` // 加入时间
LastActiveTime types.DateTime `json:"lastActiveTime"` // 最后活跃时间 LastActiveTime types.DateTime `json:"lastActiveTime"` // 最后活跃时间
LeaveTime types.DateTime `json:"leaveTime"` // 离开时间 LeaveTime types.DateTime `json:"leaveTime"` // 离开时间