2019-01-05 02:28:13 +01:00
|
|
|
/*
|
|
|
|
* IXWebSocketPerMessageDeflateCodec.cpp
|
|
|
|
* Author: Benjamin Sergeant
|
|
|
|
* Copyright (c) 2018-2019 Machine Zone, Inc. All rights reserved.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "IXWebSocketPerMessageDeflateCodec.h"
|
|
|
|
|
2019-09-23 19:25:23 +02:00
|
|
|
#include "IXWebSocketPerMessageDeflateOptions.h"
|
2019-01-05 02:28:13 +01:00
|
|
|
#include <cassert>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
namespace
|
|
|
|
{
|
|
|
|
// The passed in size (4) is important, without it the string litteral
|
2019-02-21 03:59:07 +01:00
|
|
|
// is treated as a char* and the null termination (\x00) makes it
|
2019-01-05 02:28:13 +01:00
|
|
|
// look like an empty string.
|
|
|
|
const std::string kEmptyUncompressedBlock = std::string("\x00\x00\xff\xff", 4);
|
2019-09-23 19:25:23 +02:00
|
|
|
} // namespace
|
2019-01-05 02:28:13 +01:00
|
|
|
|
|
|
|
namespace ix
|
|
|
|
{
|
|
|
|
//
|
|
|
|
// Compressor
|
|
|
|
//
|
|
|
|
WebSocketPerMessageDeflateCompressor::WebSocketPerMessageDeflateCompressor()
|
|
|
|
{
|
2020-08-01 07:54:57 +02:00
|
|
|
#ifdef IXWEBSOCKET_USE_ZLIB
|
2019-01-05 02:28:13 +01:00
|
|
|
memset(&_deflateState, 0, sizeof(_deflateState));
|
|
|
|
|
|
|
|
_deflateState.zalloc = Z_NULL;
|
|
|
|
_deflateState.zfree = Z_NULL;
|
|
|
|
_deflateState.opaque = Z_NULL;
|
2020-08-01 07:54:57 +02:00
|
|
|
#endif
|
2019-01-05 02:28:13 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
WebSocketPerMessageDeflateCompressor::~WebSocketPerMessageDeflateCompressor()
|
|
|
|
{
|
2020-08-01 07:54:57 +02:00
|
|
|
#ifdef IXWEBSOCKET_USE_ZLIB
|
2019-01-05 02:28:13 +01:00
|
|
|
deflateEnd(&_deflateState);
|
2020-08-01 07:54:57 +02:00
|
|
|
#endif
|
2019-01-05 02:28:13 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
bool WebSocketPerMessageDeflateCompressor::init(uint8_t deflateBits,
|
|
|
|
bool clientNoContextTakeOver)
|
|
|
|
{
|
2020-08-01 07:54:57 +02:00
|
|
|
#ifdef IXWEBSOCKET_USE_ZLIB
|
2019-09-23 19:25:23 +02:00
|
|
|
int ret = deflateInit2(&_deflateState,
|
|
|
|
Z_DEFAULT_COMPRESSION,
|
|
|
|
Z_DEFLATED,
|
|
|
|
-1 * deflateBits,
|
|
|
|
4, // memory level 1-9
|
|
|
|
Z_DEFAULT_STRATEGY);
|
2019-01-05 02:28:13 +01:00
|
|
|
|
|
|
|
if (ret != Z_OK) return false;
|
|
|
|
|
2019-09-23 19:25:23 +02:00
|
|
|
_flush = (clientNoContextTakeOver) ? Z_FULL_FLUSH : Z_SYNC_FLUSH;
|
2019-01-05 02:28:13 +01:00
|
|
|
|
|
|
|
return true;
|
2020-08-01 07:54:57 +02:00
|
|
|
#else
|
|
|
|
return false;
|
|
|
|
#endif
|
2019-01-05 02:28:13 +01:00
|
|
|
}
|
|
|
|
|
2020-07-08 03:17:44 +02:00
|
|
|
template<typename T>
|
|
|
|
bool WebSocketPerMessageDeflateCompressor::endsWithEmptyUnCompressedBlock(const T& value)
|
2019-01-05 02:28:13 +01:00
|
|
|
{
|
2020-07-08 03:17:44 +02:00
|
|
|
if (kEmptyUncompressedBlock.size() > value.size()) return false;
|
|
|
|
auto N = value.size();
|
|
|
|
return value[N - 1] == kEmptyUncompressedBlock[3] &&
|
|
|
|
value[N - 2] == kEmptyUncompressedBlock[2] &&
|
|
|
|
value[N - 3] == kEmptyUncompressedBlock[1] &&
|
|
|
|
value[N - 4] == kEmptyUncompressedBlock[0];
|
2019-01-05 02:28:13 +01:00
|
|
|
}
|
|
|
|
|
2019-09-23 19:25:23 +02:00
|
|
|
bool WebSocketPerMessageDeflateCompressor::compress(const std::string& in, std::string& out)
|
2020-07-08 03:17:44 +02:00
|
|
|
{
|
|
|
|
return compressData(in, out);
|
|
|
|
}
|
|
|
|
|
2022-01-10 19:34:24 +01:00
|
|
|
bool WebSocketPerMessageDeflateCompressor::compress(const IXWebSocketSendData& in,
|
|
|
|
std::string& out)
|
|
|
|
{
|
|
|
|
return compressData(in, out);
|
|
|
|
}
|
|
|
|
|
2020-07-08 19:39:46 +02:00
|
|
|
bool WebSocketPerMessageDeflateCompressor::compress(const std::string& in,
|
|
|
|
std::vector<uint8_t>& out)
|
2020-07-08 03:17:44 +02:00
|
|
|
{
|
|
|
|
return compressData(in, out);
|
|
|
|
}
|
|
|
|
|
2020-07-08 19:39:46 +02:00
|
|
|
bool WebSocketPerMessageDeflateCompressor::compress(const std::vector<uint8_t>& in,
|
|
|
|
std::string& out)
|
2020-07-08 03:17:44 +02:00
|
|
|
{
|
|
|
|
return compressData(in, out);
|
|
|
|
}
|
|
|
|
|
2020-07-08 19:39:46 +02:00
|
|
|
bool WebSocketPerMessageDeflateCompressor::compress(const std::vector<uint8_t>& in,
|
|
|
|
std::vector<uint8_t>& out)
|
2020-07-08 03:17:44 +02:00
|
|
|
{
|
|
|
|
return compressData(in, out);
|
|
|
|
}
|
|
|
|
|
2020-07-08 19:39:46 +02:00
|
|
|
template<typename T, typename S>
|
|
|
|
bool WebSocketPerMessageDeflateCompressor::compressData(const T& in, S& out)
|
2019-01-05 02:28:13 +01:00
|
|
|
{
|
2020-08-01 07:54:57 +02:00
|
|
|
#ifdef IXWEBSOCKET_USE_ZLIB
|
2019-01-05 02:28:13 +01:00
|
|
|
//
|
|
|
|
// 7.2.1. Compression
|
2019-02-21 03:59:07 +01:00
|
|
|
//
|
2019-01-05 02:28:13 +01:00
|
|
|
// An endpoint uses the following algorithm to compress a message.
|
2019-02-21 03:59:07 +01:00
|
|
|
//
|
2019-01-05 02:28:13 +01:00
|
|
|
// 1. Compress all the octets of the payload of the message using
|
|
|
|
// DEFLATE.
|
2019-02-21 03:59:07 +01:00
|
|
|
//
|
2019-01-05 02:28:13 +01:00
|
|
|
// 2. If the resulting data does not end with an empty DEFLATE block
|
|
|
|
// with no compression (the "BTYPE" bits are set to 00), append an
|
|
|
|
// empty DEFLATE block with no compression to the tail end.
|
2019-02-21 03:59:07 +01:00
|
|
|
//
|
2019-01-05 02:28:13 +01:00
|
|
|
// 3. Remove 4 octets (that are 0x00 0x00 0xff 0xff) from the tail end.
|
|
|
|
// After this step, the last octet of the compressed data contains
|
|
|
|
// (possibly part of) the DEFLATE header bits with the "BTYPE" bits
|
|
|
|
// set to 00.
|
|
|
|
//
|
|
|
|
size_t output;
|
|
|
|
|
2020-04-14 06:38:15 +02:00
|
|
|
// Clear output
|
|
|
|
out.clear();
|
|
|
|
|
2019-01-05 02:28:13 +01:00
|
|
|
if (in.empty())
|
|
|
|
{
|
2020-03-23 23:21:53 +01:00
|
|
|
// See issue #167
|
|
|
|
// The normal buffer size should be 6 but
|
|
|
|
// we remove the 4 octets from the tail (#4)
|
|
|
|
uint8_t buf[2] = {0x02, 0x00};
|
2020-07-08 03:17:44 +02:00
|
|
|
out.push_back(buf[0]);
|
|
|
|
out.push_back(buf[1]);
|
2020-03-23 23:21:53 +01:00
|
|
|
|
2019-01-05 02:28:13 +01:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
_deflateState.avail_in = (uInt) in.size();
|
|
|
|
_deflateState.next_in = (Bytef*) in.data();
|
|
|
|
|
|
|
|
do
|
|
|
|
{
|
|
|
|
// Output to local buffer
|
2020-08-18 01:36:24 +02:00
|
|
|
_deflateState.avail_out = (uInt) _compressBuffer.size();
|
|
|
|
_deflateState.next_out = &_compressBuffer.front();
|
2019-01-05 02:28:13 +01:00
|
|
|
|
|
|
|
deflate(&_deflateState, _flush);
|
|
|
|
|
2020-08-18 01:36:24 +02:00
|
|
|
output = _compressBuffer.size() - _deflateState.avail_out;
|
2019-01-05 02:28:13 +01:00
|
|
|
|
2020-08-18 01:36:24 +02:00
|
|
|
out.insert(out.end(), _compressBuffer.begin(), _compressBuffer.begin() + output);
|
2019-01-05 02:28:13 +01:00
|
|
|
} while (_deflateState.avail_out == 0);
|
|
|
|
|
2020-07-08 03:17:44 +02:00
|
|
|
if (endsWithEmptyUnCompressedBlock(out))
|
2019-01-05 02:28:13 +01:00
|
|
|
{
|
|
|
|
out.resize(out.size() - 4);
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
2020-08-01 07:54:57 +02:00
|
|
|
#else
|
|
|
|
return false;
|
|
|
|
#endif
|
2019-01-05 02:28:13 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Decompressor
|
|
|
|
//
|
|
|
|
WebSocketPerMessageDeflateDecompressor::WebSocketPerMessageDeflateDecompressor()
|
|
|
|
{
|
2020-08-01 07:54:57 +02:00
|
|
|
#ifdef IXWEBSOCKET_USE_ZLIB
|
2019-01-05 02:28:13 +01:00
|
|
|
memset(&_inflateState, 0, sizeof(_inflateState));
|
|
|
|
|
|
|
|
_inflateState.zalloc = Z_NULL;
|
|
|
|
_inflateState.zfree = Z_NULL;
|
|
|
|
_inflateState.opaque = Z_NULL;
|
|
|
|
_inflateState.avail_in = 0;
|
|
|
|
_inflateState.next_in = Z_NULL;
|
2020-08-01 07:54:57 +02:00
|
|
|
#endif
|
2019-01-05 02:28:13 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
WebSocketPerMessageDeflateDecompressor::~WebSocketPerMessageDeflateDecompressor()
|
|
|
|
{
|
2020-08-01 07:54:57 +02:00
|
|
|
#ifdef IXWEBSOCKET_USE_ZLIB
|
2019-01-05 02:28:13 +01:00
|
|
|
inflateEnd(&_inflateState);
|
2020-08-01 07:54:57 +02:00
|
|
|
#endif
|
2019-01-05 02:28:13 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
bool WebSocketPerMessageDeflateDecompressor::init(uint8_t inflateBits,
|
|
|
|
bool clientNoContextTakeOver)
|
|
|
|
{
|
2020-08-01 07:54:57 +02:00
|
|
|
#ifdef IXWEBSOCKET_USE_ZLIB
|
2019-09-23 19:25:23 +02:00
|
|
|
int ret = inflateInit2(&_inflateState, -1 * inflateBits);
|
2019-01-05 02:28:13 +01:00
|
|
|
|
|
|
|
if (ret != Z_OK) return false;
|
|
|
|
|
2019-09-23 19:25:23 +02:00
|
|
|
_flush = (clientNoContextTakeOver) ? Z_FULL_FLUSH : Z_SYNC_FLUSH;
|
2019-01-05 02:28:13 +01:00
|
|
|
|
|
|
|
return true;
|
2020-08-01 07:54:57 +02:00
|
|
|
#else
|
|
|
|
return false;
|
|
|
|
#endif
|
2019-01-05 02:28:13 +01:00
|
|
|
}
|
|
|
|
|
2019-09-23 19:25:23 +02:00
|
|
|
bool WebSocketPerMessageDeflateDecompressor::decompress(const std::string& in, std::string& out)
|
2019-01-05 02:28:13 +01:00
|
|
|
{
|
2020-08-01 07:54:57 +02:00
|
|
|
#ifdef IXWEBSOCKET_USE_ZLIB
|
2019-01-05 02:28:13 +01:00
|
|
|
//
|
|
|
|
// 7.2.2. Decompression
|
2019-02-21 03:59:07 +01:00
|
|
|
//
|
2019-01-05 02:28:13 +01:00
|
|
|
// An endpoint uses the following algorithm to decompress a message.
|
2019-02-21 03:59:07 +01:00
|
|
|
//
|
2019-01-05 02:28:13 +01:00
|
|
|
// 1. Append 4 octets of 0x00 0x00 0xff 0xff to the tail end of the
|
|
|
|
// payload of the message.
|
2019-02-21 03:59:07 +01:00
|
|
|
//
|
2019-01-05 02:28:13 +01:00
|
|
|
// 2. Decompress the resulting data using DEFLATE.
|
2019-02-21 03:59:07 +01:00
|
|
|
//
|
2019-01-05 02:28:13 +01:00
|
|
|
std::string inFixed(in);
|
|
|
|
inFixed += kEmptyUncompressedBlock;
|
|
|
|
|
|
|
|
_inflateState.avail_in = (uInt) inFixed.size();
|
2019-09-23 19:25:23 +02:00
|
|
|
_inflateState.next_in = (unsigned char*) (const_cast<char*>(inFixed.data()));
|
2019-01-05 02:28:13 +01:00
|
|
|
|
2020-04-14 06:38:15 +02:00
|
|
|
// Clear output
|
|
|
|
out.clear();
|
|
|
|
|
2019-01-05 02:28:13 +01:00
|
|
|
do
|
|
|
|
{
|
2020-08-18 01:36:24 +02:00
|
|
|
_inflateState.avail_out = (uInt) _compressBuffer.size();
|
|
|
|
_inflateState.next_out = &_compressBuffer.front();
|
2019-01-05 02:28:13 +01:00
|
|
|
|
|
|
|
int ret = inflate(&_inflateState, Z_SYNC_FLUSH);
|
|
|
|
|
|
|
|
if (ret == Z_NEED_DICT || ret == Z_DATA_ERROR || ret == Z_MEM_ERROR)
|
|
|
|
{
|
|
|
|
return false; // zlib error
|
|
|
|
}
|
|
|
|
|
2020-08-18 01:36:24 +02:00
|
|
|
out.append(reinterpret_cast<char*>(&_compressBuffer.front()),
|
|
|
|
_compressBuffer.size() - _inflateState.avail_out);
|
2019-01-05 02:28:13 +01:00
|
|
|
} while (_inflateState.avail_out == 0);
|
|
|
|
|
|
|
|
return true;
|
2020-08-01 07:54:57 +02:00
|
|
|
#else
|
|
|
|
return false;
|
|
|
|
#endif
|
2019-01-05 02:28:13 +01:00
|
|
|
}
|
2019-09-23 19:25:23 +02:00
|
|
|
} // namespace ix
|