From 63896c5090aff6a4f7be668ada3415bdc5f60665 Mon Sep 17 00:00:00 2001 From: 0xFEEDC0DE64 Date: Thu, 16 Jul 2020 22:37:19 +0100 Subject: [PATCH] Implemented more packets for keepAlive, chat and disconnect --- client.cpp | 79 ++++++++++++++++++++++++++++++++++++++++-------- client.h | 12 ++++++++ main.cpp | 34 +++++++++++++++++++-- mcdatastream.cpp | 14 +++++++++ mcdatastream.h | 4 +++ packets.cpp | 69 +++++++++++++++++++++++++++++++++++------- packets.h | 78 +++++++++++++++++++++++++++++++++++++++++------ 7 files changed, 255 insertions(+), 35 deletions(-) diff --git a/client.cpp b/client.cpp index 5bf2e7a..66e0481 100644 --- a/client.cpp +++ b/client.cpp @@ -7,7 +7,7 @@ Client::Client(QTcpSocket *socket, QObject *parent) : QObject(parent), m_socket(socket), m_dataStream(m_socket), - m_packetSize(0), m_state(HandshakingState) + m_packetSize(0), m_state(HandshakingState), m_connectedSince(QDateTime::currentDateTime()) { m_socket->setParent(this); @@ -17,6 +17,50 @@ Client::Client(QTcpSocket *socket, QObject *parent) : qDebug() << m_socket->peerPort(); } +void Client::keepAlive() +{ + const auto now = QDateTime::currentDateTime(); + if (!m_lastKeepAliveSent.isValid() || m_lastKeepAliveSent.secsTo(now) >= 1) + { + packets::play::clientbound::KeepAlive packet; + packet.keepAliveId = 0; + packet.serialize(m_dataStream); + m_lastKeepAliveSent = now; + } +} + +void Client::sendChatMessage() +{ + const auto now = QDateTime::currentDateTime(); + if (!m_lastChatMessage.isValid() || m_lastChatMessage.secsTo(now) >= 2) + { + packets::play::clientbound::ChatMessage packet; + packet.jsonData = "{" + "\"text\": \"Chat message\", " + "\"bold\": \"true\" " + "}"; + packet.position = packets::play::clientbound::ChatMessage::Chat; + packet.serialize(m_dataStream); + m_lastChatMessage = now; + } +} + +void Client::trialDisconnect() +{ + const auto now = QDateTime::currentDateTime(); + if (m_connectedSince.secsTo(now) >= 20) + { + packets::play::clientbound::Disconnect packet; + packet.reason = "{" + "\"text\": \"Your trial has ended.\", " + "\"bold\": \"true\" " + "}"; + packet.serialize(m_dataStream); + m_socket->flush(); + deleteLater(); + } +} + void Client::readyRead() { while(m_socket->bytesAvailable()) @@ -75,7 +119,7 @@ void Client::readPacketHandshaking(packets::handshaking::serverbound::PacketType switch(type) { using namespace packets::handshaking; - case serverbound::PacketHandshake: + case serverbound::PacketType::Handshake: { serverbound::Handshake packet(dataStream); m_state = packet.nextState; @@ -95,7 +139,7 @@ void Client::readPacketStatus(const packets::status::serverbound::PacketType typ switch(type) { using namespace packets::status; - case serverbound::PacketRequest: + case serverbound::PacketType::Request: { { serverbound::Request packet(dataStream); @@ -113,21 +157,21 @@ void Client::readPacketStatus(const packets::status::serverbound::PacketType typ " \"online\": 2000," " \"sample\": [" " {" - " \"name\": \"0xFEEDC0DE64\"," + " \"name\": \"feedc0de\"," " \"id\": \"6ebf7396-b6da-40b1-b7d9-9b7961450d5a\"" " }" " ]" " }, " " \"description\": {" - " \"text\": \"Mein monster server in C++\"" + " \"text\": \"Minecraft server implemented in C++\"" " }," - " \"favicon\": \"\"" + " \"favicon\": \"\"" "}"; packet.serialize(m_dataStream); } break; } - case serverbound::PacketPing: + case serverbound::PacketType::Ping: { qint64 payload; { @@ -155,7 +199,7 @@ void Client::readPacketLogin(const packets::login::serverbound::PacketType type, switch(type) { using namespace packets::login; - case serverbound::PacketLogin: + case serverbound::PacketType::Login: { QString name; { @@ -174,8 +218,8 @@ void Client::readPacketLogin(const packets::login::serverbound::PacketType type, { packets::play::clientbound::JoinGame packet; packet.entityid = 1; - packet.gamemode = 0; - packet.dimension = 0; + packet.gamemode = packets::play::clientbound::JoinGame::Creative; + packet.dimension = packets::play::clientbound::JoinGame::Overworld; packet.difficulty = 2; packet.maxPlayers = 255; packet.levelType = QStringLiteral("default"); @@ -221,7 +265,7 @@ void Client::readPacketPlay(const packets::play::serverbound::PacketType type, c switch(type) { using namespace packets::play; - case serverbound::PacketClientSettings: + case serverbound::PacketType::ClientSettings: { { serverbound::ClientSettings packet(dataStream); @@ -245,7 +289,18 @@ void Client::readPacketPlay(const packets::play::serverbound::PacketType type, c } break; } - case serverbound::PacketPluginMessage: + case serverbound::PacketType::InteractEntity: + { + serverbound::InteractEntity packet(dataStream); + qDebug() << "entityId" << packet.entityId; + qDebug() << "type" << packet.type; +// qDebug() << "targetX" << packet.targetX; +// qDebug() << "targetY" << packet.targetY; +// qDebug() << "targetZ" << packet.targetZ; +// qDebug() << "hand" << packet.hand; + break; + } + case serverbound::PacketType::PluginMessage: { serverbound::PluginMessage packet(dataStream); qDebug() << "channel" << packet.channel; diff --git a/client.h b/client.h index 8973654..3596f2d 100644 --- a/client.h +++ b/client.h @@ -1,6 +1,7 @@ #pragma once #include +#include #include "mcdatastream.h" #include "packets.h" @@ -14,6 +15,12 @@ class Client : public QObject public: explicit Client(QTcpSocket *socket, QObject *parent = nullptr); + SocketState state() const { return m_state; } + + void keepAlive(); + void sendChatMessage(); + void trialDisconnect(); + private Q_SLOTS: void readyRead(); void disconnected(); @@ -30,4 +37,9 @@ private: qint32 m_packetSize; SocketState m_state; + + const QDateTime m_connectedSince; + QDateTime m_lastKeepAliveSent; + QDateTime m_lastKeepAliveReceived; + QDateTime m_lastChatMessage; }; diff --git a/main.cpp b/main.cpp index e1c3f29..f791fb5 100644 --- a/main.cpp +++ b/main.cpp @@ -1,6 +1,9 @@ #include #include #include +#include +#include +#include #include "client.h" @@ -19,10 +22,37 @@ int main(int argc, char *argv[]) "%{function}(): " "%{message}")); + QList> clients; + + QTimer timer; + timer.setInterval(100); + QObject::connect(&timer, &QTimer::timeout, [&clients](){ + const auto now = QDateTime::currentDateTime(); + for (auto iter = std::begin(clients); iter != std::end(clients); ) + { + if ((*iter).isNull()) + { + iter = clients.erase(iter); + continue; + } + + auto &client = **iter; + if (client.state() == PlayState) + { + client.keepAlive(); + client.sendChatMessage(); + client.trialDisconnect(); + } + + iter++; + } + }); + timer.start(); + QTcpServer server; - QObject::connect(&server, &QTcpServer::newConnection, [&server](){ - new Client(server.nextPendingConnection()); + QObject::connect(&server, &QTcpServer::newConnection, [&server,&clients](){ + clients.append(new Client(server.nextPendingConnection())); }); if(!server.listen(QHostAddress::Any, 25565)) diff --git a/mcdatastream.cpp b/mcdatastream.cpp index cd5da16..253da20 100644 --- a/mcdatastream.cpp +++ b/mcdatastream.cpp @@ -82,6 +82,20 @@ void McDataStream::writeDouble(double value) setFloatingPointPrecision(precision); } +QUuid McDataStream::readUuid() +{ + char buf[16]; + readRawData(buf, 16); + return {}; +} + +void McDataStream::writeUuid(const QUuid &uuid) +{ + Q_UNUSED(uuid) + + writeRawData("\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0", 16); +} + template<> qint32 McDataStream::readVar(qint32 &bytesRead) { diff --git a/mcdatastream.h b/mcdatastream.h index ee83ba6..7808763 100644 --- a/mcdatastream.h +++ b/mcdatastream.h @@ -1,6 +1,7 @@ #pragma once #include +#include #include @@ -30,6 +31,9 @@ public: template void writeVar(T value); + + QUuid readUuid(); + void writeUuid(const QUuid &uuid); }; template diff --git a/packets.cpp b/packets.cpp index 9e92565..d01d4c4 100644 --- a/packets.cpp +++ b/packets.cpp @@ -26,7 +26,7 @@ void packets::status::clientbound::Response::serialize(McDataStream &stream) { QByteArray buffer; McDataStream tempStream(&buffer, QIODevice::WriteOnly); - tempStream.writeVar(PacketResponse); + tempStream.writeVar(qint32(PacketType::Response)); tempStream.writeVar(jsonResponse); stream.writeVar(buffer.length()); stream.writeRawData(buffer.constData(), buffer.length()); @@ -36,7 +36,7 @@ void packets::status::clientbound::Pong::serialize(McDataStream &stream) { QByteArray buffer; McDataStream tempStream(&buffer, QIODevice::WriteOnly); - tempStream.writeVar(PacketPong); + tempStream.writeVar(qint32(PacketType::Pong)); tempStream << payload; stream.writeVar(buffer.length()); stream.writeRawData(buffer.constData(), buffer.length()); @@ -51,7 +51,8 @@ void packets::login::clientbound::LoginSuccess::serialize(McDataStream &stream) { QByteArray buffer; McDataStream tempStream(&buffer, QIODevice::WriteOnly); - tempStream.writeVar(PacketLoginSuccess); + tempStream.writeVar(qint32(PacketType::LoginSuccess)); + //tempStream.writeUuid(uuid); tempStream.writeVar(uuid); tempStream.writeVar(username); stream.writeVar(buffer.length()); @@ -68,6 +69,21 @@ packets::play::serverbound::ClientSettings::ClientSettings(McDataStream &stream) mainHand = stream.readVar(); } +packets::play::serverbound::InteractEntity::InteractEntity(McDataStream &stream) +{ + entityId = stream.readVar(); + type = Type(stream.readVar()); + switch (type) + { + case InteractAt: + targetX = stream.readFloat(); + targetY = stream.readFloat(); + targetZ = stream.readFloat(); + case Interact: + hand = Hand(stream.readVar()); + } +} + packets::play::serverbound::PluginMessage::PluginMessage(McDataStream &stream) { channel = stream.readVar(); @@ -78,31 +94,52 @@ void packets::play::clientbound::ServerDifficulty::serialize(McDataStream &strea { QByteArray buffer; McDataStream tempStream(&buffer, QIODevice::WriteOnly); - tempStream.writeVar(PacketServerDifficulty); + tempStream.writeVar(qint32(PacketType::ServerDifficulty)); tempStream << difficulty; stream.writeVar(buffer.length()); stream.writeRawData(buffer.constData(), buffer.length()); } +void packets::play::clientbound::ChatMessage::serialize(McDataStream &stream) +{ + QByteArray buffer; + McDataStream tempStream(&buffer, QIODevice::WriteOnly); + tempStream.writeVar(qint32(PacketType::ChatMessage)); + tempStream.writeVar(jsonData); + tempStream << qint8(position); + stream.writeVar(buffer.length()); + stream.writeRawData(buffer.constData(), buffer.length()); +} + void packets::play::clientbound::PluginMessage::serialize(McDataStream &stream) { QByteArray buffer; McDataStream tempStream(&buffer, QIODevice::WriteOnly); - tempStream.writeVar(PacketPluginMessage); + tempStream.writeVar(qint32(PacketType::PluginMessage)); tempStream.writeVar(channel); buffer.append(data); stream.writeVar(buffer.length()); stream.writeRawData(buffer.constData(), buffer.length()); } +void packets::play::clientbound::Disconnect::serialize(McDataStream &stream) +{ + QByteArray buffer; + McDataStream tempStream(&buffer, QIODevice::WriteOnly); + tempStream.writeVar(qint32(PacketType::Disconnect)); + tempStream.writeVar(reason); + stream.writeVar(buffer.length()); + stream.writeRawData(buffer.constData(), buffer.length()); +} + void packets::play::clientbound::JoinGame::serialize(McDataStream &stream) { QByteArray buffer; McDataStream tempStream(&buffer, QIODevice::WriteOnly); - tempStream.writeVar(PacketJoinGame); + tempStream.writeVar(qint32(PacketType::JoinGame)); tempStream << entityid - << gamemode - << dimension + << quint8(gamemode) + << qint32(dimension) << difficulty << maxPlayers; tempStream.writeVar(levelType); @@ -115,7 +152,7 @@ void packets::play::clientbound::PlayerAbilities::serialize(McDataStream &stream { QByteArray buffer; McDataStream tempStream(&buffer, QIODevice::WriteOnly); - tempStream.writeVar(PacketPlayerAbilities); + tempStream.writeVar(qint32(PacketType::PlayerAbilities)); tempStream << flags; tempStream.writeFloat(flyingSpeed); tempStream.writeFloat(fieldOfViewModifier); @@ -127,7 +164,7 @@ void packets::play::clientbound::PlayerPositionAndLook::serialize(McDataStream & { QByteArray buffer; McDataStream tempStream(&buffer, QIODevice::WriteOnly); - tempStream.writeVar(PacketPlayerPositionAndLook); + tempStream.writeVar(qint32(PacketType::PlayerPositionAndLook)); tempStream.writeDouble(x); tempStream.writeDouble(y); tempStream.writeDouble(z); @@ -143,8 +180,18 @@ void packets::play::clientbound::SpawnPosition::serialize(McDataStream &stream) { QByteArray buffer; McDataStream tempStream(&buffer, QIODevice::WriteOnly); - tempStream.writeVar(PacketSpawnPosition); + tempStream.writeVar(qint32(PacketType::SpawnPosition)); tempStream.writePosition(location); stream.writeVar(buffer.length()); stream.writeRawData(buffer.constData(), buffer.length()); } + +void packets::play::clientbound::KeepAlive::serialize(McDataStream &stream) +{ + QByteArray buffer; + McDataStream tempStream(&buffer, QIODevice::WriteOnly); + tempStream.writeVar(qint32(PacketType::KeepAlive)); + tempStream << keepAliveId; + stream.writeVar(buffer.length()); + stream.writeRawData(buffer.constData(), buffer.length()); +} diff --git a/packets.h b/packets.h index ccaa68c..b1340e8 100644 --- a/packets.h +++ b/packets.h @@ -2,6 +2,8 @@ #include #include +#include +#include #include @@ -12,7 +14,10 @@ enum SocketState { HandshakingState, StatusState, LoginState, PlayState, ClosedS namespace packets { namespace handshaking { namespace serverbound { - enum PacketType { PacketHandshake }; + Q_NAMESPACE + + enum class PacketType { Handshake }; + Q_ENUM_NS(PacketType) struct Handshake { Handshake(McDataStream &stream); @@ -30,7 +35,10 @@ namespace packets { namespace status { namespace serverbound { - enum PacketType { PacketRequest, PacketPing }; + Q_NAMESPACE + + enum class PacketType { Request, Ping }; + Q_ENUM_NS(PacketType) struct Request { Request(McDataStream &stream); @@ -44,7 +52,10 @@ namespace packets { } namespace clientbound { - enum PacketType { PacketResponse, PacketPong }; + Q_NAMESPACE + + enum class PacketType { Response, Pong }; + Q_ENUM_NS(PacketType) struct Response { QString jsonResponse; @@ -62,7 +73,10 @@ namespace packets { namespace login { namespace serverbound { - enum PacketType { PacketLogin }; + Q_NAMESPACE + + enum class PacketType { Login }; + Q_ENUM_NS(PacketType) struct Login { Login(McDataStream &stream); @@ -72,7 +86,10 @@ namespace packets { } namespace clientbound { - enum PacketType { PacketLoginSuccess = 0x02 }; + Q_NAMESPACE + + enum class PacketType { LoginSuccess = 0x02 }; + Q_ENUM_NS(PacketType) struct LoginSuccess { QString uuid; @@ -85,7 +102,10 @@ namespace packets { namespace play { namespace serverbound { - enum PacketType { PacketClientSettings = 0x04, PacketPluginMessage = 0x0A }; + Q_NAMESPACE + + enum class PacketType { ClientSettings = 0x04, InteractEntity = 0x0E, PluginMessage = 0x0A }; + Q_ENUM_NS(PacketType) struct ClientSettings { ClientSettings(McDataStream &stream); @@ -98,6 +118,19 @@ namespace packets { qint32 mainHand; }; + struct InteractEntity { + InteractEntity(McDataStream &stream); + + qint32 entityId; + enum Type : qint32 { Interact, Attack, InteractAt }; + Type type; + std::optional targetX; + std::optional targetY; + std::optional targetZ; + enum Hand : qint32 { MainHand, OffHand }; + std::optional hand; + }; + struct PluginMessage { PluginMessage(McDataStream &stream); @@ -107,8 +140,11 @@ namespace packets { } namespace clientbound { - enum PacketType { PacketServerDifficulty = 0x0D, PacketPluginMessage = 0x19, PacketJoinGame = 0x25, PacketPlayerAbilities = 0x2E, PacketPlayerPositionAndLook = 0x32, - PacketSpawnPosition = 0x49 }; + Q_NAMESPACE + + enum class PacketType { ServerDifficulty = 0x0D, ChatMessage = 0x0E, PluginMessage = 0x19, Disconnect = 0x1B, KeepAlive = 0x21, + JoinGame = 0x25, PlayerAbilities = 0x2E, PlayerPositionAndLook = 0x32, SpawnPosition = 0x49 }; + Q_ENUM_NS(PacketType) struct ServerDifficulty { quint8 difficulty; @@ -116,6 +152,14 @@ namespace packets { void serialize(McDataStream &stream); }; + struct ChatMessage { + QString jsonData; + enum Position : qint8 { Chat, SystemMessage, GameInfo }; + Position position; + + void serialize(McDataStream &stream); + }; + struct PluginMessage { QString channel; QByteArray data; @@ -123,10 +167,18 @@ namespace packets { void serialize(McDataStream &stream); }; + struct Disconnect { + QString reason; + + void serialize(McDataStream &stream); + }; + struct JoinGame { qint32 entityid; - quint8 gamemode; - qint32 dimension; + enum Gamemode : quint8 { Survival, Creative, Adventure, Spectator, Hardcore = 0x08 }; + Gamemode gamemode; + enum Dimension : qint32 { Nether=-1, Overworld=0, End=1 }; + Dimension dimension; quint8 difficulty; quint8 maxPlayers; QString levelType; @@ -160,6 +212,12 @@ namespace packets { void serialize(McDataStream &stream); }; + + struct KeepAlive { + qint64 keepAliveId; + + void serialize(McDataStream &stream); + }; } } }