mirror of
https://github.com/NapNeko/NapCatQQ.git
synced 2025-07-19 12:03:37 +00:00
Compare commits
26 Commits
Author | SHA1 | Date | |
---|---|---|---|
![]() |
cfae4f5acd | ||
![]() |
de541e3249 | ||
![]() |
f5187c5c01 | ||
![]() |
9936279443 | ||
![]() |
2818773fd4 | ||
![]() |
b9293cbcd0 | ||
![]() |
5b9e44ddfc | ||
![]() |
1791accab7 | ||
![]() |
08081360f3 | ||
![]() |
e933a95e97 | ||
![]() |
4ef457fe6f | ||
![]() |
bd9cae8921 | ||
![]() |
303a74f8fd | ||
![]() |
0b7f126ce1 | ||
![]() |
308b5c027f | ||
![]() |
ed3abc4b43 | ||
![]() |
87ecb3b380 | ||
![]() |
7e31763a25 | ||
![]() |
c9df57d16a | ||
![]() |
3d0f8ee657 | ||
![]() |
6421bb4f5c | ||
![]() |
3919743885 | ||
![]() |
a5a57b9e20 | ||
![]() |
e31d2810ad | ||
![]() |
140e62fdcd | ||
![]() |
014b4deb87 |
10
README.md
10
README.md
@@ -1,8 +1,9 @@
|
|||||||
|
<img src="https://napneko.github.io/assets/newnewlogo.png" width = "305" height = "411" alt="NapCat" align=right />
|
||||||
<div align="center">
|
<div align="center">
|
||||||
|
|
||||||
# NapCat
|
# NapCat
|
||||||
|
|
||||||

