cc_tools_qt
Common Environment for Protocol Analysis.
Loading...
Searching...
No Matches
ToolsFrameBase.h
1//
2// Copyright 2024 - 2025 (C). Alex Robenko. All rights reserved.
3//
4
5// This file is free software: you can redistribute it and/or modify
6// it under the terms of the GNU General Public License as published by
7// the Free Software Foundation, either version 3 of the License, or
8// (at your option) any later version.
9//
10// This program is distributed in the hope that it will be useful,
11// but WITHOUT ANY WARRANTY; without even the implied warranty of
12// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13// GNU General Public License for more details.
14//
15// You should have received a copy of the GNU General Public License
16// along with this program. If not, see <http://www.gnu.org/licenses/>.
17
18
19#pragma once
20
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"
25
26#include <QtCore/QJsonObject>
27#include <QtCore/QJsonDocument>
28#include <QtCore/QByteArray>
29
30#include <algorithm>
31#include <cassert>
32#include <iterator>
33#include <iostream>
34
35namespace cc_tools_qt
36{
37
38template <typename TMsgBase, class TProtFrame, typename TMsgFactory, typename TTransportMsg>
39class ToolsFrameBase : public ToolsFrame
40{
41 using Base = ToolsFrame;
42public:
43 using ProtInterface = typename TMsgBase::ProtInterface;
44 using DataSeq = typename Base::DataSeq;
45
46 using TransportMsg = TTransportMsg;
47
49 using InvalidMsg = ToolsInvalidMessage<TMsgBase>;
50
51 using RawDataMsg = ToolsRawDataMessage<TMsgBase>;
52
54 using ExtraInfoMsg = ToolsExtraInfoMessage<TMsgBase>;
55
56 using ProtFrame = TProtFrame;
57
58 ToolsFrameBase() = default;
59
60protected:
61 virtual ToolsMessagesList readDataImpl(const ToolsDataInfo& dataInfo, bool final) override
62 {
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());
65
66 ToolsMessagesList allMsgs;
67 std::size_t consumed = 0U;
68
69 auto checkGarbageFunc =
70 [this, &allMsgs]()
71 {
72 if (m_garbage.empty()) {
73 return;
74 }
75
76 ToolsMessagePtr invalidMsgPtr(new InvalidMsg);
77 updateRawDataInternal(m_garbage, *invalidMsgPtr);
78 allMsgs.push_back(std::move(invalidMsgPtr));
79 m_garbage.clear();
80 };
81
82
83 using ProtMsgPtr = typename ProtFrame::MsgPtr;
84 using ReadIter = typename ProtInterface::ReadIterator;
85 while (consumed < m_inData.size()) {
86 ProtMsgPtr msgPtr;
87
88 ReadIter readIterBeg = m_inData.data() + consumed;
89 ReadIter readIter = readIterBeg;
90 auto remLen = m_inData.size() - consumed;
91 assert(0U < remLen);
92
93 qlonglong msgId = 0;
94 std::size_t idx = 0;
95 auto es =
96 m_frame.read(
97 msgPtr,
98 readIter,
99 remLen,
100 comms::protocol::msgId(msgId),
101 comms::protocol::msgIndex(idx));
102
103 if (es == comms::ErrorStatus::NotEnoughData) {
104 break;
105 }
106
107 if (es == comms::ErrorStatus::MsgAllocFailure) {
108 [[maybe_unused]] static constexpr bool Must_not_be_happen = false;
109 assert(Must_not_be_happen);
110 break;
111 }
112
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()) {
117 checkGarbageFunc();
118 }
119
120 ++consumed;
121 continue;
122 }
123
124 checkGarbageFunc();
125 assert(msgPtr);
126 auto diff = static_cast<std::size_t>(std::distance(readIterBeg, readIter));
127 consumed += diff;
128
129 auto toolsMsg = m_factory.createMessage(msgId, static_cast<unsigned>(idx));
130 if (!toolsMsg) {
131 [[maybe_unused]] static constexpr bool Protocol_and_Tools_Frames_Out_of_Sync = false;
132 assert(Protocol_and_Tools_Frames_Out_of_Sync);
133
134 m_garbage.reserve(m_garbage.size() + diff);
135 m_garbage.insert(m_garbage.end(), readIterBeg, readIter);
136 checkGarbageFunc();
137 continue;
138 }
139
140 toolsMsg->assignProtMessage(msgPtr.get());
141
142 DataSeq data(readIterBeg, readIter);
143 updateTransportInternal(data, *toolsMsg);
144 updateRawDataInternal(data, *toolsMsg);
145 allMsgs.push_back(std::move(toolsMsg));
146 }
147
148 static_cast<void>(final);
149
150 assert(consumed <= m_inData.size());
151 m_inData.erase(m_inData.begin(), m_inData.begin() + consumed);
152
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());
156 m_inData.clear();
157 checkGarbageFunc();
158 }
159
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());
165
166 for (auto& m : allMsgs) {
167 property::message::ToolsMsgExtraInfo().setTo(dataInfo.m_extraProperties, *m);
168 updateExtraInfoInternal(jsonRawBytes, *m);
169 }
170 }
171
172 return allMsgs;
173 }
174
175 virtual void updateMessageImpl(ToolsMessage& msg) override
176 {
177 auto data = msg.encodeFramed(*this);
178 updateTransportInternal(data, msg);
179 updateRawDataInternal(data, msg);
180
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);
185 return;
186 }
187
188 auto extraInfoMsgPtr = std::make_unique<ExtraInfoMsg>();
189 if (extraProps.isEmpty()) {
190 property::message::ToolsMsgExtraInfoMsg().setTo(ToolsMessagePtr(extraInfoMsgPtr.release()), msg);
191 return;
192 }
193
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);
199 }
200
201 virtual ToolsMessagePtr createInvalidMessageImpl() override
202 {
203 return ToolsMessagePtr(new InvalidMsg());
204 }
205
206 virtual ToolsMessagePtr createRawDataMessageImpl() override
207 {
208 return ToolsMessagePtr(new RawDataMsg());
209 }
210
211 virtual ToolsMessagePtr createExtraInfoMessageImpl() override
212 {
213 return ToolsMessagePtr(new ExtraInfoMsg());
214 }
215
216 virtual ToolsMessagesList createAllMessagesImpl() override
217 {
218 return m_factory.createAllMessages();
219 }
220
221 virtual ToolsMessagePtr createMessageImpl(const QString& idAsString, unsigned idx) override
222 {
223 return m_factory.createMessage(idAsString, idx);
224 }
225
226 virtual DataSeq writeProtMsgImpl(const void* protInterface) override
227 {
228 assert(protInterface != nullptr);
229 auto& msg = *(reinterpret_cast<const ProtInterface*>(protInterface));
230 DataSeq data;
231 data.reserve(m_frame.length(msg));
232
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());
238 }
239
240 if (es != comms::ErrorStatus::Success) {
241 [[maybe_unused]] static constexpr bool Unexpected_write_update_failure = false;
242 assert(Unexpected_write_update_failure);
243 data.clear();
244 }
245
246 return data;
247 }
248
249private:
250 void updateTransportInternal(const DataSeq& data, ToolsMessage& msg)
251 {
252 ToolsMessagePtr transportMsg(new TransportMsg);
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;
257
258 [[maybe_unused]] static constexpr bool Must_not_be_happen = false;
259 assert(Must_not_be_happen);
260 }
261
262 property::message::ToolsMsgTransportMsg().setTo(std::move(transportMsg), msg);
263 }
264
265 void updateRawDataInternal(const DataSeq& data, ToolsMessage& msg)
266 {
267 ToolsMessagePtr rawDataMsg(new RawDataMsg);
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;
272
273 [[maybe_unused]] static constexpr bool Must_not_be_happen = false;
274 assert(Must_not_be_happen);
275 }
276
277 property::message::ToolsMsgRawDataMsg().setTo(std::move(rawDataMsg), msg);
278 }
279
280 void updateExtraInfoInternal(const DataSeq& jsonRawBytes, ToolsMessage& msg)
281 {
282 ToolsMessagePtr extraInfoMsg(new ExtraInfoMsg);
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;
287
288 [[maybe_unused]] static constexpr bool Must_not_be_happen = false;
289 assert(Must_not_be_happen);
290 }
291
292 property::message::ToolsMsgExtraInfoMsg().setTo(std::move(extraInfoMsg), msg);
293 }
294
295 ProtFrame m_frame;
296 TMsgFactory m_factory;
297 DataSeq m_inData;
298 DataSeq m_garbage;
299};
300
301} // namespace cc_tools_qt
302
303
Main namespace for all classes / functions of the shared library.
std::shared_ptr< ToolsMessage > ToolsMessagePtr
Smart pointer to ToolsMessage.
Definition ToolsMessage.h:163