这个功能其实很容易的,之前在各个群里都重复分享过,但问的人多干的人少,一直也没人提个 PR😳

用途

用途比较多,简单列举几个:

  • 群里 @人,比起简单的昵称,群名片可能更好些。
  • 防重,目前已经发现有好几个朋友,同时加入了多个群,一拉成员一比较,就可以解决。
  • 退群检测,通过群成员,可以以一种笨方法实现退群检测功能。
  • 爆粉,这个不宜多说。

说起退群检测,跟大家分享一个近日发生的趣事儿。

段老板需要 70 个群成员做些事情,于是便邀请大家进群,在群里呆两天就可以,一人可领 0.5 元红包。当群里超了 70 人后,在群友的呼喊中,段老板发了红包。大伙都很高兴,然后瞬间群人员变成了 66 个……

结果有群友上了黑科技,把逃跑的人挂出来了:

都不想给他们打码

这些人,估计都要上群友们的黑名单了。

原理

复杂的实现办法,可以去找 Call,然后实现一个方法;简单的办法,直接通过数据库,查询群成员。

时间紧任务重,用简单的办法。

群名片

MicroMsg.db 库里,有个 ChatRoom 表,表里有个 RoomData 字段,是二进制类型。

其实这是 Protocol Buffers 编码的群名片信息,解码一下就出来了。

具体怎么知道的,这里不展开了。

解码完后,就会拿到 wxid群名片;但如果没设置,群名片就是空了,这时候回落到用户昵称就可以了。

用户昵称

MicroMsg.db 库里,有个 Contact 表,表里有 UserNameNickName 字段,分别是 wxid昵称

实现

搞清楚了原理,实现就很简单了:

  • 先去查 ChatRoom,看看有没有名片;
    • 有的话,直接返回;
    • 没有的话,回落昵称。

ShowCode

# 获取所有群成员及名片
members = {}
contacts = self.query_sql("MicroMsg.db", "SELECT UserName, NickName FROM Contact;")
contacts = {contact["UserName"]: contact["NickName"]for contact in contacts}
crs = self.query_sql("MicroMsg.db", f"SELECT RoomData FROM ChatRoom WHERE ChatRoomName = '{roomid}';")
if not crs:
    return members

bs = crs[0].get("RoomData")
if not bs:
    return members

crd = RoomData()
crd.ParseFromString(bs)
if not bs:
    return members

for member in crd.members:
    members[member.wxid] = member.name if member.name else contacts.get(member.wxid, "")

有时候,我们并不需要所有群成员及名片,只需要查询某个成员的名片:

nickname = self.query_sql("MicroMsg.db", f"SELECT NickName FROM Contact WHERE UserName = '{wxid}';")
if not nickname:
    return ""

nickname = nickname[0].get("NickName", "")

crs = self.query_sql("MicroMsg.db", f"SELECT RoomData FROM ChatRoom WHERE ChatRoomName = '{roomid}';")
if not crs:
    return ""

bs = crs[0].get("RoomData")
if not bs:
    return ""

crd = RoomData()
crd.ParseFromString(bs)
for member in crd.members:
    if member.wxid == wxid:
        return member.name if member.name else nickname

return ""

好吧,简单分享一下,我又要搞事情去了。

Logo

DAMO开发者矩阵,由阿里巴巴达摩院和中国互联网协会联合发起,致力于探讨最前沿的技术趋势与应用成果,搭建高质量的交流与分享平台,推动技术创新与产业应用链接,围绕“人工智能与新型计算”构建开放共享的开发者生态。

更多推荐