Snake/SnakeServer/ClientProtocol/clientprotocol.cpp

310 lines
5.9 KiB
C++
Raw Normal View History

2019-02-12 13:12:11 +03:00
#include "clientprotocol.h"
2019-02-13 13:03:40 +03:00
#include <QVariantMap>
#include <typeinfo>
2019-02-13 22:03:29 +03:00
#include "snakeitem.h"
#include "player.h"
2019-02-13 13:03:40 +03:00
#define DEFAULT_GAME_PORT 7777
namespace ClientProtocol {
Header::Header() {
reset();
}
bool Header::isValid() const {
2019-02-13 22:03:29 +03:00
if (sizeof (*this) != 2) {
2019-02-13 13:03:40 +03:00
return false;
}
switch (command) {
2019-02-15 23:36:59 +03:00
case Ping: {
2019-02-13 13:03:40 +03:00
2019-02-13 22:03:29 +03:00
if (type > 1 || size > 0)
2019-02-13 13:03:40 +03:00
return false;
return true;
}
2019-02-15 23:36:59 +03:00
case Item: {
2019-02-13 22:03:29 +03:00
switch (type) {
case Request: return size == 36; // key sha256 (32byte) + id item 4
case Responke: return size < (snakeSize + sizeof (Class));
2019-02-13 22:03:29 +03:00
}
return false;
}
2019-02-15 23:36:59 +03:00
case Login: {
2019-02-13 22:03:29 +03:00
switch (type) {
case Request: return size == 96; // key sha256 (32byte) + maxsize of name of gmail (64)
case Responke: return size < MAX_SIZE_PLAYER && size > MIN_SIZE_PLAYER;
}
return false;
}
2019-02-15 23:36:59 +03:00
case PlayerData: {
2019-02-13 22:03:29 +03:00
switch (type) {
case Request: return size == 96; // key sha256 (32byte) + maxsize of name of gmail (64)
case Responke: return size < MAX_SIZE_PLAYER && size > MIN_SIZE_PLAYER;
}
return false;
}
2019-02-13 13:03:40 +03:00
default: return false;
}
}
void Header::reset() {
size = 0;
2019-02-15 23:36:59 +03:00
command = Undefined;
2019-02-13 13:03:40 +03:00
type = Responke;
}
Package::Package() {
reset();
}
bool Package::isValid() const {
if (!hdr.isValid()) {
return false;
}
return hdr.size == data.size();
}
2019-02-15 23:36:59 +03:00
bool Package::parse(QVariantMap& res) const {
2019-02-13 13:03:40 +03:00
if (!isValid())
2019-02-15 23:36:59 +03:00
return false;
2019-02-13 13:03:40 +03:00
res["command"] = hdr.command;
res["type"] = hdr.type;
2019-02-13 13:03:40 +03:00
switch (hdr.command) {
2019-02-15 23:36:59 +03:00
case Ping: {
2019-02-13 13:03:40 +03:00
if (hdr.type == Responke) {
res["res"] = "Pong";
} else {
res["value"] = "Ping";
}
break;
}
2019-02-15 23:36:59 +03:00
case Item: {
if (hdr.type == Responke) {
QDataStream stream(data);
unsigned char type;
stream >> type;
switch (type) {
case SnakeData:{
if (!SnakeItem::read(stream, res)) {
2019-02-15 23:36:59 +03:00
return false;
}
break;
}
default: break;
}
} else {
res["hash"] = data.left(32);
res["id"] = data.right(4).toInt();
}
break;
}
2019-02-15 23:36:59 +03:00
case Login: {
if (hdr.type == Responke) {
QDataStream stream(data);
if (!Player::read(stream, res)) {
2019-02-15 23:36:59 +03:00
return false;
}
} else {
QDataStream stream(data);
QString gmail;
QByteArray hash;
stream >> gmail;
stream >> hash;
res["gmail"] = gmail;
res["hash"] = hash;
}
break;
}
2019-02-15 23:36:59 +03:00
case PlayerData: {
2019-02-13 22:03:29 +03:00
if (hdr.type == Responke) {
QDataStream stream(data);
if (!Player::read(stream, res)) {
2019-02-15 23:36:59 +03:00
return false;
}
} else {
QDataStream stream(data);
QString gmail;
QByteArray hash;
stream >> gmail;
stream >> hash;
res["gmail"] = gmail;
res["hash"] = hash;
}
break;
}
2019-02-13 13:03:40 +03:00
default:
2019-02-15 23:36:59 +03:00
return false;
2019-02-13 13:03:40 +03:00
}
2019-02-15 23:36:59 +03:00
return true;
}
bool Package::create(const QVariantMap &map) {
auto command = static_cast<unsigned char>(map.value("command", 0xff).toInt());
auto type = static_cast<unsigned char>(map.value("type", 0xff).toInt());
if (command == 0xff || type == 0xff) {
return false;
}
switch (command) {
case Ping: {
break;
}
case Item: {
if (type == Responke) {
QDataStream stream(data);
auto cls = static_cast<unsigned char>(map.value("class", 0xff).toInt());
stream << static_cast<unsigned char>(cls);
switch (cls) {
case SnakeData:{
if (!SnakeItem::write(stream, map)) {
return false;
}
break;
}
default: return false;
}
} else {
QDataStream stream(data);
QByteArray hash = map.value("hash", "").toByteArray();
int id = static_cast<Class>(map.value("id", 0).toInt());
if (hash.size() != 32 || !id) {
return false;
}
data.append(hash.data());
data.append(reinterpret_cast<char*>(&id), sizeof (id));
}
break;
}
case Login: {
if (type == Responke) {
QDataStream stream(data);
if (!Player::write(stream, map)) {
return false;
}
} else {
QDataStream stream(data);
QByteArray hash = map.value("hash", "").toByteArray();
QString gmail = map.value("id", "").toString();
if (hash.size() != 32 || gmail.isEmpty()) {
return false;
}
stream << gmail;
stream << hash;
}
break;
}
case PlayerData: {
if (type == Responke) {
QDataStream stream(data);
if (!Player::write(stream, map)) {
return false;
}
} else {
QDataStream stream(data);
QByteArray hash = map.value("hash", "").toByteArray();
QString gmail = map.value("id", "").toString();
if (hash.size() != 32 || gmail.isEmpty()) {
return false;
}
stream << gmail;
stream << hash;
}
break;
}
default: return false;
}
hdr.command = command;
hdr.type = type;
return isValid();
2019-02-13 13:03:40 +03:00
}
QByteArray Package::toBytes() const {
QByteArray res;
res.append(reinterpret_cast<char*>(const_cast<Header*>(&hdr)),
sizeof (hdr));
res.append(data);
return res;
}
void Package::reset() {
hdr.reset();
data.clear();
}
}