Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add ZLIB compression support #4080

Open
wants to merge 6 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
15 changes: 15 additions & 0 deletions Shared/mods/deathmatch/logic/Enums.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -69,6 +69,7 @@ ADD_ENUM(StringEncodeFunction::AES128, "aes128")
ADD_ENUM(StringEncodeFunction::RSA, "rsa")
ADD_ENUM(StringEncodeFunction::BASE64, "base64")
ADD_ENUM(StringEncodeFunction::BASE32, "base32")
ADD_ENUM(StringEncodeFunction::ZLIB, "zlib")
IMPLEMENT_ENUM_CLASS_END("string-encode-function")

IMPLEMENT_ENUM_CLASS_BEGIN(KeyPairAlgorithm)
Expand All @@ -84,6 +85,20 @@ ADD_ENUM(HmacAlgorithm::SHA384, "sha384")
ADD_ENUM(HmacAlgorithm::SHA512, "sha512")
IMPLEMENT_ENUM_CLASS_END("hmac-algorithm")

IMPLEMENT_ENUM_CLASS_BEGIN(ZLibFormat)
ADD_ENUM(ZLibFormat::ZRAW, "raw")
ADD_ENUM(ZLibFormat::ZLIB, "zlib")
ADD_ENUM(ZLibFormat::GZIP, "gzip")
IMPLEMENT_ENUM_CLASS_END("zlib-format")

IMPLEMENT_ENUM_CLASS_BEGIN(ZLibStrategy)
ADD_ENUM(ZLibStrategy::DEFAULT, "default")
ADD_ENUM(ZLibStrategy::FILTERED, "filtered")
ADD_ENUM(ZLibStrategy::HUFFMAN_ONLY, "huffman")
ADD_ENUM(ZLibStrategy::RLE, "rle")
ADD_ENUM(ZLibStrategy::FIXED, "fixed")
IMPLEMENT_ENUM_CLASS_END("zlib-strategy")

IMPLEMENT_ENUM_CLASS_BEGIN(WorldSpecialProperty)
ADD_ENUM(WorldSpecialProperty::HOVERCARS, "hovercars")
ADD_ENUM(WorldSpecialProperty::AIRCARS, "aircars")
Expand Down
2 changes: 2 additions & 0 deletions Shared/mods/deathmatch/logic/Enums.h
Original file line number Diff line number Diff line change
Expand Up @@ -72,6 +72,8 @@ DECLARE_ENUM_CLASS(PasswordHashFunction);
DECLARE_ENUM_CLASS(StringEncodeFunction);
DECLARE_ENUM_CLASS(KeyPairAlgorithm);
DECLARE_ENUM_CLASS(HmacAlgorithm);
DECLARE_ENUM_CLASS(ZLibFormat);
DECLARE_ENUM_CLASS(ZLibStrategy);