|
|
||||||
|
|
||||||
_Modern protocol-side framework implemented based on NTQQ._
|
_Modern protocol-side framework implemented based on NTQQ._
|
||||||
|
|
||||||
@@ -40,8 +41,11 @@ _Modern protocol-side framework implemented based on NTQQ._
|
|||||||
| Docs | [](https://napneko.pages.dev/) | [](https://napcat.cyou/) | [](https://www.napcat.wiki) |
|
| Docs | [](https://napneko.pages.dev/) | [](https://napcat.cyou/) | [](https://www.napcat.wiki) |
|
||||||
|:-:|:-:|:-:|:-:|
|
|:-:|:-:|:-:|:-:|
|
||||||
|
|
||||||
| Contact | [](https://qm.qq.com/q/I6LU87a0Yq) | [](https://qm.qq.com/q/HaRcfrHpUk) | [](https://t.me/MelodicMoonlight) |
|
| QQ Group | [](https://qm.qq.com/q/CMmPbGw0jA) | [](https://qm.qq.com/q/8zJMLjqy2Y) | [](https://qm.qq.com/q/HaRcfrHpUk) | [](https://qm.qq.com/q/I6LU87a0Yq) |
|
||||||
|:-:|:-:|:-:|:-:|
|
|:-:|:-:|:-:|:-:|:-:|
|
||||||
|
|
||||||
|
| Telegram | [](https://t.me/MelodicMoonlight) |
|
||||||
|
|:-:|:-:|
|
||||||
|
|
||||||
## Thanks
|
## Thanks
|
||||||
|
|
||||||
|
Binary file not shown.
@@ -4,7 +4,7 @@
|
|||||||
"name": "NapCatQQ",
|
"name": "NapCatQQ",
|
||||||
"slug": "NapCat.Framework",
|
"slug": "NapCat.Framework",
|
||||||
"description": "高性能的 OneBot 11 协议实现",
|
"description": "高性能的 OneBot 11 协议实现",
|
||||||
"version": "4.7.43",
|
"version": "4.5.50",
|
||||||
"icon": "./logo.png",
|
"icon": "./logo.png",
|
||||||
"authors": [
|
"authors": [
|
||||||
{
|
{
|
||||||
|
@@ -2,7 +2,7 @@
|
|||||||
"name": "napcat",
|
"name": "napcat",
|
||||||
"private": true,
|
"private": true,
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"version": "4.7.43",
|
"version": "4.5.50",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"build:universal": "npm run build:webui && vite build --mode universal || exit 1",
|
"build:universal": "npm run build:webui && vite build --mode universal || exit 1",
|
||||||
"build:framework": "npm run build:webui && vite build --mode framework || exit 1",
|
"build:framework": "npm run build:webui && vite build --mode framework || exit 1",
|
||||||
|
@@ -145,8 +145,8 @@ export enum FileUriType {
|
|||||||
|
|
||||||
export async function checkUriType(Uri: string) {
|
export async function checkUriType(Uri: string) {
|
||||||
const LocalFileRet = await solveProblem((uri: string) => {
|
const LocalFileRet = await solveProblem((uri: string) => {
|
||||||
if (fs.existsSync(uri)) {
|
if (fs.existsSync(path.normalize(uri))) {
|
||||||
return { Uri: uri, Type: FileUriType.Local };
|
return { Uri: path.normalize(uri), Type: FileUriType.Local };
|
||||||
}
|
}
|
||||||
return undefined;
|
return undefined;
|
||||||
}, Uri);
|
}, Uri);
|
||||||
|
@@ -1 +1 @@
|
|||||||
export const napCatVersion = '4.7.43';
|
export const napCatVersion = '4.5.50';
|
||||||
|
@@ -28,6 +28,8 @@ import { SendMessageContext } from '@/onebot/api';
|
|||||||
import { getFileTypeForSendType } from '../helper/msg';
|
import { getFileTypeForSendType } from '../helper/msg';
|
||||||
import { FFmpegService } from '@/common/ffmpeg';
|
import { FFmpegService } from '@/common/ffmpeg';
|
||||||
import { rkeyDataType } from '../types/file';
|
import { rkeyDataType } from '../types/file';
|
||||||
|
import { NapProtoMsg } from '@napneko/nap-proto-core';
|
||||||
|
import { FileId } from '../packet/transformer/proto/misc/fileid';
|
||||||
|
|
||||||
export class NTQQFileApi {
|
export class NTQQFileApi {
|
||||||
context: InstanceContext;
|
context: InstanceContext;
|
||||||
@@ -63,6 +65,76 @@ export class NTQQFileApi {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async getFileUrl(chatType: ChatType, peer: string, fileUUID?: string, file10MMd5?: string | undefined) {
|
||||||
|
if (this.core.apis.PacketApi.available) {
|
||||||
|
try {
|
||||||
|
if (chatType === ChatType.KCHATTYPEGROUP && fileUUID) {
|
||||||
|
return this.core.apis.PacketApi.pkt.operation.GetGroupFileUrl(+peer, fileUUID);
|
||||||
|
} else if (file10MMd5 && fileUUID) {
|
||||||
|
return this.core.apis.PacketApi.pkt.operation.GetPrivateFileUrl(peer, fileUUID, file10MMd5);
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
this.context.logger.logError('获取文件URL失败', (error as Error).message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
throw new Error('fileUUID or file10MMd5 is undefined');
|
||||||
|
}
|
||||||
|
|
||||||
|
async getPttUrl(peer: string, fileUUID?: string) {
|
||||||
|
if (this.core.apis.PacketApi.available && fileUUID) {
|
||||||
|
let appid = new NapProtoMsg(FileId).decode(Buffer.from(fileUUID.replaceAll('-', '+').replaceAll('_', '/'), 'base64')).appid;
|
||||||
|
try {
|
||||||
|
if (appid && appid === 1403) {
|
||||||
|
return this.core.apis.PacketApi.pkt.operation.GetGroupPttUrl(+peer, {
|
||||||
|
fileUuid: fileUUID,
|
||||||
|
storeId: 1,
|
||||||
|
uploadTime: 0,
|
||||||
|
ttl: 0,
|
||||||
|
subType: 0,
|
||||||
|
});
|
||||||
|
} else if (fileUUID) {
|
||||||
|
return this.core.apis.PacketApi.pkt.operation.GetPttUrl(peer, {
|
||||||
|
fileUuid: fileUUID,
|
||||||
|
storeId: 1,
|
||||||
|
uploadTime: 0,
|
||||||
|
ttl: 0,
|
||||||
|
subType: 0,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
this.context.logger.logError('获取文件URL失败', (error as Error).message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
throw new Error('packet cant get ptt url');
|
||||||
|
}
|
||||||
|
|
||||||
|
async getVideoUrlPacket(peer: string, fileUUID?: string) {
|
||||||
|
if (this.core.apis.PacketApi.available && fileUUID) {
|
||||||
|
let appid = new NapProtoMsg(FileId).decode(Buffer.from(fileUUID.replaceAll('-', '+').replaceAll('_', '/'), 'base64')).appid;
|
||||||
|
try {
|
||||||
|
if (appid && appid === 1415) {
|
||||||
|
return this.core.apis.PacketApi.pkt.operation.GetGroupVideoUrl(+peer, {
|
||||||
|
fileUuid: fileUUID,
|
||||||
|
storeId: 1,
|
||||||
|
uploadTime: 0,
|
||||||
|
ttl: 0,
|
||||||
|
subType: 0,
|
||||||
|
});
|
||||||
|
} else if (fileUUID) {
|
||||||
|
return this.core.apis.PacketApi.pkt.operation.GetVideoUrl(peer, {
|
||||||
|
fileUuid: fileUUID,
|
||||||
|
storeId: 1,
|
||||||
|
uploadTime: 0,
|
||||||
|
ttl: 0,
|
||||||
|
subType: 0,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
this.context.logger.logError('获取文件URL失败', (error as Error).message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
throw new Error('packet cant get video url');
|
||||||
|
}
|
||||||
|
|
||||||
async copyFile(filePath: string, destPath: string) {
|
async copyFile(filePath: string, destPath: string) {
|
||||||
await this.core.util.copyFile(filePath, destPath);
|
await this.core.util.copyFile(filePath, destPath);
|
||||||
|
@@ -71,6 +71,7 @@ export class NTQQMsgApi {
|
|||||||
async queryMsgsWithFilterExWithSeq(peer: Peer, msgSeq: string) {
|
async queryMsgsWithFilterExWithSeq(peer: Peer, msgSeq: string) {
|
||||||
return await this.context.session.getMsgService().queryMsgsWithFilterEx('0', '0', msgSeq, {
|
return await this.context.session.getMsgService().queryMsgsWithFilterEx('0', '0', msgSeq, {
|
||||||
chatInfo: peer,
|
chatInfo: peer,
|
||||||
|
//searchFields: 3,
|
||||||
filterMsgType: [],
|
filterMsgType: [],
|
||||||
filterSendersUid: [],
|
filterSendersUid: [],
|
||||||
filterMsgToTime: '0',
|
filterMsgToTime: '0',
|
||||||
@@ -84,6 +85,7 @@ export class NTQQMsgApi {
|
|||||||
return await this.context.session.getMsgService().queryMsgsWithFilterEx('0', '0', msgSeq, {
|
return await this.context.session.getMsgService().queryMsgsWithFilterEx('0', '0', msgSeq, {
|
||||||
chatInfo: peer,
|
chatInfo: peer,
|
||||||
filterMsgType: [],
|
filterMsgType: [],
|
||||||
|
//searchFields: 3,
|
||||||
filterSendersUid: SendersUid,
|
filterSendersUid: SendersUid,
|
||||||
filterMsgToTime: MsgTime,
|
filterMsgToTime: MsgTime,
|
||||||
filterMsgFromTime: MsgTime,
|
filterMsgFromTime: MsgTime,
|
||||||
@@ -100,6 +102,7 @@ export class NTQQMsgApi {
|
|||||||
filterMsgToTime: '0',
|
filterMsgToTime: '0',
|
||||||
filterMsgFromTime: '0',
|
filterMsgFromTime: '0',
|
||||||
isReverseOrder: false,
|
isReverseOrder: false,
|
||||||
|
//searchFields: 3,
|
||||||
isIncludeCurrent: true,
|
isIncludeCurrent: true,
|
||||||
pageLimit: 1,
|
pageLimit: 1,
|
||||||
});
|
});
|
||||||
@@ -110,6 +113,7 @@ export class NTQQMsgApi {
|
|||||||
filterMsgType: [],
|
filterMsgType: [],
|
||||||
filterSendersUid: [],
|
filterSendersUid: [],
|
||||||
filterMsgToTime: '0',
|
filterMsgToTime: '0',
|
||||||
|
//searchFields: 3,
|
||||||
filterMsgFromTime: '0',
|
filterMsgFromTime: '0',
|
||||||
isReverseOrder: true,
|
isReverseOrder: true,
|
||||||
isIncludeCurrent: true,
|
isIncludeCurrent: true,
|
||||||
@@ -128,6 +132,7 @@ export class NTQQMsgApi {
|
|||||||
chatInfo: peer,//此处为Peer 为关键查询参数 没有啥也没有 by mlik iowa
|
chatInfo: peer,//此处为Peer 为关键查询参数 没有啥也没有 by mlik iowa
|
||||||
filterMsgType: [],
|
filterMsgType: [],
|
||||||
filterSendersUid: [],
|
filterSendersUid: [],
|
||||||
|
//searchFields: 3,
|
||||||
filterMsgToTime: filterMsgToTime,
|
filterMsgToTime: filterMsgToTime,
|
||||||
filterMsgFromTime: filterMsgFromTime,
|
filterMsgFromTime: filterMsgFromTime,
|
||||||
isReverseOrder: false,
|
isReverseOrder: false,
|
||||||
@@ -142,6 +147,7 @@ export class NTQQMsgApi {
|
|||||||
chatInfo: peer,
|
chatInfo: peer,
|
||||||
filterMsgType: [],
|
filterMsgType: [],
|
||||||
filterSendersUid: SendersUid,
|
filterSendersUid: SendersUid,
|
||||||
|
//searchFields: 3,
|
||||||
filterMsgToTime: '0',
|
filterMsgToTime: '0',
|
||||||
filterMsgFromTime: '0',
|
filterMsgFromTime: '0',
|
||||||
isReverseOrder: true,
|
isReverseOrder: true,
|
||||||
|
8
src/core/external/appid.json
vendored
8
src/core/external/appid.json
vendored
@@ -274,5 +274,13 @@
|
|||||||
"9.9.19-34606": {
|
"9.9.19-34606": {
|
||||||
"appid": 537282307,
|
"appid": 537282307,
|
||||||
"qua": "V1_WIN_NQ_9.9.19_34606_GW_B"
|
"qua": "V1_WIN_NQ_9.9.19_34606_GW_B"
|
||||||
|
},
|
||||||
|
"9.9.19-34740": {
|
||||||
|
"appid": 537290691,
|
||||||
|
"qua": "V1_WIN_NQ_9.9.19_34740_GW_B"
|
||||||
|
},
|
||||||
|
"3.2.17-34740": {
|
||||||
|
"appid": 537290727,
|
||||||
|
"qua": "V1_LNX_NQ_3.2.17_34740_GW_B"
|
||||||
}
|
}
|
||||||
}
|
}
|
4
src/core/external/offset.json
vendored
4
src/core/external/offset.json
vendored
@@ -350,5 +350,9 @@
|
|||||||
"3.2.17-34606-arm64": {
|
"3.2.17-34606-arm64": {
|
||||||
"send": "7711270",
|
"send": "7711270",
|
||||||
"recv": "7714BA0"
|
"recv": "7714BA0"
|
||||||
|
},
|
||||||
|
"9.9.19-34740-x64": {
|
||||||
|
"send": "3BDD8D0",
|
||||||
|
"recv": "3BE20D0"
|
||||||
}
|
}
|
||||||
}
|
}
|
@@ -6,13 +6,14 @@ import {
|
|||||||
PacketMsgFileElement,
|
PacketMsgFileElement,
|
||||||
PacketMsgPicElement,
|
PacketMsgPicElement,
|
||||||
PacketMsgPttElement,
|
PacketMsgPttElement,
|
||||||
PacketMsgVideoElement
|
PacketMsgReplyElement,
|
||||||
|
PacketMsgVideoElement,
|
||||||
} from '@/core/packet/message/element';
|
} from '@/core/packet/message/element';
|
||||||
import { ChatType, MsgSourceType, NTMsgType, RawMessage } from '@/core';
|
import { ChatType, MsgSourceType, NTMsgType, RawMessage } from '@/core';
|
||||||
import { MiniAppRawData, MiniAppReqParams } from '@/core/packet/entities/miniApp';
|
import { MiniAppRawData, MiniAppReqParams } from '@/core/packet/entities/miniApp';
|
||||||
import { AIVoiceChatType } from '@/core/packet/entities/aiChat';
|
import { AIVoiceChatType } from '@/core/packet/entities/aiChat';
|
||||||
import { NapProtoDecodeStructType, NapProtoEncodeStructType, NapProtoMsg } from '@napneko/nap-proto-core';
|
import { NapProtoDecodeStructType, NapProtoEncodeStructType, NapProtoMsg } from '@napneko/nap-proto-core';
|
||||||
import { IndexNode, LongMsgResult, MsgInfo } from '@/core/packet/transformer/proto';
|
import { IndexNode, LongMsgResult, MsgInfo, PushMsgBody } from '@/core/packet/transformer/proto';
|
||||||
import { OidbPacket } from '@/core/packet/transformer/base';
|
import { OidbPacket } from '@/core/packet/transformer/base';
|
||||||
import { ImageOcrResult } from '@/core/packet/entities/ocrResult';
|
import { ImageOcrResult } from '@/core/packet/entities/ocrResult';
|
||||||
import { gunzipSync } from 'zlib';
|
import { gunzipSync } from 'zlib';
|
||||||
@@ -76,22 +77,24 @@ export class PacketOperationContext {
|
|||||||
async UploadResources(msg: PacketMsg[], groupUin: number = 0) {
|
async UploadResources(msg: PacketMsg[], groupUin: number = 0) {
|
||||||
const chatType = groupUin ? ChatType.KCHATTYPEGROUP : ChatType.KCHATTYPEC2C;
|
const chatType = groupUin ? ChatType.KCHATTYPEGROUP : ChatType.KCHATTYPEC2C;
|
||||||
const peerUid = groupUin ? String(groupUin) : this.context.napcore.basicInfo.uid;
|
const peerUid = groupUin ? String(groupUin) : this.context.napcore.basicInfo.uid;
|
||||||
const reqList = msg.flatMap(m =>
|
const reqList = msg.flatMap((m) =>
|
||||||
m.msg.map(e => {
|
m.msg
|
||||||
if (e instanceof PacketMsgPicElement) {
|
.map((e) => {
|
||||||
return this.context.highway.uploadImage({ chatType, peerUid }, e);
|
if (e instanceof PacketMsgPicElement) {
|
||||||
} else if (e instanceof PacketMsgVideoElement) {
|
return this.context.highway.uploadImage({ chatType, peerUid }, e);
|
||||||
return this.context.highway.uploadVideo({ chatType, peerUid }, e);
|
} else if (e instanceof PacketMsgVideoElement) {
|
||||||
} else if (e instanceof PacketMsgPttElement) {
|
return this.context.highway.uploadVideo({ chatType, peerUid }, e);
|
||||||
return this.context.highway.uploadPtt({ chatType, peerUid }, e);
|
} else if (e instanceof PacketMsgPttElement) {
|
||||||
} else if (e instanceof PacketMsgFileElement) {
|
return this.context.highway.uploadPtt({ chatType, peerUid }, e);
|
||||||
return this.context.highway.uploadFile({ chatType, peerUid }, e);
|
} else if (e instanceof PacketMsgFileElement) {
|
||||||
}
|
return this.context.highway.uploadFile({ chatType, peerUid }, e);
|
||||||
return null;
|
}
|
||||||
}).filter(Boolean)
|
return null;
|
||||||
|
})
|
||||||
|
.filter(Boolean)
|
||||||
);
|
);
|
||||||
const res = await Promise.allSettled(reqList);
|
const res = await Promise.allSettled(reqList);
|
||||||
this.context.logger.info(`上传资源${res.length}个,失败${res.filter(r => r.status === 'rejected').length}个`);
|
this.context.logger.info(`上传资源${res.length}个,失败${res.filter((r) => r.status === 'rejected').length}个`);
|
||||||
res.forEach((result, index) => {
|
res.forEach((result, index) => {
|
||||||
if (result.status === 'rejected') {
|
if (result.status === 'rejected') {
|
||||||
this.context.logger.error(`上传第${index + 1}个资源失败:${result.reason.stack}`);
|
this.context.logger.error(`上传第${index + 1}个资源失败:${result.reason.stack}`);
|
||||||
@@ -100,10 +103,13 @@ export class PacketOperationContext {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async UploadImage(img: PacketMsgPicElement) {
|
async UploadImage(img: PacketMsgPicElement) {
|
||||||
await this.context.highway.uploadImage({
|
await this.context.highway.uploadImage(
|
||||||
chatType: ChatType.KCHATTYPEC2C,
|
{
|
||||||
peerUid: this.context.napcore.basicInfo.uid
|
chatType: ChatType.KCHATTYPEC2C,
|
||||||
}, img);
|
peerUid: this.context.napcore.basicInfo.uid,
|
||||||
|
},
|
||||||
|
img
|
||||||
|
);
|
||||||
const index = img.msgInfo?.msgInfoBody?.at(0)?.index;
|
const index = img.msgInfo?.msgInfoBody?.at(0)?.index;
|
||||||
if (!index) {
|
if (!index) {
|
||||||
throw new Error('img.msgInfo?.msgInfoBody![0].index! is undefined');
|
throw new Error('img.msgInfo?.msgInfoBody![0].index! is undefined');
|
||||||
@@ -118,6 +124,20 @@ export class PacketOperationContext {
|
|||||||
return `https://${res.download.info.domain}${res.download.info.urlPath}${res.download.rKeyParam}`;
|
return `https://${res.download.info.domain}${res.download.info.urlPath}${res.download.rKeyParam}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async GetPttUrl(selfUid: string, node: NapProtoEncodeStructType<typeof IndexNode>) {
|
||||||
|
const req = trans.DownloadPtt.build(selfUid, node);
|
||||||
|
const resp = await this.context.client.sendOidbPacket(req, true);
|
||||||
|
const res = trans.DownloadPtt.parse(resp);
|
||||||
|
return `https://${res.download.info.domain}${res.download.info.urlPath}${res.download.rKeyParam}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
async GetVideoUrl(selfUid: string, node: NapProtoEncodeStructType<typeof IndexNode>) {
|
||||||
|
const req = trans.DownloadVideo.build(selfUid, node);
|
||||||
|
const resp = await this.context.client.sendOidbPacket(req, true);
|
||||||
|
const res = trans.DownloadVideo.parse(resp);
|
||||||
|
return `https://${res.download.info.domain}${res.download.info.urlPath}${res.download.rKeyParam}`;
|
||||||
|
}
|
||||||
|
|
||||||
async GetGroupImageUrl(groupUin: number, node: NapProtoEncodeStructType<typeof IndexNode>) {
|
async GetGroupImageUrl(groupUin: number, node: NapProtoEncodeStructType<typeof IndexNode>) {
|
||||||
const req = trans.DownloadGroupImage.build(groupUin, node);
|
const req = trans.DownloadGroupImage.build(groupUin, node);
|
||||||
const resp = await this.context.client.sendOidbPacket(req, true);
|
const resp = await this.context.client.sendOidbPacket(req, true);
|
||||||
@@ -125,6 +145,21 @@ export class PacketOperationContext {
|
|||||||
return `https://${res.download.info.domain}${res.download.info.urlPath}${res.download.rKeyParam}`;
|
return `https://${res.download.info.domain}${res.download.info.urlPath}${res.download.rKeyParam}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async GetGroupPttUrl(groupUin: number, node: NapProtoEncodeStructType<typeof IndexNode>) {
|
||||||
|
const req = trans.DownloadGroupPtt.build(groupUin, node);
|
||||||
|
const resp = await this.context.client.sendOidbPacket(req, true);
|
||||||
|
const res = trans.DownloadImage.parse(resp);
|
||||||
|
return `https://${res.download.info.domain}${res.download.info.urlPath}${res.download.rKeyParam}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
async GetGroupVideoUrl(groupUin: number, node: NapProtoEncodeStructType<typeof IndexNode>) {
|
||||||
|
const req = trans.DownloadGroupVideo.build(groupUin, node);
|
||||||
|
const resp = await this.context.client.sendOidbPacket(req, true);
|
||||||
|
const res = trans.DownloadImage.parse(resp);
|
||||||
|
return `https://${res.download.info.domain}${res.download.info.urlPath}${res.download.rKeyParam}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
async ImageOCR(imgUrl: string) {
|
async ImageOCR(imgUrl: string) {
|
||||||
const req = trans.ImageOCR.build(imgUrl);
|
const req = trans.ImageOCR.build(imgUrl);
|
||||||
const resp = await this.context.client.sendOidbPacket(req, true);
|
const resp = await this.context.client.sendOidbPacket(req, true);
|
||||||
@@ -137,24 +172,66 @@ export class PacketOperationContext {
|
|||||||
coordinates: item.polygon.coordinates.map((c) => {
|
coordinates: item.polygon.coordinates.map((c) => {
|
||||||
return {
|
return {
|
||||||
x: c.x,
|
x: c.x,
|
||||||
y: c.y
|
y: c.y,
|
||||||
};
|
};
|
||||||
}),
|
}),
|
||||||
};
|
};
|
||||||
}),
|
}),
|
||||||
language: res.ocrRspBody.language
|
language: res.ocrRspBody.language,
|
||||||
} as ImageOcrResult;
|
} as ImageOcrResult;
|
||||||
}
|
}
|
||||||
|
|
||||||
async UploadForwardMsg(msg: PacketMsg[], groupUin: number = 0) {
|
private async SendPreprocess(msg: PacketMsg[], groupUin: number = 0) {
|
||||||
|
const ps = msg.map((m) => {
|
||||||
|
return m.msg.map(async (e) => {
|
||||||
|
if (e instanceof PacketMsgReplyElement && !e.targetElems) {
|
||||||
|
this.context.logger.debug(`Cannot find reply element's targetElems, prepare to fetch it...`);
|
||||||
|
if (!e.targetPeer?.peerUid) {
|
||||||
|
this.context.logger.error(`targetPeer is undefined!`);
|
||||||
|
}
|
||||||
|
let targetMsg: NapProtoEncodeStructType<typeof PushMsgBody>[] | undefined;
|
||||||
|
if (e.isGroupReply) {
|
||||||
|
targetMsg = await this.FetchGroupMessage(+(e.targetPeer?.peerUid ?? 0), e.targetMessageSeq, e.targetMessageSeq);
|
||||||
|
} else {
|
||||||
|
targetMsg = await this.FetchC2CMessage(await this.context.napcore.basicInfo.uin2uid(e.targetUin), e.targetMessageSeq, e.targetMessageSeq);
|
||||||
|
}
|
||||||
|
e.targetElems = targetMsg.at(0)?.body?.richText?.elems;
|
||||||
|
e.targetSourceMsg = targetMsg.at(0);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}).flat();
|
||||||
|
await Promise.all(ps)
|
||||||
await this.UploadResources(msg, groupUin);
|
await this.UploadResources(msg, groupUin);
|
||||||
|
}
|
||||||
|
|
||||||
|
async FetchGroupMessage(groupUin: number, startSeq: number, endSeq: number): Promise<NapProtoDecodeStructType<typeof PushMsgBody>[]> {
|
||||||
|
const req = trans.FetchGroupMessage.build(groupUin, startSeq, endSeq);
|
||||||
|
const resp = await this.context.client.sendOidbPacket(req, true);
|
||||||
|
const res = trans.FetchGroupMessage.parse(resp);
|
||||||
|
return res.body.messages
|
||||||
|
}
|
||||||
|
|
||||||
|
async FetchC2CMessage(targetUid: string, startSeq: number, endSeq: number): Promise<NapProtoDecodeStructType<typeof PushMsgBody>[]> {
|
||||||
|
const req = trans.FetchC2CMessage.build(targetUid, startSeq, endSeq);
|
||||||
|
const resp = await this.context.client.sendOidbPacket(req, true);
|
||||||
|
const res = trans.FetchC2CMessage.parse(resp);
|
||||||
|
return res.messages
|
||||||
|
}
|
||||||
|
|
||||||
|
async UploadForwardMsg(msg: PacketMsg[], groupUin: number = 0) {
|
||||||
|
await this.SendPreprocess(msg, groupUin);
|
||||||
const req = trans.UploadForwardMsg.build(this.context.napcore.basicInfo.uid, msg, groupUin);
|
const req = trans.UploadForwardMsg.build(this.context.napcore.basicInfo.uid, msg, groupUin);
|
||||||
const resp = await this.context.client.sendOidbPacket(req, true);
|
const resp = await this.context.client.sendOidbPacket(req, true);
|
||||||
const res = trans.UploadForwardMsg.parse(resp);
|
const res = trans.UploadForwardMsg.parse(resp);
|
||||||
return res.result.resId;
|
return res.result.resId;
|
||||||
}
|
}
|
||||||
|
|
||||||
async MoveGroupFile(groupUin: number, fileUUID: string, currentParentDirectory: string, targetParentDirectory: string) {
|
async MoveGroupFile(
|
||||||
|
groupUin: number,
|
||||||
|
fileUUID: string,
|
||||||
|
currentParentDirectory: string,
|
||||||
|
targetParentDirectory: string
|
||||||
|
) {
|
||||||
const req = trans.MoveGroupFile.build(groupUin, fileUUID, currentParentDirectory, targetParentDirectory);
|
const req = trans.MoveGroupFile.build(groupUin, fileUUID, currentParentDirectory, targetParentDirectory);
|
||||||
const resp = await this.context.client.sendOidbPacket(req, true);
|
const resp = await this.context.client.sendOidbPacket(req, true);
|
||||||
const res = trans.MoveGroupFile.parse(resp);
|
const res = trans.MoveGroupFile.parse(resp);
|
||||||
@@ -174,6 +251,7 @@ export class PacketOperationContext {
|
|||||||
const res = trans.DownloadGroupFile.parse(resp);
|
const res = trans.DownloadGroupFile.parse(resp);
|
||||||
return `https://${res.download.downloadDns}/ftn_handler/${Buffer.from(res.download.downloadUrl).toString('hex')}/?fname=`;
|
return `https://${res.download.downloadDns}/ftn_handler/${Buffer.from(res.download.downloadUrl).toString('hex')}/?fname=`;
|
||||||
}
|
}
|
||||||
|
|
||||||
async GetPrivateFileUrl(self_id: string, fileUUID: string, md5: string) {
|
async GetPrivateFileUrl(self_id: string, fileUUID: string, md5: string) {
|
||||||
const req = trans.DownloadPrivateFile.build(self_id, fileUUID, md5);
|
const req = trans.DownloadPrivateFile.build(self_id, fileUUID, md5);
|
||||||
const resp = await this.context.client.sendOidbPacket(req, true);
|
const resp = await this.context.client.sendOidbPacket(req, true);
|
||||||
@@ -181,13 +259,6 @@ export class PacketOperationContext {
|
|||||||
return `http://${res.body?.result?.server}:${res.body?.result?.port}${res.body?.result?.url?.slice(8)}&isthumb=0`;
|
return `http://${res.body?.result?.server}:${res.body?.result?.port}${res.body?.result?.url?.slice(8)}&isthumb=0`;
|
||||||
}
|
}
|
||||||
|
|
||||||
async GetGroupPttUrl(groupUin: number, node: NapProtoEncodeStructType<typeof IndexNode>) {
|
|
||||||
const req = trans.DownloadGroupPtt.build(groupUin, node);
|
|
||||||
const resp = await this.context.client.sendOidbPacket(req, true);
|
|
||||||
const res = trans.DownloadGroupPtt.parse(resp);
|
|
||||||
return `https://${res.download.info.domain}${res.download.info.urlPath}${res.download.rKeyParam}`;
|
|
||||||
}
|
|
||||||
|
|
||||||
async GetMiniAppAdaptShareInfo(param: MiniAppReqParams) {
|
async GetMiniAppAdaptShareInfo(param: MiniAppReqParams) {
|
||||||
const req = trans.GetMiniAppAdaptShareInfo.build(param);
|
const req = trans.GetMiniAppAdaptShareInfo.build(param);
|
||||||
const resp = await this.context.client.sendOidbPacket(req, true);
|
const resp = await this.context.client.sendOidbPacket(req, true);
|
||||||
@@ -203,12 +274,17 @@ export class PacketOperationContext {
|
|||||||
return res.content.map((item) => {
|
return res.content.map((item) => {
|
||||||
return {
|
return {
|
||||||
category: item.category,
|
category: item.category,
|
||||||
voices: item.voices
|
voices: item.voices,
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
async GetAiVoice(groupUin: number, voiceId: string, text: string, chatType: AIVoiceChatType): Promise<NapProtoDecodeStructType<typeof MsgInfo>> {
|
async GetAiVoice(
|
||||||
|
groupUin: number,
|
||||||
|
voiceId: string,
|
||||||
|
text: string,
|
||||||
|
chatType: AIVoiceChatType
|
||||||
|
): Promise<NapProtoDecodeStructType<typeof MsgInfo>> {
|
||||||
let reqTime = 0;
|
let reqTime = 0;
|
||||||
const reqMaxTime = 30;
|
const reqMaxTime = 30;
|
||||||
const sessionId = crypto.randomBytes(4).readUInt32BE(0);
|
const sessionId = crypto.randomBytes(4).readUInt32BE(0);
|
||||||
@@ -236,6 +312,7 @@ export class PacketOperationContext {
|
|||||||
if (!main?.actionData.msgBody) {
|
if (!main?.actionData.msgBody) {
|
||||||
throw new Error('msgBody is empty');
|
throw new Error('msgBody is empty');
|
||||||
}
|
}
|
||||||
|
this.context.logger.debug('rawChains ', inflate.toString('hex'));
|
||||||
|
|
||||||
const messagesPromises = main.actionData.msgBody.map(async (msg) => {
|
const messagesPromises = main.actionData.msgBody.map(async (msg) => {
|
||||||
if (!msg?.body?.richText?.elems) {
|
if (!msg?.body?.richText?.elems) {
|
||||||
@@ -251,12 +328,12 @@ export class PacketOperationContext {
|
|||||||
const groupUin = msg?.responseHead.grp?.groupUin ?? 0;
|
const groupUin = msg?.responseHead.grp?.groupUin ?? 0;
|
||||||
element.picElement = {
|
element.picElement = {
|
||||||
...element.picElement,
|
...element.picElement,
|
||||||
originImageUrl: await this.GetGroupImageUrl(groupUin, index!)
|
originImageUrl: await this.GetGroupImageUrl(groupUin, index!),
|
||||||
};
|
};
|
||||||
} else {
|
} else {
|
||||||
element.picElement = {
|
element.picElement = {
|
||||||
...element.picElement,
|
...element.picElement,
|
||||||
originImageUrl: await this.GetImageUrl(this.context.napcore.basicInfo.uid, index!)
|
originImageUrl: await this.GetImageUrl(this.context.napcore.basicInfo.uid, index!),
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
return element;
|
return element;
|
||||||
@@ -269,7 +346,7 @@ export class PacketOperationContext {
|
|||||||
elements: elements,
|
elements: elements,
|
||||||
guildId: '',
|
guildId: '',
|
||||||
isOnlineMsg: false,
|
isOnlineMsg: false,
|
||||||
msgId: '7467703692092974645', // TODO: no necessary
|
msgId: '7467703692092974645', // TODO: no necessary
|
||||||
msgRandom: '0',
|
msgRandom: '0',
|
||||||
msgSeq: String(msg.contentHead.sequence ?? 0),
|
msgSeq: String(msg.contentHead.sequence ?? 0),
|
||||||
msgTime: String(msg.contentHead.timeStamp ?? 0),
|
msgTime: String(msg.contentHead.timeStamp ?? 0),
|
||||||
|
@@ -24,12 +24,15 @@ export class PacketMsgBuilder {
|
|||||||
}
|
}
|
||||||
return {
|
return {
|
||||||
responseHead: {
|
responseHead: {
|
||||||
fromUid: '',
|
|
||||||
fromUin: node.senderUin,
|
fromUin: node.senderUin,
|
||||||
toUid: node.groupId ? undefined : selfUid,
|
type: 0,
|
||||||
|
sigMap: 0,
|
||||||
|
toUin: 0,
|
||||||
|
fromUid: '',
|
||||||
forward: node.groupId ? undefined : {
|
forward: node.groupId ? undefined : {
|
||||||
friendName: node.senderName,
|
friendName: node.senderName,
|
||||||
},
|
},
|
||||||
|
toUid: node.groupId ? undefined : selfUid,
|
||||||
grp: node.groupId ? {
|
grp: node.groupId ? {
|
||||||
groupUin: node.groupId,
|
groupUin: node.groupId,
|
||||||
memberName: node.senderName,
|
memberName: node.senderName,
|
||||||
@@ -40,16 +43,13 @@ export class PacketMsgBuilder {
|
|||||||
type: node.groupId ? 82 : 9,
|
type: node.groupId ? 82 : 9,
|
||||||
subType: node.groupId ? undefined : 4,
|
subType: node.groupId ? undefined : 4,
|
||||||
divSeq: node.groupId ? undefined : 4,
|
divSeq: node.groupId ? undefined : 4,
|
||||||
msgId: crypto.randomBytes(4).readUInt32LE(0),
|
autoReply: 0,
|
||||||
sequence: crypto.randomBytes(4).readUInt32LE(0),
|
sequence: crypto.randomBytes(4).readUInt32LE(0),
|
||||||
timeStamp: +node.time.toString().substring(0, 10),
|
timeStamp: +node.time.toString().substring(0, 10),
|
||||||
field7: BigInt(1),
|
|
||||||
field8: 0,
|
|
||||||
field9: 0,
|
|
||||||
forward: {
|
forward: {
|
||||||
field1: 0,
|
field1: 0,
|
||||||
field2: 0,
|
field2: 0,
|
||||||
field3: node.groupId ? 0 : 2,
|
field3: node.groupId ? 1 : 2,
|
||||||
unknownBase64: avatar,
|
unknownBase64: avatar,
|
||||||
avatar: avatar
|
avatar: avatar
|
||||||
}
|
}
|
||||||
|
@@ -10,6 +10,7 @@ import {
|
|||||||
MsgInfo,
|
MsgInfo,
|
||||||
NotOnlineImage,
|
NotOnlineImage,
|
||||||
OidbSvcTrpcTcp0XE37_800Response,
|
OidbSvcTrpcTcp0XE37_800Response,
|
||||||
|
PushMsgBody,
|
||||||
QBigFaceExtra,
|
QBigFaceExtra,
|
||||||
QSmallFaceExtra,
|
QSmallFaceExtra,
|
||||||
} from '@/core/packet/transformer/proto';
|
} from '@/core/packet/transformer/proto';
|
||||||
@@ -29,7 +30,8 @@ import {
|
|||||||
SendReplyElement,
|
SendReplyElement,
|
||||||
SendMultiForwardMsgElement,
|
SendMultiForwardMsgElement,
|
||||||
SendTextElement,
|
SendTextElement,
|
||||||
SendVideoElement
|
SendVideoElement,
|
||||||
|
Peer
|
||||||
} from '@/core';
|
} from '@/core';
|
||||||
import {ForwardMsgBuilder} from '@/common/forward-msg-builder';
|
import {ForwardMsgBuilder} from '@/common/forward-msg-builder';
|
||||||
import {PacketMsg, PacketSendMsgElement} from '@/core/packet/message/message';
|
import {PacketMsg, PacketSendMsgElement} from '@/core/packet/message/message';
|
||||||
@@ -146,41 +148,40 @@ export class PacketMsgAtElement extends PacketMsgTextElement {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export class PacketMsgReplyElement extends IPacketMsgElement<SendReplyElement> {
|
export class PacketMsgReplyElement extends IPacketMsgElement<SendReplyElement> {
|
||||||
messageId: bigint;
|
time: number;
|
||||||
messageSeq: number;
|
targetMessageId: bigint;
|
||||||
messageClientSeq: number;
|
targetMessageSeq: number;
|
||||||
|
targetMessageClientSeq: number;
|
||||||
targetUin: number;
|
targetUin: number;
|
||||||
targetUid: string;
|
targetUid: string;
|
||||||
time: number;
|
targetElems?: NapProtoEncodeStructType<typeof Elem>[];
|
||||||
elems: PacketMsg[];
|
targetSourceMsg?: NapProtoEncodeStructType<typeof PushMsgBody>;
|
||||||
|
targetPeer?: Peer;
|
||||||
|
|
||||||
constructor(element: SendReplyElement) {
|
constructor(element: SendReplyElement) {
|
||||||
super(element);
|
super(element);
|
||||||
this.messageId = BigInt(element.replyElement.replayMsgId ?? 0);
|
this.time = +(element.replyElement.replyMsgTime ?? Math.floor(Date.now() / 1000));
|
||||||
this.messageSeq = +(element.replyElement.replayMsgSeq ?? 0);
|
this.targetMessageId = BigInt(element.replyElement.replayMsgId ?? 0);
|
||||||
this.messageClientSeq = +(element.replyElement.replyMsgClientSeq ?? 0);
|
this.targetMessageSeq = +(element.replyElement.replayMsgSeq ?? 0);
|
||||||
|
this.targetMessageClientSeq = +(element.replyElement.replyMsgClientSeq ?? 0);
|
||||||
this.targetUin = +(element.replyElement.senderUin ?? 0);
|
this.targetUin = +(element.replyElement.senderUin ?? 0);
|
||||||
this.targetUid = element.replyElement.senderUidStr ?? '';
|
this.targetUid = element.replyElement.senderUidStr ?? '';
|
||||||
this.time = +(element.replyElement.replyMsgTime ?? 0);
|
this.targetPeer = element.replyElement._replyMsgPeer;
|
||||||
this.elems = []; // TODO: in replyElement.sourceMsgTextElems
|
|
||||||
}
|
}
|
||||||
|
|
||||||
get isGroupReply(): boolean {
|
get isGroupReply(): boolean {
|
||||||
return this.messageClientSeq === 0;
|
return this.targetMessageClientSeq === 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
override buildElement(): NapProtoEncodeStructType<typeof Elem>[] {
|
override buildElement(): NapProtoEncodeStructType<typeof Elem>[] {
|
||||||
return [{
|
return [{
|
||||||
srcMsg: {
|
srcMsg: {
|
||||||
origSeqs: [this.isGroupReply ? this.messageClientSeq : this.messageSeq],
|
origSeqs: [this.isGroupReply ? this.targetMessageSeq : this.targetMessageClientSeq],
|
||||||
senderUin: BigInt(this.targetUin),
|
senderUin: BigInt(this.targetUin),
|
||||||
time: this.time,
|
time: this.time,
|
||||||
elems: [], // TODO: in replyElement.sourceMsgTextElems
|
elems: this.targetElems ?? [],
|
||||||
pbReserve: {
|
sourceMsg: new NapProtoMsg(PushMsgBody).encode(this.targetSourceMsg ?? {}),
|
||||||
messageId: this.messageId,
|
toUin: BigInt(0),
|
||||||
},
|
|
||||||
toUin: BigInt(this.targetUin),
|
|
||||||
type: 1,
|
|
||||||
}
|
}
|
||||||
}];
|
}];
|
||||||
}
|
}
|
||||||
|
50
src/core/packet/transformer/highway/DownloadGroupVideo.ts
Normal file
50
src/core/packet/transformer/highway/DownloadGroupVideo.ts
Normal file
@@ -0,0 +1,50 @@
|
|||||||
|
import * as proto from '@/core/packet/transformer/proto';
|
||||||
|
import { NapProtoEncodeStructType, NapProtoMsg } from '@napneko/nap-proto-core';
|
||||||
|
import { OidbPacket, PacketTransformer } from '@/core/packet/transformer/base';
|
||||||
|
import OidbBase from '@/core/packet/transformer/oidb/oidbBase';
|
||||||
|
import { IndexNode } from '@/core/packet/transformer/proto';
|
||||||
|
|
||||||
|
class DownloadGroupVideo extends PacketTransformer<typeof proto.NTV2RichMediaResp> {
|
||||||
|
constructor() {
|
||||||
|
super();
|
||||||
|
}
|
||||||
|
|
||||||
|
build(groupUin: number, node: NapProtoEncodeStructType<typeof IndexNode>): OidbPacket {
|
||||||
|
const body = new NapProtoMsg(proto.NTV2RichMediaReq).encode({
|
||||||
|
reqHead: {
|
||||||
|
common: {
|
||||||
|
requestId: 1,
|
||||||
|
command: 200
|
||||||
|
},
|
||||||
|
scene: {
|
||||||
|
requestType: 2,
|
||||||
|
businessType: 2,
|
||||||
|
sceneType: 2,
|
||||||
|
group: {
|
||||||
|
groupUin: groupUin
|
||||||
|
}
|
||||||
|
},
|
||||||
|
client: {
|
||||||
|
agentType: 2,
|
||||||
|
}
|
||||||
|
},
|
||||||
|
download: {
|
||||||
|
node: node,
|
||||||
|
download: {
|
||||||
|
video: {
|
||||||
|
busiType: 0,
|
||||||
|
sceneType: 0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return OidbBase.build(0x11EA, 200, body, true, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
parse(data: Buffer) {
|
||||||
|
const oidbBody = OidbBase.parse(data).body;
|
||||||
|
return new NapProtoMsg(proto.NTV2RichMediaResp).decode(oidbBody);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export default new DownloadGroupVideo();
|
51
src/core/packet/transformer/highway/DownloadPtt.ts
Normal file
51
src/core/packet/transformer/highway/DownloadPtt.ts
Normal file
@@ -0,0 +1,51 @@
|
|||||||
|
import * as proto from '@/core/packet/transformer/proto';
|
||||||
|
import { NapProtoEncodeStructType, NapProtoMsg } from '@napneko/nap-proto-core';
|
||||||
|
import { OidbPacket, PacketTransformer } from '@/core/packet/transformer/base';
|
||||||
|
import OidbBase from '@/core/packet/transformer/oidb/oidbBase';
|
||||||
|
import { IndexNode } from '@/core/packet/transformer/proto';
|
||||||
|
|
||||||
|
class DownloadPtt extends PacketTransformer<typeof proto.NTV2RichMediaResp> {
|
||||||
|
constructor() {
|
||||||
|
super();
|
||||||
|
}
|
||||||
|
|
||||||
|
build(selfUid: string, node: NapProtoEncodeStructType<typeof IndexNode>): OidbPacket {
|
||||||
|
const body = new NapProtoMsg(proto.NTV2RichMediaReq).encode({
|
||||||
|
reqHead: {
|
||||||
|
common: {
|
||||||
|
requestId: 1,
|
||||||
|
command: 200
|
||||||
|
},
|
||||||
|
scene: {
|
||||||
|
requestType: 1,
|
||||||
|
businessType: 3,
|
||||||
|
sceneType: 1,
|
||||||
|
c2C: {
|
||||||
|
accountType: 2,
|
||||||
|
targetUid: selfUid
|
||||||
|
},
|
||||||
|
},
|
||||||
|
client: {
|
||||||
|
agentType: 2,
|
||||||
|
}
|
||||||
|
},
|
||||||
|
download: {
|
||||||
|
node: node,
|
||||||
|
download: {
|
||||||
|
video: {
|
||||||
|
busiType: 0,
|
||||||
|
sceneType: 0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return OidbBase.build(0x126D, 200, body, true, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
parse(data: Buffer) {
|
||||||
|
const oidbBody = OidbBase.parse(data).body;
|
||||||
|
return new NapProtoMsg(proto.NTV2RichMediaResp).decode(oidbBody);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export default new DownloadPtt();
|
51
src/core/packet/transformer/highway/DownloadVideo.ts
Normal file
51
src/core/packet/transformer/highway/DownloadVideo.ts
Normal file
@@ -0,0 +1,51 @@
|
|||||||
|
import * as proto from '@/core/packet/transformer/proto';
|
||||||
|
import { NapProtoEncodeStructType, NapProtoMsg } from '@napneko/nap-proto-core';
|
||||||
|
import { OidbPacket, PacketTransformer } from '@/core/packet/transformer/base';
|
||||||
|
import OidbBase from '@/core/packet/transformer/oidb/oidbBase';
|
||||||
|
import { IndexNode } from '@/core/packet/transformer/proto';
|
||||||
|
|
||||||
|
class DownloadVideo extends PacketTransformer<typeof proto.NTV2RichMediaResp> {
|
||||||
|
constructor() {
|
||||||
|
super();
|
||||||
|
}
|
||||||
|
|
||||||
|
build(selfUid: string, node: NapProtoEncodeStructType<typeof IndexNode>): OidbPacket {
|
||||||
|
const body = new NapProtoMsg(proto.NTV2RichMediaReq).encode({
|
||||||
|
reqHead: {
|
||||||
|
common: {
|
||||||
|
requestId: 1,
|
||||||
|
command: 200
|
||||||
|
},
|
||||||
|
scene: {
|
||||||
|
requestType: 2,
|
||||||
|
businessType: 2,
|
||||||
|
sceneType: 1,
|
||||||
|
c2C: {
|
||||||
|
accountType: 2,
|
||||||
|
targetUid: selfUid
|
||||||
|
},
|
||||||
|
},
|
||||||
|
client: {
|
||||||
|
agentType: 2,
|
||||||
|
}
|
||||||
|
},
|
||||||
|
download: {
|
||||||
|
node: node,
|
||||||
|
download: {
|
||||||
|
video: {
|
||||||
|
busiType: 0,
|
||||||
|
sceneType: 0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return OidbBase.build(0x11E9, 200, body, true, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
parse(data: Buffer) {
|
||||||
|
const oidbBody = OidbBase.parse(data).body;
|
||||||
|
return new NapProtoMsg(proto.NTV2RichMediaResp).decode(oidbBody);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export default new DownloadVideo();
|
@@ -13,3 +13,6 @@ export { default as UploadPrivatePtt } from './UploadPrivatePtt';
|
|||||||
export { default as UploadPrivateVideo } from './UploadPrivateVideo';
|
export { default as UploadPrivateVideo } from './UploadPrivateVideo';
|
||||||
export { default as DownloadImage } from './DownloadImage';
|
export { default as DownloadImage } from './DownloadImage';
|
||||||
export { default as DownloadGroupImage } from './DownloadGroupImage';
|
export { default as DownloadGroupImage } from './DownloadGroupImage';
|
||||||
|
export { default as DownloadVideo } from './DownloadVideo';
|
||||||
|
export { default as DownloadGroupVideo } from './DownloadGroupVideo';
|
||||||
|
export { default as DownloadPtt } from './DownloadPtt';
|
27
src/core/packet/transformer/message/FetchC2CMessage.ts
Normal file
27
src/core/packet/transformer/message/FetchC2CMessage.ts
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
import * as proto from '@/core/packet/transformer/proto';
|
||||||
|
import { NapProtoMsg } from '@napneko/nap-proto-core';
|
||||||
|
import { OidbPacket, PacketHexStrBuilder, PacketTransformer } from '@/core/packet/transformer/base';
|
||||||
|
|
||||||
|
class FetchC2CMessage extends PacketTransformer<typeof proto.SsoGetC2cMsgResponse> {
|
||||||
|
constructor() {
|
||||||
|
super();
|
||||||
|
}
|
||||||
|
|
||||||
|
build(targetUid: string, startSeq: number, endSeq: number): OidbPacket {
|
||||||
|
const req = new NapProtoMsg(proto.SsoGetC2cMsg).encode({
|
||||||
|
friendUid: targetUid,
|
||||||
|
startSequence: startSeq,
|
||||||
|
endSequence: endSeq,
|
||||||
|
});
|
||||||
|
return {
|
||||||
|
cmd: 'trpc.msg.register_proxy.RegisterProxy.SsoGetC2cMsg',
|
||||||
|
data: PacketHexStrBuilder(req)
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
parse(data: Buffer) {
|
||||||
|
return new NapProtoMsg(proto.SsoGetC2cMsgResponse).decode(data);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export default new FetchC2CMessage();
|
30
src/core/packet/transformer/message/FetchGroupMessage.ts
Normal file
30
src/core/packet/transformer/message/FetchGroupMessage.ts
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
import * as proto from '@/core/packet/transformer/proto';
|
||||||
|
import { NapProtoMsg } from '@napneko/nap-proto-core';
|
||||||
|
import { OidbPacket, PacketHexStrBuilder, PacketTransformer } from '@/core/packet/transformer/base';
|
||||||
|
|
||||||
|
class FetchGroupMessage extends PacketTransformer<typeof proto.SsoGetGroupMsgResponse> {
|
||||||
|
constructor() {
|
||||||
|
super();
|
||||||
|
}
|
||||||
|
|
||||||
|
build(groupUin: number, startSeq: number, endSeq: number): OidbPacket {
|
||||||
|
const req = new NapProtoMsg(proto.SsoGetGroupMsg).encode({
|
||||||
|
info: {
|
||||||
|
groupUin: groupUin,
|
||||||
|
startSequence: startSeq,
|
||||||
|
endSequence: endSeq
|
||||||
|
},
|
||||||
|
direction: true
|
||||||
|
});
|
||||||
|
return {
|
||||||
|
cmd: 'trpc.msg.register_proxy.RegisterProxy.SsoGetGroupMsg',
|
||||||
|
data: PacketHexStrBuilder(req)
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
parse(data: Buffer) {
|
||||||
|
return new NapProtoMsg(proto.SsoGetGroupMsgResponse).decode(data);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export default new FetchGroupMessage();
|
@@ -1,2 +1,4 @@
|
|||||||
export { default as UploadForwardMsg } from './UploadForwardMsg';
|
export { default as UploadForwardMsg } from './UploadForwardMsg';
|
||||||
export { default as DownloadForwardMsg } from './DownloadForwardMsg';
|
export { default as FetchGroupMessage } from './FetchGroupMessage';
|
||||||
|
export { default as FetchC2CMessage } from './FetchC2CMessage';
|
||||||
|
export { default as DownloadForwardMsg } from './DownloadForwardMsg';
|
||||||
|
@@ -13,13 +13,15 @@ import {
|
|||||||
export const ContentHead = {
|
export const ContentHead = {
|
||||||
type: ProtoField(1, ScalarType.UINT32),
|
type: ProtoField(1, ScalarType.UINT32),
|
||||||
subType: ProtoField(2, ScalarType.UINT32, true),
|
subType: ProtoField(2, ScalarType.UINT32, true),
|
||||||
divSeq: ProtoField(3, ScalarType.UINT32, true),
|
c2cCmd: ProtoField(3, ScalarType.UINT32, true),
|
||||||
msgId: ProtoField(4, ScalarType.UINT32, true),
|
ranDom: ProtoField(4, ScalarType.UINT32, true),
|
||||||
sequence: ProtoField(5, ScalarType.UINT32, true),
|
sequence: ProtoField(5, ScalarType.UINT32, true),
|
||||||
timeStamp: ProtoField(6, ScalarType.UINT32, true),
|
timeStamp: ProtoField(6, ScalarType.UINT32, true),
|
||||||
field7: ProtoField(7, ScalarType.UINT64, true),
|
pkgNum: ProtoField(7, ScalarType.UINT64, true),
|
||||||
field8: ProtoField(8, ScalarType.UINT32, true),
|
pkgIndex: ProtoField(8, ScalarType.UINT32, true),
|
||||||
field9: ProtoField(9, ScalarType.UINT32, true),
|
divSeq: ProtoField(9, ScalarType.UINT32, true),
|
||||||
|
autoReply: ProtoField(10, ScalarType.UINT32),
|
||||||
|
ntMsgSeq: ProtoField(10, ScalarType.UINT32, true),
|
||||||
newId: ProtoField(12, ScalarType.UINT64, true),
|
newId: ProtoField(12, ScalarType.UINT64, true),
|
||||||
forward: ProtoField(15, () => ForwardHead, true),
|
forward: ProtoField(15, () => ForwardHead, true),
|
||||||
};
|
};
|
||||||
|
6
src/core/packet/transformer/proto/misc/fileid.ts
Normal file
6
src/core/packet/transformer/proto/misc/fileid.ts
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
import { ProtoField, ScalarType } from '@napneko/nap-proto-core';
|
||||||
|
|
||||||
|
export const FileId = {
|
||||||
|
appid: ProtoField(4, ScalarType.UINT32, true),
|
||||||
|
ttl: ProtoField(10, ScalarType.UINT32, true),
|
||||||
|
};
|
@@ -148,10 +148,11 @@ export interface NodeIKernelMsgService {
|
|||||||
msgList: RawMessage[]
|
msgList: RawMessage[]
|
||||||
}>;
|
}>;
|
||||||
|
|
||||||
//@deprecated
|
// getMsgService/getMsgs { chatType: 2, peerUid: '975206796', privilegeFlag: 336068800 } 0 20 true
|
||||||
getMsgs(peer: Peer, msgId: string, count: unknown, queryOrder: boolean): Promise<unknown>;
|
getMsgs(peer: Peer & { privilegeFlag: number }, msgId: string, count: number, queryOrder: boolean): Promise<GeneralCallResult & {
|
||||||
|
msgList: RawMessage[]
|
||||||
|
}>;
|
||||||
|
|
||||||
//@deprecated
|
|
||||||
getMsgsIncludeSelf(peer: Peer, msgId: string, count: number, queryOrder: boolean): Promise<GeneralCallResult & {
|
getMsgsIncludeSelf(peer: Peer, msgId: string, count: number, queryOrder: boolean): Promise<GeneralCallResult & {
|
||||||
msgList: RawMessage[]
|
msgList: RawMessage[]
|
||||||
}>;
|
}>;
|
||||||
|
@@ -1,4 +1,15 @@
|
|||||||
import { ElementType, MessageElement, NTGrayTipElementSubTypeV2, PicSubType, PicType, TipAioOpGrayTipElement, TipGroupElement, NTVideoType, FaceType } from './msg';
|
import {
|
||||||
|
ElementType,
|
||||||
|
MessageElement,
|
||||||
|
NTGrayTipElementSubTypeV2,
|
||||||
|
PicSubType,
|
||||||
|
PicType,
|
||||||
|
TipAioOpGrayTipElement,
|
||||||
|
TipGroupElement,
|
||||||
|
NTVideoType,
|
||||||
|
FaceType,
|
||||||
|
Peer
|
||||||
|
} from './msg';
|
||||||
|
|
||||||
type ElementFullBase = Omit<MessageElement, 'elementType' | 'elementId' | 'extBufForUI'>;
|
type ElementFullBase = Omit<MessageElement, 'elementType' | 'elementId' | 'extBufForUI'>;
|
||||||
|
|
||||||
@@ -213,6 +224,9 @@ export interface ReplyElement {
|
|||||||
senderUidStr?: string;
|
senderUidStr?: string;
|
||||||
replyMsgTime?: string;
|
replyMsgTime?: string;
|
||||||
replyMsgClientSeq?: string;
|
replyMsgClientSeq?: string;
|
||||||
|
// HACK: Attributes that were not originally available,
|
||||||
|
// but were added due to NTQQ and NapCat's internal implementation, are used to supplement NapCat
|
||||||
|
_replyMsgPeer?: Peer;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface CalendarElement {
|
export interface CalendarElement {
|
||||||
|
@@ -403,7 +403,7 @@ export interface NTGroupGrayMember {
|
|||||||
}
|
}
|
||||||
/**
|
/**
|
||||||
* 群灰色提示邀请者和被邀请者接口
|
* 群灰色提示邀请者和被邀请者接口
|
||||||
*
|
*
|
||||||
* */
|
* */
|
||||||
export interface NTGroupGrayInviterAndInvite {
|
export interface NTGroupGrayInviterAndInvite {
|
||||||
invited: NTGroupGrayMember;
|
invited: NTGroupGrayMember;
|
||||||
@@ -501,13 +501,15 @@ export interface RawMessage {
|
|||||||
elements: MessageElement[];// 消息元素
|
elements: MessageElement[];// 消息元素
|
||||||
sourceType: MsgSourceType;// 消息来源类型
|
sourceType: MsgSourceType;// 消息来源类型
|
||||||
isOnlineMsg: boolean;// 是否为在线消息
|
isOnlineMsg: boolean;// 是否为在线消息
|
||||||
|
clientSeq?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 查询消息参数接口
|
* 查询消息参数接口
|
||||||
*/
|
*/
|
||||||
export interface QueryMsgsParams {
|
export interface QueryMsgsParams {
|
||||||
chatInfo: Peer;
|
chatInfo: Peer & { privilegeFlag?: number };
|
||||||
|
//searchFields: number;
|
||||||
filterMsgType: Array<{ type: NTMsgType, subType: Array<number> }>;
|
filterMsgType: Array<{ type: NTMsgType, subType: Array<number> }>;
|
||||||
filterSendersUid: string[];
|
filterSendersUid: string[];
|
||||||
filterMsgFromTime: string;
|
filterMsgFromTime: string;
|
||||||
@@ -565,4 +567,4 @@ export enum FaceType {
|
|||||||
AniSticke = 3, // 动画贴纸
|
AniSticke = 3, // 动画贴纸
|
||||||
Lottie = 4,// 新格式表情
|
Lottie = 4,// 新格式表情
|
||||||
Poke = 5 // 可变Poke
|
Poke = 5 // 可变Poke
|
||||||
}
|
}
|
||||||
|
@@ -100,7 +100,7 @@ export class OneBotMsgApi {
|
|||||||
let qq: string = 'all';
|
let qq: string = 'all';
|
||||||
if (element.atType !== NTMsgAtType.ATTYPEALL) {
|
if (element.atType !== NTMsgAtType.ATTYPEALL) {
|
||||||
const { atNtUid, atUid } = element;
|
const { atNtUid, atUid } = element;
|
||||||
qq = !atUid || atUid === '0' ? await this.core.apis.UserApi.getUinByUidV2(atNtUid) : atUid;
|
qq = !atUid || atUid === '0' ? await this.core.apis.UserApi.getUinByUidV2(atNtUid) : String(Number(atUid) >>> 0);
|
||||||
}
|
}
|
||||||
return {
|
return {
|
||||||
type: OB11MessageDataType.at,
|
type: OB11MessageDataType.at,
|
||||||
@@ -150,12 +150,31 @@ export class OneBotMsgApi {
|
|||||||
};
|
};
|
||||||
FileNapCatOneBotUUID.encode(peer, msg.msgId, elementWrapper.elementId, element.fileUuid, element.fileUuid);
|
FileNapCatOneBotUUID.encode(peer, msg.msgId, elementWrapper.elementId, element.fileUuid, element.fileUuid);
|
||||||
FileNapCatOneBotUUID.encode(peer, msg.msgId, elementWrapper.elementId, element.fileUuid, element.fileName);
|
FileNapCatOneBotUUID.encode(peer, msg.msgId, elementWrapper.elementId, element.fileUuid, element.fileName);
|
||||||
|
if (this.core.apis.PacketApi.available) {
|
||||||
|
let url;
|
||||||
|
try {
|
||||||
|
url = await this.core.apis.FileApi.getFileUrl(msg.chatType, msg.peerUid, element.fileUuid, element.file10MMd5)
|
||||||
|
} catch (error) {
|
||||||
|
url = '';
|
||||||
|
}
|
||||||
|
if (url) {
|
||||||
|
return {
|
||||||
|
type: OB11MessageDataType.file,
|
||||||
|
data: {
|
||||||
|
file: element.fileName,
|
||||||
|
file_id: element.fileUuid,
|
||||||
|
file_size: element.fileSize,
|
||||||
|
url: url,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
return {
|
return {
|
||||||
type: OB11MessageDataType.file,
|
type: OB11MessageDataType.file,
|
||||||
data: {
|
data: {
|
||||||
file: element.fileName,
|
file: element.fileName,
|
||||||
file_id: element.fileUuid,
|
file_id: element.fileUuid,
|
||||||
file_size: element.fileSize,
|
file_size: element.fileSize
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
@@ -225,17 +244,13 @@ export class OneBotMsgApi {
|
|||||||
},
|
},
|
||||||
|
|
||||||
replyElement: async (element, msg) => {
|
replyElement: async (element, msg) => {
|
||||||
const records = msg.records.find(msgRecord => msgRecord.msgId === element?.sourceMsgIdInRecords);
|
|
||||||
const peer = {
|
const peer = {
|
||||||
chatType: msg.chatType,
|
chatType: msg.chatType,
|
||||||
peerUid: msg.peerUid,
|
peerUid: msg.peerUid,
|
||||||
guildId: '',
|
guildId: '',
|
||||||
};
|
};
|
||||||
if (!records || !element.replyMsgTime || !element.senderUidStr) {
|
|
||||||
this.core.context.logger.logError('似乎是旧版客户端,获取不到引用的消息', element.replayMsgSeq);
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
// 创建回复数据的通用方法
|
||||||
const createReplyData = (msgId: string): OB11MessageData => ({
|
const createReplyData = (msgId: string): OB11MessageData => ({
|
||||||
type: OB11MessageDataType.reply,
|
type: OB11MessageDataType.reply,
|
||||||
data: {
|
data: {
|
||||||
@@ -243,48 +258,96 @@ export class OneBotMsgApi {
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
if (records.peerUin === '284840486' || records.peerUin === '1094950020') {
|
// 查找记录
|
||||||
|
const records = msg.records.find(msgRecord => msgRecord.msgId === element?.sourceMsgIdInRecords);
|
||||||
|
|
||||||
|
// 特定账号的特殊处理
|
||||||
|
if (records && (records.peerUin === '284840486' || records.peerUin === '1094950020')) {
|
||||||
return createReplyData(records.msgId);
|
return createReplyData(records.msgId);
|
||||||
}
|
}
|
||||||
let replyMsgList = (await this.core.apis.MsgApi.queryMsgsWithFilterExWithSeqV2(peer, element.replayMsgSeq, records.msgTime, [element.senderUidStr])).msgList;
|
|
||||||
let replyMsg = replyMsgList.find(msg => msg.msgRandom === records.msgRandom);
|
|
||||||
|
|
||||||
if (!replyMsg || records.msgRandom !== replyMsg.msgRandom) {
|
// 获取消息的通用方法组
|
||||||
this.core.context.logger.logError(
|
const tryFetchMethods = async (msgSeq: string, senderUid?: string, msgTime?: string, msgRandom?: string): Promise<RawMessage | undefined> => {
|
||||||
'筛选结果,筛选消息失败,将使用Fallback-1 Seq: ',
|
try {
|
||||||
|
// 方法1:通过序号和时间筛选
|
||||||
|
if (senderUid && msgTime) {
|
||||||
|
const replyMsgList = (await this.core.apis.MsgApi.queryMsgsWithFilterExWithSeqV2(
|
||||||
|
peer, msgSeq, msgTime, [senderUid]
|
||||||
|
)).msgList;
|
||||||
|
|
||||||
|
const replyMsg = msgRandom
|
||||||
|
? replyMsgList.find(msg => msg.msgRandom === msgRandom)
|
||||||
|
: replyMsgList.find(msg => msg.msgSeq === msgSeq);
|
||||||
|
|
||||||
|
if (replyMsg) return replyMsg;
|
||||||
|
|
||||||
|
this.core.context.logger.logWarn(`方法1查询失败,序号: ${msgSeq}, 消息数: ${replyMsgList.length}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 方法2:直接通过序号获取
|
||||||
|
const replyMsgList = (await this.core.apis.MsgApi.getMsgsBySeqAndCount(
|
||||||
|
peer, msgSeq, 1, true, true
|
||||||
|
)).msgList;
|
||||||
|
|
||||||
|
const replyMsg = msgRandom
|
||||||
|
? replyMsgList.find(msg => msg.msgRandom === msgRandom)
|
||||||
|
: replyMsgList.find(msg => msg.msgSeq === msgSeq);
|
||||||
|
|
||||||
|
if (replyMsg) return replyMsg;
|
||||||
|
|
||||||
|
this.core.context.logger.logWarn(`方法2查询失败,序号: ${msgSeq}, 消息数: ${replyMsgList.length}`);
|
||||||
|
|
||||||
|
// 方法3:另一种筛选方式
|
||||||
|
if (senderUid) {
|
||||||
|
const replyMsgList = (await this.core.apis.MsgApi.queryMsgsWithFilterExWithSeqV3(
|
||||||
|
peer, msgSeq, [senderUid]
|
||||||
|
)).msgList;
|
||||||
|
|
||||||
|
const replyMsg = msgRandom
|
||||||
|
? replyMsgList.find(msg => msg.msgRandom === msgRandom)
|
||||||
|
: replyMsgList.find(msg => msg.msgSeq === msgSeq);
|
||||||
|
|
||||||
|
if (replyMsg) return replyMsg;
|
||||||
|
|
||||||
|
this.core.context.logger.logWarn(`方法3查询失败,序号: ${msgSeq}, 消息数: ${replyMsgList.length}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
return undefined;
|
||||||
|
} catch (error) {
|
||||||
|
this.core.context.logger.logError('查询回复消息出错', error);
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// 有记录情况下,使用完整信息查询
|
||||||
|
if (records && element.replyMsgTime && element.senderUidStr) {
|
||||||
|
const replyMsg = await tryFetchMethods(
|
||||||
element.replayMsgSeq,
|
element.replayMsgSeq,
|
||||||
',消息长度:',
|
element.senderUidStr,
|
||||||
replyMsgList.length
|
records.msgTime,
|
||||||
|
records.msgRandom
|
||||||
);
|
);
|
||||||
replyMsgList = (await this.core.apis.MsgApi.getMsgsBySeqAndCount(peer, element.replayMsgSeq, 1, true, true)).msgList;
|
|
||||||
replyMsg = replyMsgList.find(msg => msg.msgRandom === records.msgRandom);
|
if (replyMsg) {
|
||||||
|
return createReplyData(replyMsg.msgId);
|
||||||
|
}
|
||||||
|
|
||||||
|
this.core.context.logger.logError('所有查找方法均失败,获取不到带记录的引用消息', element.replayMsgSeq);
|
||||||
|
} else {
|
||||||
|
// 旧版客户端或不完整记录的情况,也尝试使用相同流程
|
||||||
|
this.core.context.logger.logWarn('似乎是旧版客户端,尝试仅通过序号获取引用消息', element.replayMsgSeq);
|
||||||
|
|
||||||
|
const replyMsg = await tryFetchMethods(element.replayMsgSeq);
|
||||||
|
|
||||||
|
if (replyMsg) {
|
||||||
|
return createReplyData(replyMsg.msgId);
|
||||||
|
}
|
||||||
|
|
||||||
|
this.core.context.logger.logError('所有查找方法均失败,获取不到旧客户端的引用消息', element.replayMsgSeq);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!replyMsg || records.msgRandom !== replyMsg.msgRandom) {
|
return null;
|
||||||
this.core.context.logger.logWarn(
|
|
||||||
'筛选消息失败,将使用Fallback-2 Seq:',
|
|
||||||
element.replayMsgSeq,
|
|
||||||
',消息长度:',
|
|
||||||
replyMsgList.length
|
|
||||||
);
|
|
||||||
replyMsgList = (await this.core.apis.MsgApi.queryMsgsWithFilterExWithSeqV3(peer, element.replayMsgSeq, [element.senderUidStr])).msgList;
|
|
||||||
replyMsg = replyMsgList.find(msg => msg.msgRandom === records.msgRandom);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
// 丢弃该消息段
|
|
||||||
if (!replyMsg || records.msgRandom !== replyMsg.msgRandom) {
|
|
||||||
this.core.context.logger.logError(
|
|
||||||
'最终筛选结果,筛选消息失败,获取不到引用的消息 Seq: ',
|
|
||||||
element.replayMsgSeq,
|
|
||||||
',消息长度:',
|
|
||||||
replyMsgList.length
|
|
||||||
);
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
return createReplyData(replyMsg.msgId);
|
|
||||||
},
|
},
|
||||||
|
|
||||||
videoElement: async (element, msg, elementWrapper) => {
|
videoElement: async (element, msg, elementWrapper) => {
|
||||||
const peer = {
|
const peer = {
|
||||||
chatType: msg.chatType,
|
chatType: msg.chatType,
|
||||||
@@ -331,7 +394,17 @@ export class OneBotMsgApi {
|
|||||||
|
|
||||||
//开始兜底
|
//开始兜底
|
||||||
if (!videoDownUrl) {
|
if (!videoDownUrl) {
|
||||||
videoDownUrl = element.filePath;
|
if (this.core.apis.PacketApi.available) {
|
||||||
|
try {
|
||||||
|
videoDownUrl = await this.core.apis.FileApi.getVideoUrlPacket(msg.peerUid, element.fileUuid);
|
||||||
|
} catch (e) {
|
||||||
|
this.core.context.logger.logError('获取视频url失败', (e as Error).stack);
|
||||||
|
videoDownUrl = element.filePath;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
videoDownUrl = element.filePath;
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
const fileCode = FileNapCatOneBotUUID.encode(peer, msg.msgId, elementWrapper.elementId, element.fileUuid, element.fileName);
|
const fileCode = FileNapCatOneBotUUID.encode(peer, msg.msgId, elementWrapper.elementId, element.fileUuid, element.fileName);
|
||||||
return {
|
return {
|
||||||
@@ -351,6 +424,28 @@ export class OneBotMsgApi {
|
|||||||
guildId: '',
|
guildId: '',
|
||||||
};
|
};
|
||||||
const fileCode = FileNapCatOneBotUUID.encode(peer, msg.msgId, elementWrapper.elementId, '', element.fileName);
|
const fileCode = FileNapCatOneBotUUID.encode(peer, msg.msgId, elementWrapper.elementId, '', element.fileName);
|
||||||
|
let pttUrl = '';
|
||||||
|
if (this.core.apis.PacketApi.available) {
|
||||||
|
try {
|
||||||
|
pttUrl = await this.core.apis.FileApi.getPttUrl(msg.peerUid, element.fileUuid);
|
||||||
|
} catch (e) {
|
||||||
|
this.core.context.logger.logError('获取语音url失败', (e as Error).stack);
|
||||||
|
pttUrl = element.filePath;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
pttUrl = element.filePath;
|
||||||
|
}
|
||||||
|
if (pttUrl) {
|
||||||
|
return {
|
||||||
|
type: OB11MessageDataType.voice,
|
||||||
|
data: {
|
||||||
|
file: fileCode,
|
||||||
|
path: element.filePath,
|
||||||
|
url: pttUrl,
|
||||||
|
file_size: element.fileSize,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
return {
|
return {
|
||||||
type: OB11MessageDataType.voice,
|
type: OB11MessageDataType.voice,
|
||||||
data: {
|
data: {
|
||||||
@@ -467,6 +562,8 @@ export class OneBotMsgApi {
|
|||||||
replayMsgId: replyMsg.msgId, // raw.msgId
|
replayMsgId: replyMsg.msgId, // raw.msgId
|
||||||
senderUin: replyMsg.senderUin,
|
senderUin: replyMsg.senderUin,
|
||||||
senderUinStr: replyMsg.senderUin,
|
senderUinStr: replyMsg.senderUin,
|
||||||
|
replyMsgClientSeq: replyMsg.clientSeq,
|
||||||
|
_replyMsgPeer: replyMsgM.Peer
|
||||||
},
|
},
|
||||||
} :
|
} :
|
||||||
undefined;
|
undefined;
|
||||||
|
@@ -39,8 +39,11 @@ export class OB11WebSocketServerAdapter extends IOB11NetworkAdapter<WebsocketSer
|
|||||||
wsClient.close();
|
wsClient.close();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
//鉴权
|
// 鉴权 close 不会立刻销毁 当前返回可避免挂载message事件 close 并未立刻关闭 而是存在timer操作后关闭
|
||||||
this.authorize(this.config.token, wsClient, wsReq);
|
// 引发高危漏洞
|
||||||
|
if (!this.authorize(this.config.token, wsClient, wsReq)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
const paramUrl = wsReq.url?.indexOf('?') !== -1 ? wsReq.url?.substring(0, wsReq.url?.indexOf('?')) : wsReq.url;
|
const paramUrl = wsReq.url?.indexOf('?') !== -1 ? wsReq.url?.substring(0, wsReq.url?.indexOf('?')) : wsReq.url;
|
||||||
const isApiConnect = paramUrl === '/api' || paramUrl === '/api/';
|
const isApiConnect = paramUrl === '/api' || paramUrl === '/api/';
|
||||||
if (!isApiConnect) {
|
if (!isApiConnect) {
|
||||||
@@ -145,15 +148,16 @@ export class OB11WebSocketServerAdapter extends IOB11NetworkAdapter<WebsocketSer
|
|||||||
}
|
}
|
||||||
|
|
||||||
private authorize(token: string | undefined, wsClient: WebSocket, wsReq: IncomingMessage) {
|
private authorize(token: string | undefined, wsClient: WebSocket, wsReq: IncomingMessage) {
|
||||||
if (!token || token.length == 0) return;//客户端未设置密钥
|
if (!token || token.length == 0) return true;//客户端未设置密钥
|
||||||
const QueryClientToken = urlParse.parse(wsReq?.url || '', true).query['access_token'];
|
const QueryClientToken = urlParse.parse(wsReq?.url || '', true).query['access_token'];
|
||||||
const HeaderClientToken = wsReq.headers.authorization?.split('Bearer ').pop() || '';
|
const HeaderClientToken = wsReq.headers.authorization?.split('Bearer ').pop() || '';
|
||||||
const ClientToken = typeof (QueryClientToken) === 'string' && QueryClientToken !== '' ? QueryClientToken : HeaderClientToken;
|
const ClientToken = typeof (QueryClientToken) === 'string' && QueryClientToken !== '' ? QueryClientToken : HeaderClientToken;
|
||||||
if (ClientToken === token) {
|
if (ClientToken === token) {
|
||||||
return;
|
return true;
|
||||||
}
|
}
|
||||||
wsClient.send(JSON.stringify(OB11Response.res(null, 'failed', 1403, 'token验证失败')));
|
wsClient.send(JSON.stringify(OB11Response.res(null, 'failed', 1403, 'token验证失败')));
|
||||||
wsClient.close();
|
wsClient.close();
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
private checkStateAndReply<T>(data: T, wsClient: WebSocket) {
|
private checkStateAndReply<T>(data: T, wsClient: WebSocket) {
|
||||||
|
@@ -1,6 +1,7 @@
|
|||||||
import { LogWrapper } from '@/common/log';
|
import { LogWrapper } from '@/common/log';
|
||||||
import * as net from 'net';
|
import * as net from 'net';
|
||||||
import * as process from 'process';
|
import * as process from 'process';
|
||||||
|
import { Writable } from 'stream';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 连接到命名管道并重定向stdout
|
* 连接到命名管道并重定向stdout
|
||||||
@@ -25,12 +26,50 @@ export function connectToNamedPipe(logger: LogWrapper, timeoutMs: number = 5000)
|
|||||||
}, timeoutMs);
|
}, timeoutMs);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
let originalStdoutWrite = process.stdout.write.bind(process.stdout);
|
const originalStdoutWrite = process.stdout.write.bind(process.stdout);
|
||||||
const pipeSocket = net.connect(pipePath, () => {
|
const pipeSocket = net.connect(pipePath, () => {
|
||||||
// 清除超时
|
// 清除超时
|
||||||
clearTimeout(timeoutId);
|
clearTimeout(timeoutId);
|
||||||
|
|
||||||
|
// 优化网络性能设置
|
||||||
|
pipeSocket.setNoDelay(true); // 减少延迟
|
||||||
|
|
||||||
|
// 设置更高的高水位线,允许更多数据缓冲
|
||||||
|
|
||||||
logger.log(`[StdOut] 已重定向到命名管道: ${pipePath}`);
|
logger.log(`[StdOut] 已重定向到命名管道: ${pipePath}`);
|
||||||
|
|
||||||
|
// 创建拥有更优雅背压处理的 Writable 流
|
||||||
|
const pipeWritable = new Writable({
|
||||||
|
highWaterMark: 1024 * 64, // 64KB 高水位线
|
||||||
|
write(chunk, encoding, callback) {
|
||||||
|
if (!pipeSocket.writable) {
|
||||||
|
// 如果管道不可写,退回到原始stdout
|
||||||
|
logger.log('[StdOut] 管道不可写,回退到控制台输出');
|
||||||
|
return originalStdoutWrite(chunk, encoding, callback);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 尝试写入数据到管道
|
||||||
|
const canContinue = pipeSocket.write(chunk, encoding, () => {
|
||||||
|
// 数据已被发送或放入内部缓冲区
|
||||||
|
});
|
||||||
|
|
||||||
|
if (canContinue) {
|
||||||
|
// 如果返回true,表示可以继续写入更多数据
|
||||||
|
// 立即通知写入流可以继续
|
||||||
|
process.nextTick(callback);
|
||||||
|
} else {
|
||||||
|
// 如果返回false,表示内部缓冲区已满
|
||||||
|
// 等待drain事件再恢复写入
|
||||||
|
pipeSocket.once('drain', () => {
|
||||||
|
callback();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
// 明确返回true,表示写入已处理
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 重定向stdout
|
||||||
process.stdout.write = (
|
process.stdout.write = (
|
||||||
chunk: any,
|
chunk: any,
|
||||||
encoding?: BufferEncoding | (() => void),
|
encoding?: BufferEncoding | (() => void),
|
||||||
@@ -40,8 +79,11 @@ export function connectToNamedPipe(logger: LogWrapper, timeoutMs: number = 5000)
|
|||||||
cb = encoding;
|
cb = encoding;
|
||||||
encoding = undefined;
|
encoding = undefined;
|
||||||
}
|
}
|
||||||
return pipeSocket.write(chunk, encoding as BufferEncoding, cb);
|
|
||||||
|
// 使用优化的writable流处理写入
|
||||||
|
return pipeWritable.write(chunk, encoding as BufferEncoding, cb as () => void);
|
||||||
};
|
};
|
||||||
|
|
||||||
// 提供断开连接的方法
|
// 提供断开连接的方法
|
||||||
const disconnect = () => {
|
const disconnect = () => {
|
||||||
process.stdout.write = originalStdoutWrite;
|
process.stdout.write = originalStdoutWrite;
|
||||||
@@ -53,6 +95,7 @@ export function connectToNamedPipe(logger: LogWrapper, timeoutMs: number = 5000)
|
|||||||
resolve({ disconnect });
|
resolve({ disconnect });
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// 管道错误处理
|
||||||
pipeSocket.on('error', (err) => {
|
pipeSocket.on('error', (err) => {
|
||||||
clearTimeout(timeoutId);
|
clearTimeout(timeoutId);
|
||||||
process.stdout.write = originalStdoutWrite;
|
process.stdout.write = originalStdoutWrite;
|
||||||
@@ -60,11 +103,18 @@ export function connectToNamedPipe(logger: LogWrapper, timeoutMs: number = 5000)
|
|||||||
reject(err);
|
reject(err);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// 管道关闭处理
|
||||||
pipeSocket.on('end', () => {
|
pipeSocket.on('end', () => {
|
||||||
process.stdout.write = originalStdoutWrite;
|
process.stdout.write = originalStdoutWrite;
|
||||||
logger.log('命名管道连接已关闭');
|
logger.log('命名管道连接已关闭');
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// 确保在连接意外关闭时恢复stdout
|
||||||
|
pipeSocket.on('close', () => {
|
||||||
|
process.stdout.write = originalStdoutWrite;
|
||||||
|
logger.log('命名管道连接已关闭');
|
||||||
|
});
|
||||||
|
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
clearTimeout(timeoutId);
|
clearTimeout(timeoutId);
|
||||||
logger.log(`尝试连接命名管道 ${pipePath} 时发生异常:`, error);
|
logger.log(`尝试连接命名管道 ${pipePath} 时发生异常:`, error);
|
||||||
|
Reference in New Issue
Block a user