21#include "cc_tools_qt/ToolsExtraInfoMessage.h"
22#include "cc_tools_qt/ToolsInvalidMessage.h"
23#include "cc_tools_qt/ToolsRawDataMessage.h"
24#include "cc_tools_qt/property/message.h"
26#include <QtCore/QJsonObject>
27#include <QtCore/QJsonDocument>
28#include <QtCore/QByteArray>
38template <
typename TMsgBase,
class TProtFrame,
typename TMsgFactory,
typename TTransportMsg>
39class ToolsFrameBase :
public ToolsFrame
41 using Base = ToolsFrame;
43 using ProtInterface =
typename TMsgBase::ProtInterface;
44 using DataSeq =
typename Base::DataSeq;
46 using TransportMsg = TTransportMsg;
49 using InvalidMsg = ToolsInvalidMessage<TMsgBase>;
51 using RawDataMsg = ToolsRawDataMessage<TMsgBase>;
54 using ExtraInfoMsg = ToolsExtraInfoMessage<TMsgBase>;
56 using ProtFrame = TProtFrame;
58 ToolsFrameBase() =
default;
61 virtual ToolsMessagesList readDataImpl(
const ToolsDataInfo& dataInfo,
bool final)
override
63 m_inData.reserve(m_inData.size() + dataInfo.m_data.size());
64 m_inData.insert(m_inData.end(), dataInfo.m_data.begin(), dataInfo.m_data.end());
66 ToolsMessagesList allMsgs;
67 std::size_t consumed = 0U;
69 auto checkGarbageFunc =
72 if (m_garbage.empty()) {
77 updateRawDataInternal(m_garbage, *invalidMsgPtr);
78 allMsgs.push_back(std::move(invalidMsgPtr));
83 using ProtMsgPtr =
typename ProtFrame::MsgPtr;
84 using ReadIter =
typename ProtInterface::ReadIterator;
85 while (consumed < m_inData.size()) {
88 ReadIter readIterBeg = m_inData.data() + consumed;
89 ReadIter readIter = readIterBeg;
90 auto remLen = m_inData.size() - consumed;
100 comms::protocol::msgId(msgId),
101 comms::protocol::msgIndex(idx));
103 if (es == comms::ErrorStatus::NotEnoughData) {
107 if (es == comms::ErrorStatus::MsgAllocFailure) {
108 [[maybe_unused]]
static constexpr bool Must_not_be_happen =
false;
109 assert(Must_not_be_happen);
113 if (es != comms::ErrorStatus::Success) {
114 m_garbage.push_back(*readIterBeg);
115 static constexpr std::size_t GarbageLimit = 512;
116 if (GarbageLimit <= m_garbage.size()) {
126 auto diff =
static_cast<std::size_t
>(std::distance(readIterBeg, readIter));
129 auto toolsMsg = m_factory.createMessage(msgId,
static_cast<unsigned>(idx));
131 [[maybe_unused]]
static constexpr bool Protocol_and_Tools_Frames_Out_of_Sync =
false;
132 assert(Protocol_and_Tools_Frames_Out_of_Sync);
134 m_garbage.reserve(m_garbage.size() + diff);
135 m_garbage.insert(m_garbage.end(), readIterBeg, readIter);
140 toolsMsg->assignProtMessage(msgPtr.get());
142 DataSeq data(readIterBeg, readIter);
143 updateTransportInternal(data, *toolsMsg);
144 updateRawDataInternal(data, *toolsMsg);
145 allMsgs.push_back(std::move(toolsMsg));
148 static_cast<void>(
final);
150 assert(consumed <= m_inData.size());
151 m_inData.erase(m_inData.begin(), m_inData.begin() + consumed);
153 if (
final && (!m_inData.empty())) {
154 m_garbage.reserve(m_garbage.size() + m_inData.size());
155 m_garbage.insert(m_garbage.end(), m_inData.begin(), m_inData.end());
160 if (!dataInfo.m_extraProperties.isEmpty()) {
161 auto jsonObj = QJsonObject::fromVariantMap(dataInfo.m_extraProperties);
162 QJsonDocument doc(jsonObj);
163 auto jsonData = doc.toJson();
164 DataSeq jsonRawBytes(jsonData.begin(), jsonData.end());
166 for (
auto& m : allMsgs) {
167 property::message::ToolsMsgExtraInfo().setTo(dataInfo.m_extraProperties, *m);
168 updateExtraInfoInternal(jsonRawBytes, *m);
175 virtual void updateMessageImpl(ToolsMessage& msg)
override
177 auto data = msg.encodeFramed(*
this);
178 updateTransportInternal(data, msg);
179 updateRawDataInternal(data, msg);
181 auto extraProps = property::message::ToolsMsgExtraInfo().getFrom(msg);
182 bool extraInfoMsgIsForced = property::message::ToolsMsgForceExtraInfoExistence().getFrom(msg);
183 if (extraProps.isEmpty() && (!extraInfoMsgIsForced)) {
184 property::message::ToolsMsgExtraInfoMsg().setTo(
ToolsMessagePtr(), msg);
188 auto extraInfoMsgPtr = std::make_unique<ExtraInfoMsg>();
189 if (extraProps.isEmpty()) {
190 property::message::ToolsMsgExtraInfoMsg().setTo(
ToolsMessagePtr(extraInfoMsgPtr.release()), msg);
194 auto jsonObj = QJsonObject::fromVariantMap(extraProps);
195 QJsonDocument doc(jsonObj);
196 auto jsonData = doc.toJson();
197 DataSeq jsonRawBytes(jsonData.begin(), jsonData.end());
198 updateExtraInfoInternal(jsonRawBytes, msg);
216 virtual ToolsMessagesList createAllMessagesImpl()
override
218 return m_factory.createAllMessages();
221 virtual ToolsMessagePtr createMessageImpl(
const QString& idAsString,
unsigned idx)
override
223 return m_factory.createMessage(idAsString, idx);
226 virtual DataSeq writeProtMsgImpl(
const void* protInterface)
override
228 assert(protInterface !=
nullptr);
229 auto& msg = *(
reinterpret_cast<const ProtInterface*
>(protInterface));
231 data.reserve(m_frame.length(msg));
233 auto writeIter = std::back_inserter(data);
234 auto es = m_frame.write(msg, writeIter, data.max_size());
235 if (es == comms::ErrorStatus::UpdateRequired) {
236 auto updateIter = data.data();
237 es = m_frame.update(msg, updateIter, data.size());
240 if (es != comms::ErrorStatus::Success) {
241 [[maybe_unused]]
static constexpr bool Unexpected_write_update_failure =
false;
242 assert(Unexpected_write_update_failure);
250 void updateTransportInternal(
const DataSeq& data, ToolsMessage& msg)
253 if (!transportMsg->decodeData(data)) {
254 std::cerr <<
"ERROR: Failed to decode transport message: " << std::hex;
255 std::copy(data.begin(), data.end(), std::ostream_iterator<unsigned>(std::cerr,
" "));
256 std::cerr << std::dec << std::endl;
258 [[maybe_unused]]
static constexpr bool Must_not_be_happen =
false;
259 assert(Must_not_be_happen);
262 property::message::ToolsMsgTransportMsg().setTo(std::move(transportMsg), msg);
265 void updateRawDataInternal(
const DataSeq& data, ToolsMessage& msg)
268 if (!rawDataMsg->decodeData(data)) {
269 std::cerr <<
"ERROR: Failed to decode raw data message: " << std::hex;
270 std::copy(data.begin(), data.end(), std::ostream_iterator<unsigned>(std::cerr,
" "));
271 std::cerr << std::dec << std::endl;
273 [[maybe_unused]]
static constexpr bool Must_not_be_happen =
false;
274 assert(Must_not_be_happen);
277 property::message::ToolsMsgRawDataMsg().setTo(std::move(rawDataMsg), msg);
280 void updateExtraInfoInternal(
const DataSeq& jsonRawBytes, ToolsMessage& msg)
283 if (!extraInfoMsg->decodeData(jsonRawBytes)) {
284 std::cerr <<
"ERROR: Failed to decode extra info:\n";
285 std::copy(jsonRawBytes.begin(), jsonRawBytes.end(), std::ostream_iterator<char>(std::cerr,
" "));
286 std::cerr << std::endl;
288 [[maybe_unused]]
static constexpr bool Must_not_be_happen =
false;
289 assert(Must_not_be_happen);
292 property::message::ToolsMsgExtraInfoMsg().setTo(std::move(extraInfoMsg), msg);
296 TMsgFactory m_factory;