import { RequestUtil } from '@/common/request'; import { GroupEssenceMsgRet, InstanceContext, WebApiGroupMember, WebApiGroupMemberRet, WebApiGroupNoticeRet, WebHonorType, } from '@/core'; import { NapCatCore } from '..'; import { createReadStream, readFileSync, statSync } from 'node:fs'; import { createHash } from 'node:crypto'; import { basename } from 'node:path'; export class NTQQWebApi { context: InstanceContext; core: NapCatCore; constructor(context: InstanceContext, core: NapCatCore) { this.context = context; this.core = core; } async shareDigest(groupCode: string, msgSeq: string, msgRandom: string, targetGroupCode: string) { const cookieObject = await this.core.apis.UserApi.getCookies('qun.qq.com'); const url = `https://qun.qq.com/cgi-bin/group_digest/share_digest?${new URLSearchParams({ bkn: this.getBknFromCookie(cookieObject), group_code: groupCode, msg_seq: msgSeq, msg_random: msgRandom, target_group_code: targetGroupCode, }).toString()}`; try { return RequestUtil.HttpGetText(url, 'GET', '', { 'Cookie': this.cookieToString(cookieObject) }); } catch (e) { return undefined; } } async getGroupEssenceMsgAll(GroupCode: string) { const ret: GroupEssenceMsgRet[] = []; for (let i = 0; i < 20; i++) { const data = await this.getGroupEssenceMsg(GroupCode, i, 50); if (!data) break; ret.push(data); if (data.data.is_end) break; } return ret; } async getGroupEssenceMsg(GroupCode: string, page_start: number = 0, page_limit: number = 50) { const cookieObject = await this.core.apis.UserApi.getCookies('qun.qq.com'); const url = `https://qun.qq.com/cgi-bin/group_digest/digest_list?${new URLSearchParams({ bkn: this.getBknFromCookie(cookieObject), page_start: page_start.toString(), page_limit: page_limit.toString(), group_code: GroupCode, }).toString()}`; try { const ret = await RequestUtil.HttpGetJson( url, 'GET', '', { 'Cookie': this.cookieToString(cookieObject) } ); return ret.retcode === 0 ? ret : undefined; } catch { return undefined; } } async getGroupMembers(GroupCode: string, cached: boolean = true): Promise { //logDebug('webapi 获取群成员', GroupCode); const memberData: Array = new Array(); const cookieObject = await this.core.apis.UserApi.getCookies('qun.qq.com'); const retList: Promise[] = []; const fastRet = await RequestUtil.HttpGetJson( `https://qun.qq.com/cgi-bin/qun_mgr/search_group_members?${new URLSearchParams({ st: '0', end: '40', sort: '1', gc: GroupCode, bkn: this.getBknFromCookie(cookieObject), }).toString()}`, 'POST', '', { 'Cookie': this.cookieToString(cookieObject) } ); if (!fastRet?.count || fastRet?.errcode !== 0 || !fastRet?.mems) { return []; } else { for (const key in fastRet.mems) { memberData.push(fastRet.mems[key]); } } //初始化获取PageNum const PageNum = Math.ceil(fastRet.count / 40); //遍历批量请求 for (let i = 2; i <= PageNum; i++) { const ret = RequestUtil.HttpGetJson( `https://qun.qq.com/cgi-bin/qun_mgr/search_group_members?${new URLSearchParams({ st: ((i - 1) * 40).toString(), end: (i * 40).toString(), sort: '1', gc: GroupCode, bkn: this.getBknFromCookie(cookieObject), }).toString()}`, 'POST', '', { 'Cookie': this.cookieToString(cookieObject) } ); retList.push(ret); } //批量等待 for (let i = 1; i <= PageNum; i++) { const ret = await (retList[i]); if (!ret?.count || ret?.errcode !== 0 || !ret?.mems) { continue; } for (const key in ret.mems) { memberData.push(ret.mems[key]); } } return memberData; } // public async addGroupDigest(groupCode: string, msgSeq: string) { // const url = `https://qun.qq.com/cgi-bin/group_digest/cancel_digest?random=665&X-CROSS-ORIGIN=fetch&group_code=${groupCode}&msg_seq=${msgSeq}&msg_random=444021292`; // const res = await this.request(url); // return await res.json(); // } // public async getGroupDigest(groupCode: string) { // const url = `https://qun.qq.com/cgi-bin/group_digest/digest_list?random=665&X-CROSS-ORIGIN=fetch&group_code=${groupCode}&page_start=0&page_limit=20`; // const res = await this.request(url); // return await res.json(); // } async setGroupNotice( GroupCode: string, Content: string, pinned: number = 0, type: number = 1, is_show_edit_card: number = 1, tip_window_type: number = 1, confirm_required: number = 1, picId: string = '', imgWidth: number = 540, imgHeight: number = 300, ) { interface SetNoticeRetSuccess { ec: number; em: string; id: number; ltsm: number; new_fid: string; read_only: number; role: number; srv_code: number; } const cookieObject = await this.core.apis.UserApi.getCookies('qun.qq.com'); try { const settings = JSON.stringify({ is_show_edit_card: is_show_edit_card, tip_window_type: tip_window_type, confirm_required: confirm_required }); const externalParam = { pic: picId, imgWidth: imgWidth.toString(), imgHeight: imgHeight.toString(), }; const ret: SetNoticeRetSuccess = await RequestUtil.HttpGetJson( `https://web.qun.qq.com/cgi-bin/announce/add_qun_notice?${new URLSearchParams({ bkn: this.getBknFromCookie(cookieObject), qid: GroupCode, text: Content, pinned: pinned.toString(), type: type.toString(), settings: settings, ...(picId === '' ? {} : externalParam) }).toString()}`, 'POST', '', { 'Cookie': this.cookieToString(cookieObject) } ); return ret; } catch (e) { return undefined; } } async getGroupNotice(GroupCode: string): Promise { const cookieObject = await this.core.apis.UserApi.getCookies('qun.qq.com'); try { const ret = await RequestUtil.HttpGetJson( `https://web.qun.qq.com/cgi-bin/announce/get_t_list?${new URLSearchParams({ bkn: this.getBknFromCookie(cookieObject), qid: GroupCode, ft: '23', ni: '1', n: '1', i: '1', log_read: '1', platform: '1', s: '-1', }).toString()}&n=20`, 'GET', '', { 'Cookie': this.cookieToString(cookieObject) } ); return ret?.ec === 0 ? ret : undefined; } catch (e) { return undefined; } } private async getDataInternal(cookieObject: any, groupCode: string, type: number) { let resJson; try { const res = await RequestUtil.HttpGetText( `https://qun.qq.com/interactive/honorlist?${new URLSearchParams({ gc: groupCode, type: type.toString(), }).toString()}`, 'GET', '', { 'Cookie': this.cookieToString(cookieObject) } ); const match = /window\.__INITIAL_STATE__=(.*?);/.exec(res); if (match) { resJson = JSON.parse(match[1].trim()); } return type === 1 ? resJson?.talkativeList : resJson?.actorList; } catch (e) { this.context.logger.logDebug('获取当前群荣耀失败', e); return undefined; } } private async getHonorList(cookieObject: any, groupCode: string, type: number) { const data = await this.getDataInternal(cookieObject, groupCode, type); if (!data) { this.context.logger.logError(`获取类型 ${type} 的荣誉信息失败`); return []; } return data.map((item: any) => ({ user_id: item?.uin, nickname: item?.name, avatar: item?.avatar, description: item?.desc, })); } async getGroupHonorInfo(groupCode: string, getType: WebHonorType) { const cookieObject = await this.core.apis.UserApi.getCookies('qun.qq.com'); const HonorInfo: any = { group_id: groupCode }; if (getType === WebHonorType.TALKATIVE || getType === WebHonorType.ALL) { const talkativeList = await this.getHonorList(cookieObject, groupCode, 1); if (talkativeList.length > 0) { HonorInfo.current_talkative = talkativeList[0]; HonorInfo.talkative_list = talkativeList; } } if (getType === WebHonorType.PERFORMER || getType === WebHonorType.ALL) { HonorInfo.performer_list = await this.getHonorList(cookieObject, groupCode, 2); } if (getType === WebHonorType.LEGEND || getType === WebHonorType.ALL) { HonorInfo.legend_list = await this.getHonorList(cookieObject, groupCode, 3); } if (getType === WebHonorType.EMOTION || getType === WebHonorType.ALL) { HonorInfo.emotion_list = await this.getHonorList(cookieObject, groupCode, 6); } // 冒尖小春笋好像已经被tx扬了 R.I.P. if (getType === WebHonorType.EMOTION || getType === WebHonorType.ALL) { HonorInfo.strong_newbie_list = []; } return HonorInfo; } private cookieToString(cookieObject: any) { return Object.entries(cookieObject).map(([key, value]) => `${key}=${value}`).join('; '); } public getBknFromCookie(cookieObject: any) { const sKey = cookieObject.skey as string; let hash = 5381; for (let i = 0; i < sKey.length; i++) { const code = sKey.charCodeAt(i); hash = hash + (hash << 5) + code; } return (hash & 0x7FFFFFFF).toString(); } public getBknFromSKey(sKey: string) { let hash = 5381; for (let i = 0; i < sKey.length; i++) { const code = sKey.charCodeAt(i); hash = hash + (hash << 5) + code; } return (hash & 0x7FFFFFFF).toString(); } async createQunAlbumSession(gc: string, sAlbumID: string, sAlbumName: string, path: string, skey: string, pskey: string, uin: string) { const img = readFileSync(path); const img_md5 = createHash('md5').update(img).digest('hex'); const img_size = img.length; const img_name = basename(path); const time = Math.floor(Date.now() / 1000); const GTK = this.getBknFromSKey(pskey); const cookie = `p_uin=${uin}; p_skey=${pskey}; skey=${skey}; uin=${uin}`; const body = { control_req: [{ uin: uin, token: { type: 4, data: pskey, appid: 5 }, appid: "qun", checksum: img_md5, check_type: 0, file_len: img_size, env: { refer: "qzone", deviceInfo: "h5" }, model: 0, biz_req: { sPicTitle: img_name, sPicDesc: "", sAlbumName: sAlbumName, sAlbumID: sAlbumID, iAlbumTypeID: 0, iBitmap: 0, iUploadType: 0, iUpPicType: 0, iBatchID: time, sPicPath: "", iPicWidth: 0, iPicHight: 0, iWaterType: 0, iDistinctUse: 0, iNeedFeeds: 1, iUploadTime: time, mapExt: { appid: "qun", userid: gc } }, session: "", asy_upload: 0, cmd: "FileUpload" }] }; const api = `https://h5.qzone.qq.com/webapp/json/sliceUpload/FileBatchControl/${img_md5}?g_tk=${GTK}`; const post = await RequestUtil.HttpGetJson(api, 'POST', body, { "Cookie": cookie, "Content-Type": "application/json" }); return post; } }