enum class WorldSpecialProperty
{
Expand Down
152 changes: 150 additions & 2 deletions Shared/mods/deathmatch/logic/luadefs/CLuaCryptDefs.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -403,7 +403,8 @@ int CLuaCryptDefs::EncodeString(lua_State* luaVM)
argStream.ReadEnumString(algorithm);
argStream.ReadString(data);

if ((algorithm != StringEncodeFunction::BASE64 && algorithm != StringEncodeFunction::BASE32) || argStream.NextIsTable())
if ((algorithm != StringEncodeFunction::BASE64 && algorithm != StringEncodeFunction::BASE32 && algorithm != StringEncodeFunction::ZLIB) ||
argStream.NextIsTable())
{
argStream.ReadStringMap(options);
}
Expand Down Expand Up @@ -727,6 +728,88 @@ int CLuaCryptDefs::EncodeString(lua_State* luaVM)
}
return 1;
}
case StringEncodeFunction::ZLIB:
{
int compression = 9;
int format = (int)ZLibFormat::GZIP;
ZLibStrategy strategy = ZLibStrategy::DEFAULT;
if (!options["format"].empty() && !StringToEnum(options["format"], (ZLibFormat&)format) && !StringToZLibFormat(options["format"], format))
{
m_pScriptDebugging->LogCustom(luaVM, "Invalid value for field 'format'");
lua::Push(luaVM, false);
return 1;
}
if (!options["strategy"].empty() && !StringToEnum(options["strategy"], strategy))
{
m_pScriptDebugging->LogCustom(luaVM, "Invalid value for field 'strategy'");
lua::Push(luaVM, false);
return 1;
}
if (!options["compression"].empty())
{
compression = atoi(options["compression"].c_str());
if (compression < 0 || compression > 9)
{
m_pScriptDebugging->LogCustom(luaVM, "Value for field 'compression' is out of range (0-9)");
lua::Push(luaVM, false);
return 1;
}
}

// Async
if (VERIFY_FUNCTION(luaFunctionRef))
{
CLuaMain* pLuaMain = m_pLuaManager->GetVirtualMachine(luaVM);
if (pLuaMain)
{
CLuaShared::GetAsyncTaskScheduler()->PushTask(
[data, format, compression, strategy]
{
// Execute time-consuming task
SString output;
int result = SharedUtil::ZLibCompress(data, output, format, compression, strategy);
if (result == Z_STREAM_END)
return std::make_pair(output, true);
else
return std::make_pair(SString("zlib error: %i", result), false);
},
[luaFunctionRef](const std::pair<SString, bool>& result)
{
CLuaMain* pLuaMain = m_pLuaManager->GetVirtualMachine(luaFunctionRef.GetLuaVM());
if (pLuaMain)
{
CLuaArguments arguments;
if (result.second)
{
arguments.PushString(result.first);
arguments.Call(pLuaMain, luaFunctionRef);
}
else
{
m_pScriptDebugging->LogWarning(luaFunctionRef.GetLuaVM(), result.first.c_str());
arguments.PushBoolean(false);
arguments.Call(pLuaMain, luaFunctionRef);
}
}
});

lua_pushboolean(luaVM, true);
}
}
else // Sync
{
SString output;
int result = SharedUtil::ZLibCompress(data, output, format, compression, strategy);
if (result == Z_STREAM_END)
lua::Push(luaVM, output);
else
{
m_pScriptDebugging->LogWarning(luaVM, "zlib error: %i", result);
lua::Push(luaVM, false);
}
}
return 1;
}
default:
{
m_pScriptDebugging->LogCustom(luaVM, "Unknown encryption algorithm");
Expand All @@ -753,7 +836,8 @@ int CLuaCryptDefs::DecodeString(lua_State* luaVM)
argStream.ReadEnumString(algorithm);
argStream.ReadString(data);

if ((algorithm != StringEncodeFunction::BASE64 && algorithm != StringEncodeFunction::BASE32) || argStream.NextIsTable())
if ((algorithm != StringEncodeFunction::BASE64 && algorithm != StringEncodeFunction::BASE32 && algorithm != StringEncodeFunction::ZLIB) ||
argStream.NextIsTable())
{
argStream.ReadStringMap(options);
}
Expand Down Expand Up @@ -1084,6 +1168,70 @@ int CLuaCryptDefs::DecodeString(lua_State* luaVM)
}
return 1;
}
case StringEncodeFunction::ZLIB:
{
int format = 0;
if (!options["format"].empty() && !StringToEnum(options["format"], (ZLibFormat&)format) && !StringToZLibFormat(options["format"], format))
{
m_pScriptDebugging->LogCustom(luaVM, "Not supported value for field 'format'");
lua::Push(luaVM, false);
return 1;
}

// Async
if (VERIFY_FUNCTION(luaFunctionRef))
{
CLuaMain* pLuaMain = m_pLuaManager->GetVirtualMachine(luaVM);
if (pLuaMain)
{
CLuaShared::GetAsyncTaskScheduler()->PushTask(
[data, format]
{
// Execute time-consuming task
SString output;
int result = SharedUtil::ZLibUncompress(data, output, format);
if (result == Z_STREAM_END)
return std::make_pair(output, true);
else
return std::make_pair(SString("zlib error: %i", result), false);
},
[luaFunctionRef](const std::pair<SString, bool>& result)
{
CLuaMain* pLuaMain = m_pLuaManager->GetVirtualMachine(luaFunctionRef.GetLuaVM());
if (pLuaMain)
{
CLuaArguments arguments;
if (result.second)
{
arguments.PushString(result.first);
arguments.Call(pLuaMain, luaFunctionRef);
}
else
{
m_pScriptDebugging->LogWarning(luaFunctionRef.GetLuaVM(), result.first.c_str());
arguments.PushBoolean(false);
arguments.Call(pLuaMain, luaFunctionRef);
}
}
});

lua_pushboolean(luaVM, true);
}
}
else // Sync
{
SString output;
int result = SharedUtil::ZLibUncompress(data, output, format);
if (result == Z_STREAM_END)
lua::Push(luaVM, output);
else
{
m_pScriptDebugging->LogWarning(luaVM, "zlib error: %i", result);
lua::Push(luaVM, false);
}
}
return 1;
}
default:
{
m_pScriptDebugging->LogCustom(luaVM, "Unknown encryption algorithm");
Expand Down
80 changes: 80 additions & 0 deletions Shared/sdk/SharedUtil.Crypto.h
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@
#include <cryptopp/hmac.h>
#include <cryptopp/hex.h>
#include <cryptopp/md5.h>
#include <zlib/zlib.h>
#include "SString.h"

namespace SharedUtil
Expand Down Expand Up @@ -202,4 +203,83 @@ namespace SharedUtil

return result;
}

