2020-11-09 22:52:22 +00:00
|
|
|
#include <IO/LZMADeflatingWriteBuffer.h>
|
2020-11-01 18:40:05 +00:00
|
|
|
|
2020-11-02 20:04:49 +00:00
|
|
|
namespace DB
|
|
|
|
{
|
2020-11-01 18:40:05 +00:00
|
|
|
namespace ErrorCodes
|
|
|
|
{
|
|
|
|
extern const int LZMA_STREAM_ENCODER_FAILED;
|
|
|
|
}
|
|
|
|
|
2020-11-09 22:52:22 +00:00
|
|
|
LZMADeflatingWriteBuffer::LZMADeflatingWriteBuffer(
|
2020-11-02 20:04:49 +00:00
|
|
|
std::unique_ptr<WriteBuffer> out_, int compression_level, size_t buf_size, char * existing_memory, size_t alignment)
|
2021-11-10 22:58:56 +00:00
|
|
|
: WriteBufferWithOwnMemoryDecorator(std::move(out_), buf_size, existing_memory, alignment)
|
2020-11-01 18:40:05 +00:00
|
|
|
{
|
2020-11-09 22:52:22 +00:00
|
|
|
|
2020-11-11 01:50:56 +00:00
|
|
|
lstr = LZMA_STREAM_INIT;
|
2020-11-01 18:40:05 +00:00
|
|
|
lstr.allocator = nullptr;
|
|
|
|
lstr.next_in = nullptr;
|
|
|
|
lstr.avail_in = 0;
|
|
|
|
lstr.next_out = nullptr;
|
|
|
|
lstr.avail_out = 0;
|
2020-11-02 20:04:49 +00:00
|
|
|
|
2020-11-01 18:40:05 +00:00
|
|
|
// options for further compression
|
|
|
|
lzma_options_lzma opt_lzma2;
|
2020-11-02 13:17:25 +00:00
|
|
|
if (lzma_lzma_preset(&opt_lzma2, compression_level))
|
2020-11-11 01:50:56 +00:00
|
|
|
throw Exception(ErrorCodes::LZMA_STREAM_ENCODER_FAILED, "lzma preset failed: lzma version: {}", LZMA_VERSION_STRING);
|
2020-11-02 13:17:25 +00:00
|
|
|
|
2020-11-01 18:40:05 +00:00
|
|
|
|
2020-11-11 01:50:56 +00:00
|
|
|
// LZMA_FILTER_X86 -
|
|
|
|
// LZMA2 - codec for *.xz files compression; LZMA is not suitable for this purpose
|
|
|
|
// VLI - variable length integer (in *.xz most integers encoded as VLI)
|
|
|
|
// LZMA_VLI_UNKNOWN (UINT64_MAX) - VLI value to denote that the value is unknown
|
2020-11-01 18:40:05 +00:00
|
|
|
lzma_filter filters[] = {
|
2020-11-04 00:35:16 +00:00
|
|
|
{.id = LZMA_FILTER_X86, .options = nullptr},
|
2020-11-02 20:04:49 +00:00
|
|
|
{.id = LZMA_FILTER_LZMA2, .options = &opt_lzma2},
|
2020-11-04 00:35:16 +00:00
|
|
|
{.id = LZMA_VLI_UNKNOWN, .options = nullptr},
|
2020-11-01 18:40:05 +00:00
|
|
|
};
|
|
|
|
lzma_ret ret = lzma_stream_encoder(&lstr, filters, LZMA_CHECK_CRC64);
|
|
|
|
|
2020-11-02 20:04:49 +00:00
|
|
|
if (ret != LZMA_OK)
|
|
|
|
throw Exception(
|
2020-11-11 01:50:56 +00:00
|
|
|
ErrorCodes::LZMA_STREAM_ENCODER_FAILED,
|
|
|
|
"lzma stream encoder init failed: error code: {} lzma version: {}",
|
|
|
|
ret,
|
|
|
|
LZMA_VERSION_STRING);
|
2020-11-01 18:40:05 +00:00
|
|
|
}
|
|
|
|
|
2020-11-09 22:52:22 +00:00
|
|
|
LZMADeflatingWriteBuffer::~LZMADeflatingWriteBuffer()
|
2020-11-01 18:40:05 +00:00
|
|
|
{
|
2021-11-11 17:27:23 +00:00
|
|
|
finalize();
|
2020-11-01 18:40:05 +00:00
|
|
|
}
|
|
|
|
|
2020-11-09 22:52:22 +00:00
|
|
|
void LZMADeflatingWriteBuffer::nextImpl()
|
2020-11-01 18:40:05 +00:00
|
|
|
{
|
|
|
|
if (!offset())
|
|
|
|
return;
|
|
|
|
|
|
|
|
lstr.next_in = reinterpret_cast<unsigned char *>(working_buffer.begin());
|
|
|
|
lstr.avail_in = offset();
|
|
|
|
|
Fix abnormal server termination for nested writers
Writers with nested writer can call next() from the dtor for nested
writer and this will cause exception again, so the buffer position
should be updated on exceptions.
Found by stress test (thread) here [1] and here [2]:
2021.02.27 19:27:53.498977 [ 302 ] {} <Fatal> BaseDaemon: (version 21.3.1.6130, build id: 2DAEC5DEBF03C5A1C3BF66B7779C886F16239345) (from thread 1614) Terminate called for uncaught exception:
Code: 24, e.displayText() = DB::Exception: Cannot write to ostream at offset 2097498, Stack trace (when copying this message, always include the lines below):
0. ./obj-x86_64-linux-gnu/../contrib/libcxx/include/exception:0: Poco::Exception::Exception(std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> > const&, int) @ 0x15bef2ab in /usr/bin/clickhouse
1. ./obj-x86_64-linux-gnu/../src/Common/Exception.cpp:56: DB::Exception::Exception(std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> > const&, int, bool) @ 0x8aea92e in /usr/bin/clickhouse
2. ./obj-x86_64-linux-gnu/../src/IO/WriteBufferFromOStream.cpp:0: DB::WriteBufferFromOStream::nextImpl() @ 0x8bbbc45 in /usr/bin/clickhouse
3. ./obj-x86_64-linux-gnu/../src/IO/BufferBase.h:39: DB::WriteBufferFromOStream::~WriteBufferFromOStream() @ 0x8bbc077 in /usr/bin/clickhouse
4. ./obj-x86_64-linux-gnu/../src/IO/WriteBufferFromOStream.cpp:44: DB::WriteBufferFromOStream::~WriteBufferFromOStream() @ 0x8bbc18a in /usr/bin/clickhouse
5. ./obj-x86_64-linux-gnu/../src/IO/BufferWithOwnMemory.h:137: DB::ZstdDeflatingWriteBuffer::~ZstdDeflatingWriteBuffer() @ 0x118bdc29 in /usr/bin/clickhouse
6. ./obj-x86_64-linux-gnu/../src/IO/ZstdDeflatingWriteBuffer.cpp:32: DB::ZstdDeflatingWriteBuffer::~ZstdDeflatingWriteBuffer() @ 0x118be3ea in /usr/bin/clickhouse
7. ./obj-x86_64-linux-gnu/../contrib/libcxx/include/memory:0: DB::WriteBufferFromHTTPServerResponse::finalize() @ 0x12f1dceb in /usr/bin/clickhouse
8. ./obj-x86_64-linux-gnu/../src/Server/HTTPHandler.cpp:703: DB::HTTPHandler::trySendExceptionToClient(std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> > const&, int, DB::HTTPServerRequest&, DB::HTTPServerResponse&, DB::HTTPHandler::Output&) @ 0x12e9fecc in /usr/bin/clickhouse
9. ./obj-x86_64-linux-gnu/../contrib/libcxx/include/string:1444: DB::HTTPHandler::handleRequest(DB::HTTPServerRequest&, DB::HTTPServerResponse&) @ 0x12ea0d60 in /usr/bin/clickhouse
10. ./obj-x86_64-linux-gnu/../src/Server/HTTP/HTTPServerConnection.cpp:0: DB::HTTPServerConnection::run() @ 0x12f16db1 in /usr/bin/clickhouse
11. ./obj-x86_64-linux-gnu/../contrib/poco/Net/src/TCPServerConnection.cpp:57: Poco::Net::TCPServerConnection::start() @ 0x15b184f3 in /usr/bin/clickhouse
12. ./obj-x86_64-linux-gnu/../contrib/poco/Net/src/TCPServerDispatcher.cpp:0: Poco::Net::TCPServerDispatcher::run() @ 0x15b18c1f in /usr/bin/clickhouse
13. ./obj-x86_64-linux-gnu/../contrib/poco/Foundation/include/Poco/ScopedLock.h:36: Poco::PooledThread::run() @ 0x15c7fdb2 in /usr/bin/clickhouse
14. ./obj-x86_64-linux-gnu/../contrib/poco/Foundation/src/Thread.cpp:56: Poco::(anonymous namespace)::RunnableHolder::run() @ 0x15c7e350 in /usr/bin/clickhouse
15. ./obj-x86_64-linux-gnu/../contrib/poco/Foundation/include/Poco/SharedPtr.h:277: Poco::ThreadImpl::runnableEntry(void*) @ 0x15c7cb58 in /usr/bin/clickhouse
16. __tsan_thread_start_func @ 0x8a04ced in /usr/bin/clickhouse
17. start_thread @ 0x9609 in /usr/lib/x86_64-linux-gnu/libpthread-2.31.so
18. __clone @ 0x122293 in /usr/lib/x86_64-linux-gnu/libc-2.31.so
(version 21.3.1.6130)
[1]: https://clickhouse-test-reports.s3.yandex.net/21279/4f61ef3099f42f17b496a0b0424773978d9a32dc/stress_test_(thread).html#fail1
[2]: https://clickhouse-test-reports.s3.yandex.net/21292/ae9fea1d0af118a8f87b224d194d61da1567188b/stress_test_(thread).html#fail1
v2: https://clickhouse-test-reports.s3.yandex.net/21305/e969daa6e86c5e09cfef08cfde19712982b64e59/stress_test_(thread).html#fail1
2021-02-28 07:18:49 +00:00
|
|
|
try
|
2020-11-02 13:17:25 +00:00
|
|
|
{
|
Fix abnormal server termination for nested writers
Writers with nested writer can call next() from the dtor for nested
writer and this will cause exception again, so the buffer position
should be updated on exceptions.
Found by stress test (thread) here [1] and here [2]:
2021.02.27 19:27:53.498977 [ 302 ] {} <Fatal> BaseDaemon: (version 21.3.1.6130, build id: 2DAEC5DEBF03C5A1C3BF66B7779C886F16239345) (from thread 1614) Terminate called for uncaught exception:
Code: 24, e.displayText() = DB::Exception: Cannot write to ostream at offset 2097498, Stack trace (when copying this message, always include the lines below):
0. ./obj-x86_64-linux-gnu/../contrib/libcxx/include/exception:0: Poco::Exception::Exception(std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> > const&, int) @ 0x15bef2ab in /usr/bin/clickhouse
1. ./obj-x86_64-linux-gnu/../src/Common/Exception.cpp:56: DB::Exception::Exception(std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> > const&, int, bool) @ 0x8aea92e in /usr/bin/clickhouse
2. ./obj-x86_64-linux-gnu/../src/IO/WriteBufferFromOStream.cpp:0: DB::WriteBufferFromOStream::nextImpl() @ 0x8bbbc45 in /usr/bin/clickhouse
3. ./obj-x86_64-linux-gnu/../src/IO/BufferBase.h:39: DB::WriteBufferFromOStream::~WriteBufferFromOStream() @ 0x8bbc077 in /usr/bin/clickhouse
4. ./obj-x86_64-linux-gnu/../src/IO/WriteBufferFromOStream.cpp:44: DB::WriteBufferFromOStream::~WriteBufferFromOStream() @ 0x8bbc18a in /usr/bin/clickhouse
5. ./obj-x86_64-linux-gnu/../src/IO/BufferWithOwnMemory.h:137: DB::ZstdDeflatingWriteBuffer::~ZstdDeflatingWriteBuffer() @ 0x118bdc29 in /usr/bin/clickhouse
6. ./obj-x86_64-linux-gnu/../src/IO/ZstdDeflatingWriteBuffer.cpp:32: DB::ZstdDeflatingWriteBuffer::~ZstdDeflatingWriteBuffer() @ 0x118be3ea in /usr/bin/clickhouse
7. ./obj-x86_64-linux-gnu/../contrib/libcxx/include/memory:0: DB::WriteBufferFromHTTPServerResponse::finalize() @ 0x12f1dceb in /usr/bin/clickhouse
8. ./obj-x86_64-linux-gnu/../src/Server/HTTPHandler.cpp:703: DB::HTTPHandler::trySendExceptionToClient(std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> > const&, int, DB::HTTPServerRequest&, DB::HTTPServerResponse&, DB::HTTPHandler::Output&) @ 0x12e9fecc in /usr/bin/clickhouse
9. ./obj-x86_64-linux-gnu/../contrib/libcxx/include/string:1444: DB::HTTPHandler::handleRequest(DB::HTTPServerRequest&, DB::HTTPServerResponse&) @ 0x12ea0d60 in /usr/bin/clickhouse
10. ./obj-x86_64-linux-gnu/../src/Server/HTTP/HTTPServerConnection.cpp:0: DB::HTTPServerConnection::run() @ 0x12f16db1 in /usr/bin/clickhouse
11. ./obj-x86_64-linux-gnu/../contrib/poco/Net/src/TCPServerConnection.cpp:57: Poco::Net::TCPServerConnection::start() @ 0x15b184f3 in /usr/bin/clickhouse
12. ./obj-x86_64-linux-gnu/../contrib/poco/Net/src/TCPServerDispatcher.cpp:0: Poco::Net::TCPServerDispatcher::run() @ 0x15b18c1f in /usr/bin/clickhouse
13. ./obj-x86_64-linux-gnu/../contrib/poco/Foundation/include/Poco/ScopedLock.h:36: Poco::PooledThread::run() @ 0x15c7fdb2 in /usr/bin/clickhouse
14. ./obj-x86_64-linux-gnu/../contrib/poco/Foundation/src/Thread.cpp:56: Poco::(anonymous namespace)::RunnableHolder::run() @ 0x15c7e350 in /usr/bin/clickhouse
15. ./obj-x86_64-linux-gnu/../contrib/poco/Foundation/include/Poco/SharedPtr.h:277: Poco::ThreadImpl::runnableEntry(void*) @ 0x15c7cb58 in /usr/bin/clickhouse
16. __tsan_thread_start_func @ 0x8a04ced in /usr/bin/clickhouse
17. start_thread @ 0x9609 in /usr/lib/x86_64-linux-gnu/libpthread-2.31.so
18. __clone @ 0x122293 in /usr/lib/x86_64-linux-gnu/libc-2.31.so
(version 21.3.1.6130)
[1]: https://clickhouse-test-reports.s3.yandex.net/21279/4f61ef3099f42f17b496a0b0424773978d9a32dc/stress_test_(thread).html#fail1
[2]: https://clickhouse-test-reports.s3.yandex.net/21292/ae9fea1d0af118a8f87b224d194d61da1567188b/stress_test_(thread).html#fail1
v2: https://clickhouse-test-reports.s3.yandex.net/21305/e969daa6e86c5e09cfef08cfde19712982b64e59/stress_test_(thread).html#fail1
2021-02-28 07:18:49 +00:00
|
|
|
lzma_action action = LZMA_RUN;
|
|
|
|
do
|
|
|
|
{
|
|
|
|
out->nextIfAtEnd();
|
|
|
|
lstr.next_out = reinterpret_cast<unsigned char *>(out->position());
|
|
|
|
lstr.avail_out = out->buffer().end() - out->position();
|
|
|
|
|
|
|
|
lzma_ret ret = lzma_code(&lstr, action);
|
|
|
|
out->position() = out->buffer().end() - lstr.avail_out;
|
|
|
|
|
|
|
|
if (ret == LZMA_STREAM_END)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (ret != LZMA_OK)
|
|
|
|
throw Exception(
|
|
|
|
ErrorCodes::LZMA_STREAM_ENCODER_FAILED,
|
|
|
|
"lzma stream encoding failed: error code: {}; lzma_version: {}",
|
|
|
|
ret,
|
|
|
|
LZMA_VERSION_STRING);
|
|
|
|
|
|
|
|
} while (lstr.avail_in > 0 || lstr.avail_out == 0);
|
|
|
|
}
|
|
|
|
catch (...)
|
|
|
|
{
|
|
|
|
/// Do not try to write next time after exception.
|
|
|
|
out->position() = out->buffer().begin();
|
|
|
|
throw;
|
|
|
|
}
|
2020-11-01 18:40:05 +00:00
|
|
|
}
|
|
|
|
|
2021-11-22 11:19:26 +00:00
|
|
|
void LZMADeflatingWriteBuffer::finalizeBefore()
|
Fix abnormal server termination for nested writers
Writers with nested writer can call next() from the dtor for nested
writer and this will cause exception again, so the buffer position
should be updated on exceptions.
Found by stress test (thread) here [1] and here [2]:
2021.02.27 19:27:53.498977 [ 302 ] {} <Fatal> BaseDaemon: (version 21.3.1.6130, build id: 2DAEC5DEBF03C5A1C3BF66B7779C886F16239345) (from thread 1614) Terminate called for uncaught exception:
Code: 24, e.displayText() = DB::Exception: Cannot write to ostream at offset 2097498, Stack trace (when copying this message, always include the lines below):
0. ./obj-x86_64-linux-gnu/../contrib/libcxx/include/exception:0: Poco::Exception::Exception(std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> > const&, int) @ 0x15bef2ab in /usr/bin/clickhouse
1. ./obj-x86_64-linux-gnu/../src/Common/Exception.cpp:56: DB::Exception::Exception(std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> > const&, int, bool) @ 0x8aea92e in /usr/bin/clickhouse
2. ./obj-x86_64-linux-gnu/../src/IO/WriteBufferFromOStream.cpp:0: DB::WriteBufferFromOStream::nextImpl() @ 0x8bbbc45 in /usr/bin/clickhouse
3. ./obj-x86_64-linux-gnu/../src/IO/BufferBase.h:39: DB::WriteBufferFromOStream::~WriteBufferFromOStream() @ 0x8bbc077 in /usr/bin/clickhouse
4. ./obj-x86_64-linux-gnu/../src/IO/WriteBufferFromOStream.cpp:44: DB::WriteBufferFromOStream::~WriteBufferFromOStream() @ 0x8bbc18a in /usr/bin/clickhouse
5. ./obj-x86_64-linux-gnu/../src/IO/BufferWithOwnMemory.h:137: DB::ZstdDeflatingWriteBuffer::~ZstdDeflatingWriteBuffer() @ 0x118bdc29 in /usr/bin/clickhouse
6. ./obj-x86_64-linux-gnu/../src/IO/ZstdDeflatingWriteBuffer.cpp:32: DB::ZstdDeflatingWriteBuffer::~ZstdDeflatingWriteBuffer() @ 0x118be3ea in /usr/bin/clickhouse
7. ./obj-x86_64-linux-gnu/../contrib/libcxx/include/memory:0: DB::WriteBufferFromHTTPServerResponse::finalize() @ 0x12f1dceb in /usr/bin/clickhouse
8. ./obj-x86_64-linux-gnu/../src/Server/HTTPHandler.cpp:703: DB::HTTPHandler::trySendExceptionToClient(std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> > const&, int, DB::HTTPServerRequest&, DB::HTTPServerResponse&, DB::HTTPHandler::Output&) @ 0x12e9fecc in /usr/bin/clickhouse
9. ./obj-x86_64-linux-gnu/../contrib/libcxx/include/string:1444: DB::HTTPHandler::handleRequest(DB::HTTPServerRequest&, DB::HTTPServerResponse&) @ 0x12ea0d60 in /usr/bin/clickhouse
10. ./obj-x86_64-linux-gnu/../src/Server/HTTP/HTTPServerConnection.cpp:0: DB::HTTPServerConnection::run() @ 0x12f16db1 in /usr/bin/clickhouse
11. ./obj-x86_64-linux-gnu/../contrib/poco/Net/src/TCPServerConnection.cpp:57: Poco::Net::TCPServerConnection::start() @ 0x15b184f3 in /usr/bin/clickhouse
12. ./obj-x86_64-linux-gnu/../contrib/poco/Net/src/TCPServerDispatcher.cpp:0: Poco::Net::TCPServerDispatcher::run() @ 0x15b18c1f in /usr/bin/clickhouse
13. ./obj-x86_64-linux-gnu/../contrib/poco/Foundation/include/Poco/ScopedLock.h:36: Poco::PooledThread::run() @ 0x15c7fdb2 in /usr/bin/clickhouse
14. ./obj-x86_64-linux-gnu/../contrib/poco/Foundation/src/Thread.cpp:56: Poco::(anonymous namespace)::RunnableHolder::run() @ 0x15c7e350 in /usr/bin/clickhouse
15. ./obj-x86_64-linux-gnu/../contrib/poco/Foundation/include/Poco/SharedPtr.h:277: Poco::ThreadImpl::runnableEntry(void*) @ 0x15c7cb58 in /usr/bin/clickhouse
16. __tsan_thread_start_func @ 0x8a04ced in /usr/bin/clickhouse
17. start_thread @ 0x9609 in /usr/lib/x86_64-linux-gnu/libpthread-2.31.so
18. __clone @ 0x122293 in /usr/lib/x86_64-linux-gnu/libc-2.31.so
(version 21.3.1.6130)
[1]: https://clickhouse-test-reports.s3.yandex.net/21279/4f61ef3099f42f17b496a0b0424773978d9a32dc/stress_test_(thread).html#fail1
[2]: https://clickhouse-test-reports.s3.yandex.net/21292/ae9fea1d0af118a8f87b224d194d61da1567188b/stress_test_(thread).html#fail1
v2: https://clickhouse-test-reports.s3.yandex.net/21305/e969daa6e86c5e09cfef08cfde19712982b64e59/stress_test_(thread).html#fail1
2021-02-28 07:18:49 +00:00
|
|
|
{
|
2020-11-01 18:40:05 +00:00
|
|
|
next();
|
|
|
|
|
2020-11-02 13:17:25 +00:00
|
|
|
do
|
|
|
|
{
|
2020-11-01 18:40:05 +00:00
|
|
|
out->nextIfAtEnd();
|
|
|
|
lstr.next_out = reinterpret_cast<unsigned char *>(out->position());
|
|
|
|
lstr.avail_out = out->buffer().end() - out->position();
|
|
|
|
|
|
|
|
lzma_ret ret = lzma_code(&lstr, LZMA_FINISH);
|
|
|
|
out->position() = out->buffer().end() - lstr.avail_out;
|
|
|
|
|
2020-11-02 20:04:49 +00:00
|
|
|
if (ret == LZMA_STREAM_END)
|
2020-11-02 13:17:25 +00:00
|
|
|
{
|
2020-11-02 20:04:49 +00:00
|
|
|
return;
|
2020-11-01 23:52:34 +00:00
|
|
|
}
|
2020-11-02 20:04:49 +00:00
|
|
|
|
2020-11-01 18:40:05 +00:00
|
|
|
if (ret != LZMA_OK)
|
2020-11-02 20:04:49 +00:00
|
|
|
throw Exception(
|
2020-11-11 01:50:56 +00:00
|
|
|
ErrorCodes::LZMA_STREAM_ENCODER_FAILED,
|
|
|
|
"lzma stream encoding failed: error code: {}; lzma version: {}",
|
|
|
|
ret,
|
|
|
|
LZMA_VERSION_STRING);
|
2020-11-02 20:04:49 +00:00
|
|
|
|
2020-11-01 18:40:05 +00:00
|
|
|
} while (lstr.avail_out == 0);
|
|
|
|
}
|
2021-11-10 22:58:56 +00:00
|
|
|
|
2021-11-22 11:19:26 +00:00
|
|
|
void LZMADeflatingWriteBuffer::finalizeAfter()
|
2021-11-10 22:58:56 +00:00
|
|
|
{
|
|
|
|
lzma_end(&lstr);
|
|
|
|
}
|
|
|
|
|
2020-11-02 13:17:25 +00:00
|
|
|
}
|
2020-11-12 09:21:33 +00:00
|
|
|
|