2025-06-16 23:41:35 +08:00
|
|
|
#include "ClientCore.h"
|
2025-06-14 23:25:16 +08:00
|
|
|
|
|
|
|
|
#include <QDebug>
|
|
|
|
|
|
|
|
|
|
ClientCore::ClientCore(QObject* parent) : QObject(parent)
|
|
|
|
|
{
|
2025-06-20 14:33:51 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ClientCore::Instance()
|
|
|
|
|
{
|
2025-06-22 01:17:44 +08:00
|
|
|
// qDebug() << "Instance() thread:" << QThread::currentThread();
|
2025-06-15 20:37:25 +08:00
|
|
|
socket_ = new QTcpSocket(this);
|
|
|
|
|
connect(socket_, &QTcpSocket::readyRead, this, &ClientCore::onReadyRead);
|
|
|
|
|
connect(socket_, &QTcpSocket::disconnected, this, &ClientCore::onDisconnected);
|
2025-06-14 23:25:16 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ClientCore::~ClientCore()
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-25 10:54:04 +08:00
|
|
|
bool ClientCore::SendFrame(QSharedPointer<FrameBuffer> frame)
|
|
|
|
|
{
|
|
|
|
|
return Send(frame);
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-20 14:33:51 +08:00
|
|
|
void ClientCore::DoConnect(const QString& ip, quint16 port)
|
|
|
|
|
{
|
2025-06-22 01:17:44 +08:00
|
|
|
// qDebug() << "doConnect thread:" << QThread::currentThread();
|
2025-06-20 14:33:51 +08:00
|
|
|
emit connecting();
|
|
|
|
|
if (!Connect(ip, port)) {
|
|
|
|
|
emit conFailed();
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
emit conSuccess();
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-14 23:25:16 +08:00
|
|
|
bool ClientCore::Connect(const QString& ip, quint16 port)
|
|
|
|
|
{
|
2025-06-26 22:59:58 +08:00
|
|
|
if (connected_) {
|
2025-11-04 15:18:28 +08:00
|
|
|
qInfo() << QString(tr("已连接。"));
|
2025-06-26 22:59:58 +08:00
|
|
|
return true;
|
|
|
|
|
}
|
2025-06-14 23:25:16 +08:00
|
|
|
socket_->connectToHost(ip, port);
|
|
|
|
|
if (!socket_->waitForConnected(3000)) {
|
2025-11-04 14:52:27 +08:00
|
|
|
qCritical() << QString(tr("%1:%2 连接失败。")).arg(ip).arg(port);
|
2025-06-14 23:25:16 +08:00
|
|
|
return false;
|
|
|
|
|
}
|
2025-11-04 14:52:27 +08:00
|
|
|
qInfo() << QString(tr("%1:%2 连接成功。")).arg(ip).arg(port);
|
2025-06-20 14:33:51 +08:00
|
|
|
connected_ = true;
|
2025-06-14 23:25:16 +08:00
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ClientCore::Disconnect()
|
|
|
|
|
{
|
2025-06-20 14:39:03 +08:00
|
|
|
QMutexLocker locker(&conMutex_);
|
|
|
|
|
if (socket_ && socket_->state() != QAbstractSocket::UnconnectedState) {
|
|
|
|
|
socket_->disconnectFromHost();
|
|
|
|
|
if (socket_->state() != QAbstractSocket::UnconnectedState) {
|
|
|
|
|
socket_->waitForDisconnected(1000);
|
|
|
|
|
}
|
|
|
|
|
}
|
2025-06-20 14:33:51 +08:00
|
|
|
connected_ = false;
|
2025-06-14 23:25:16 +08:00
|
|
|
}
|
2025-06-14 23:44:13 +08:00
|
|
|
|
|
|
|
|
void ClientCore::onReadyRead()
|
|
|
|
|
{
|
|
|
|
|
QByteArray data = socket_->readAll();
|
|
|
|
|
recvBuffer_.append(data);
|
|
|
|
|
while (true) {
|
|
|
|
|
auto frame = Protocol::ParseBuffer(recvBuffer_);
|
|
|
|
|
if (frame == nullptr) {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
UseFrame(frame);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ClientCore::onDisconnected()
|
|
|
|
|
{
|
2025-06-20 14:33:51 +08:00
|
|
|
connected_ = false;
|
2025-11-04 15:18:28 +08:00
|
|
|
qCritical() << QString("你 [%1] 断开了。").arg(ownID_);
|
2025-06-19 11:59:32 +08:00
|
|
|
emit sigDisconnect();
|
2025-06-14 23:44:13 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void ClientCore::UseFrame(QSharedPointer<FrameBuffer> frame)
|
|
|
|
|
{
|
2025-06-15 20:37:25 +08:00
|
|
|
switch (frame->type) {
|
|
|
|
|
case FrameBufferType::FBT_SER_MSG_ASKCLIENTS: {
|
|
|
|
|
InfoClientVec info = infoUnpack<InfoClientVec>(frame->data);
|
2025-06-19 15:37:39 +08:00
|
|
|
emit sigClients(info);
|
2025-06-15 20:37:25 +08:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case FrameBufferType::FBT_SER_MSG_YOURID: {
|
|
|
|
|
ownID_ = frame->data;
|
2025-06-22 01:17:44 +08:00
|
|
|
GlobalData::Ins()->SetLocalID(ownID_);
|
2025-11-04 14:52:27 +08:00
|
|
|
qInfo() << QString(tr("本机ID: %1")).arg(ownID_);
|
2025-10-21 09:48:20 +08:00
|
|
|
emit sigYourId(frame);
|
2025-06-15 20:37:25 +08:00
|
|
|
break;
|
|
|
|
|
}
|
2025-06-15 23:13:30 +08:00
|
|
|
case FrameBufferType::FBT_CLI_ANS_DIRFILE: {
|
|
|
|
|
DirFileInfoVec info = infoUnpack<DirFileInfoVec>(frame->data);
|
2025-06-19 15:37:39 +08:00
|
|
|
emit sigFiles(info);
|
2025-06-15 23:13:30 +08:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case FrameBufferType::FBT_CLI_ASK_DIRFILE: {
|
|
|
|
|
DirFileInfoVec vec;
|
|
|
|
|
InfoMsg info = infoUnpack<InfoMsg>(frame->data);
|
|
|
|
|
if (!localFile_.GetDirFile(info.msg, vec)) {
|
2025-11-04 14:52:27 +08:00
|
|
|
qWarning() << QString(tr("访问文件失败: %1")).arg(info.msg);
|
2025-06-15 23:13:30 +08:00
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (!Send<DirFileInfoVec>(vec, FBT_CLI_ANS_DIRFILE, frame->fid)) {
|
2025-11-04 14:52:27 +08:00
|
|
|
qCritical() << QString(tr("发送文件列表结果失败。"));
|
2025-06-15 23:13:30 +08:00
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case FrameBufferType::FBT_CLI_ASK_HOME: {
|
|
|
|
|
InfoMsg info;
|
|
|
|
|
info.msg = Util::GetUserHome();
|
|
|
|
|
if (!Send<InfoMsg>(info, FBT_CLI_ANS_HOME, frame->fid)) {
|
|
|
|
|
qCritical() << QString(tr("send home failed."));
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case FrameBufferType::FBT_CLI_ANS_HOME: {
|
|
|
|
|
InfoMsg info = infoUnpack<InfoMsg>(frame->data);
|
2025-11-04 15:18:28 +08:00
|
|
|
qInfo() << QString(tr("用户目录:%1")).arg(info.msg);
|
2025-06-19 15:37:39 +08:00
|
|
|
emit sigPath(info.msg);
|
2025-06-15 23:13:30 +08:00
|
|
|
break;
|
|
|
|
|
}
|
2025-06-18 15:10:28 +08:00
|
|
|
case FrameBufferType::FBT_SER_MSG_FORWARD_FAILED: {
|
2025-11-04 15:18:28 +08:00
|
|
|
qCritical() << QString(tr("转发数据失败,fid:%1, tid:%2, type:%3"))
|
2025-06-20 10:33:03 +08:00
|
|
|
.arg(frame->fid)
|
|
|
|
|
.arg(frame->tid)
|
|
|
|
|
.arg(static_cast<uint32_t>(frame->type));
|
2025-06-18 15:10:28 +08:00
|
|
|
break;
|
|
|
|
|
}
|
2025-06-19 15:37:39 +08:00
|
|
|
case FrameBufferType::FBT_CLI_REQ_SEND: {
|
|
|
|
|
emit sigReqSend(frame);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case FrameBufferType::FBT_CLI_REQ_DOWN: {
|
|
|
|
|
emit sigReqDown(frame);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case FrameBufferType::FBT_CLI_TRANS_DONE: {
|
2025-06-19 22:07:12 +08:00
|
|
|
emit sigTransDone(frame);
|
2025-06-19 15:37:39 +08:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case FrameBufferType::FBT_CLI_CAN_SEND: {
|
2025-06-19 22:07:12 +08:00
|
|
|
emit sigCanSend(frame);
|
2025-06-19 15:37:39 +08:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case FrameBufferType::FBT_CLI_CANOT_SEND: {
|
2025-06-19 22:07:12 +08:00
|
|
|
emit sigCanotSend(frame);
|
2025-06-19 15:37:39 +08:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case FBT_CLI_CANOT_DOWN: {
|
2025-06-19 22:07:12 +08:00
|
|
|
emit sigCanotDown(frame);
|
2025-06-19 15:37:39 +08:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case FBT_CLI_CAN_DOWN: {
|
2025-06-19 22:07:12 +08:00
|
|
|
emit sigCanDown(frame);
|
2025-06-19 15:37:39 +08:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case FBT_CLI_FILE_BUFFER: {
|
2025-06-19 22:07:12 +08:00
|
|
|
emit sigFileBuffer(frame);
|
2025-06-19 15:37:39 +08:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case FBT_CLI_TRANS_FAILED: {
|
2025-06-19 22:07:12 +08:00
|
|
|
emit sigTransFailed(frame);
|
2025-06-19 15:37:39 +08:00
|
|
|
break;
|
|
|
|
|
}
|
2025-06-20 09:45:39 +08:00
|
|
|
case FBT_CLI_FILE_INFO: {
|
|
|
|
|
emit sigFileInfo(frame);
|
|
|
|
|
break;
|
|
|
|
|
}
|
2025-10-20 16:40:52 +08:00
|
|
|
case FBT_SER_MSG_OFFLINE: {
|
|
|
|
|
emit sigOffline(frame);
|
|
|
|
|
break;
|
|
|
|
|
}
|
2025-11-05 10:57:40 +08:00
|
|
|
case FBT_MSGINFO_ASK: {
|
|
|
|
|
emit sigMsgAsk(frame);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case FBT_MSGINFO_ANSWER: {
|
|
|
|
|
emit sigMsgAnswer(frame);
|
|
|
|
|
break;
|
|
|
|
|
}
|
2025-06-15 20:37:25 +08:00
|
|
|
default:
|
2025-11-04 14:52:27 +08:00
|
|
|
qCritical() << QString("未知的帧类型: %1").arg(frame->type);
|
2025-06-15 20:37:25 +08:00
|
|
|
break;
|
|
|
|
|
}
|
2025-06-14 23:44:13 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool ClientCore::Send(QSharedPointer<FrameBuffer> frame)
|
|
|
|
|
{
|
|
|
|
|
if (frame == nullptr) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
auto data = Protocol::PackBuffer(frame);
|
|
|
|
|
if (data.size() == 0) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
return Send(data.constData(), data.size());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool ClientCore::Send(const char* data, qint64 len)
|
|
|
|
|
{
|
|
|
|
|
if (socket_->state() != QAbstractSocket::ConnectedState) {
|
2025-11-04 14:52:27 +08:00
|
|
|
qCritical() << QString("客户端 %1 未连接...").arg(remoteID_);
|
2025-06-14 23:44:13 +08:00
|
|
|
return false;
|
|
|
|
|
}
|
2025-06-17 11:48:13 +08:00
|
|
|
|
|
|
|
|
qint64 bytesWritten = -1;
|
|
|
|
|
{
|
|
|
|
|
QMutexLocker locker(&sockMut_);
|
|
|
|
|
bytesWritten = socket_->write(data, len);
|
|
|
|
|
}
|
2025-06-14 23:44:13 +08:00
|
|
|
if (bytesWritten == -1 || !socket_->waitForBytesWritten(5000)) {
|
2025-11-04 14:52:27 +08:00
|
|
|
qCritical() << QString("向服务器发送数据失败: %1").arg(socket_->errorString());
|
2025-06-14 23:44:13 +08:00
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
}
|
2025-06-15 14:31:54 +08:00
|
|
|
|
2025-06-25 17:06:30 +08:00
|
|
|
bool ClientCore::IsConnect()
|
|
|
|
|
{
|
|
|
|
|
return connected_;
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-15 14:31:54 +08:00
|
|
|
void ClientCore::SetRemoteID(const QString& id)
|
|
|
|
|
{
|
2025-06-22 01:17:44 +08:00
|
|
|
GlobalData::Ins()->SetRemoteID(id);
|
2025-06-15 23:13:30 +08:00
|
|
|
remoteID_ = id;
|
2025-06-15 14:31:54 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString ClientCore::GetRemoteID()
|
|
|
|
|
{
|
|
|
|
|
return remoteID_;
|
|
|
|
|
}
|
2025-06-18 14:53:56 +08:00
|
|
|
|
|
|
|
|
QString ClientCore::GetOwnID()
|
|
|
|
|
{
|
|
|
|
|
return ownID_;
|
|
|
|
|
}
|
2025-06-19 11:59:32 +08:00
|
|
|
|
|
|
|
|
SocketWorker::SocketWorker(ClientCore* core, QObject* parent) : QThread(parent), core_(core)
|
|
|
|
|
{
|
2025-06-22 01:17:44 +08:00
|
|
|
// connect(core_, &ClientCore::sigDisconnect, this, [this]() {
|
|
|
|
|
// thread()->quit();
|
|
|
|
|
// });
|
2025-06-19 11:59:32 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
SocketWorker::~SocketWorker()
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void SocketWorker::run()
|
|
|
|
|
{
|
2025-06-20 14:33:51 +08:00
|
|
|
core_->Instance();
|
2025-06-19 11:59:32 +08:00
|
|
|
exec();
|
|
|
|
|
}
|
2025-06-25 17:06:30 +08:00
|
|
|
|
|
|
|
|
HeatBeat::HeatBeat(ClientCore* core, QObject* parent) : QThread(parent), core_(core)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
HeatBeat::~HeatBeat()
|
|
|
|
|
{
|
|
|
|
|
Stop();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void HeatBeat::Stop()
|
|
|
|
|
{
|
|
|
|
|
isRun_ = false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void HeatBeat::run()
|
|
|
|
|
{
|
|
|
|
|
isRun_ = true;
|
|
|
|
|
InfoMsg info;
|
|
|
|
|
auto frame = core_->GetBuffer(info, FBT_SER_MSG_HEARTBEAT, "");
|
|
|
|
|
while (isRun_) {
|
|
|
|
|
QThread::sleep(1);
|
|
|
|
|
if (!core_->IsConnect()) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
ClientCore::syncInvoke(core_, frame);
|
2025-10-20 16:40:52 +08:00
|
|
|
auto rid = core_->GetRemoteID();
|
|
|
|
|
if (rid.isEmpty()) {
|
2025-06-25 17:06:30 +08:00
|
|
|
continue;
|
|
|
|
|
}
|
2025-10-20 16:40:52 +08:00
|
|
|
auto frame2 = core_->GetBuffer(info, FBT_SER_MSG_JUDGE_OTHER_ALIVE, rid);
|
|
|
|
|
ClientCore::syncInvoke(core_, frame2);
|
2025-06-25 17:06:30 +08:00
|
|
|
}
|
2025-10-20 16:40:52 +08:00
|
|
|
}
|