inline bool StringToZLibFormat(const std::string& format, int& outResult)
{
int value = atoi(format.c_str());
if ((value >= 9 && value <= 31) || (value >= -15 && value <= -9)) // allowed values: 9..31, -9..-15
{
outResult = value;
return true;
}
return false;
}

inline int ZLibCompress(const std::string& input, std::string& output, const int windowBits = (int)ZLibFormat::GZIP, const int compression = 9,
const ZLibStrategy strategy = ZLibStrategy::DEFAULT)
{
z_stream stream{};

int result = deflateInit2(&stream, compression, Z_DEFLATED, windowBits, MAX_MEM_LEVEL, (int)strategy);
if (result != Z_OK)
return result;

output.resize(deflateBound(&stream, input.size())); // resize to the upper bound of what the compressed size might be

stream.next_out = (Bytef*)output.data();
stream.avail_out = output.size();

stream.next_in = (z_const Bytef*)input.data();
stream.avail_in = input.size();

result = deflate(&stream, Z_FINISH);
result |= deflateEnd(&stream);

if (result == Z_STREAM_END)
output.resize(stream.total_out); // resize to the actual size

return result;
}

inline int ZLibUncompress(const std::string& input, std::string& output, int windowBits = 0)
{
if (windowBits == 0 && input.size() >= 2) // try to determine format automatically
{
if (input[0] == '\x1F' && input[1] == '\x8B')
windowBits = (int)ZLibFormat::GZIP;
else if (input[0] == '\x78')
windowBits = (int)ZLibFormat::ZLIB;
else
windowBits = (int)ZLibFormat::ZRAW;
}
z_stream stream{};

int result = inflateInit2(&stream, windowBits);
if (result != Z_OK)
return result;

stream.next_in = (z_const Bytef*)input.data();
stream.avail_in = input.size();

// Uncompress in chunks
std::string buffer;
buffer.resize(std::min(stream.avail_in, 128000U)); // use input length for chunk size (capped to 128k bytes which should be efficient enough)
while (true)
{
stream.next_out = (Bytef*)buffer.data();
stream.avail_out = buffer.size();

result = inflate(&stream, Z_NO_FLUSH);
if (result != Z_OK && result != Z_STREAM_END)
break;

output.append(buffer, 0, stream.total_out - output.size()); // append only what was written to buffer

if (result == Z_STREAM_END)
break;
}
result |= inflateEnd(&stream);
return result;
}

} // namespace SharedUtil
19 changes: 18 additions & 1 deletion Shared/sdk/SharedUtil.Hash.h
Original file line number Diff line number Diff line change
Expand Up @@ -49,7 +49,8 @@ enum class StringEncodeFunction
AES128,
RSA,
BASE64,
BASE32
BASE32,
ZLIB,
};

enum class KeyPairAlgorithm
Expand All @@ -67,6 +68,22 @@ enum class HmacAlgorithm
SHA512,
};

enum class ZLibFormat
{
ZRAW = -15,
ZLIB = 15,
GZIP = 31,
};

enum class ZLibStrategy
{
DEFAULT,
FILTERED,
HUFFMAN_ONLY,
RLE,
FIXED,
};

namespace SharedUtil
{
struct MD5
Expand Down
Loading