Byte Array Serialization
bas.hpp
Go to the documentation of this file.
1 /*
2 ** Byte Array Serialization
3 ** File description:
4 ** Header only lib which purpose is to be able to easily serialize data/classes
5 ** to send them over networking protocol and unserialize them easily afterward.
6 ** Author:
7 ** Nell Fauveau
8 ** https://github.com/Nellousan/ByteArraySerialisation
9 */
10 
11 /* Current Changelogs:
12  - BugFix: pushData specialization on the SerializedObject
13  - Changed the way serialization works, now the makeSerialization function
14  needs override, serialization() is still the way of retrieving the SerializedObject,
15  same goes for makeUnserialization and unserialization()
16  - SerializedObject doesn't need the size to construct from char * anymore. A Checksum has been added
17  to make this possible, and to add possibilities in networking (knowing if multiple objects have been received and such).
18  - Moved bodies of pushData(), popData() and popDataArray() to separate function to
19  avoid code duplication and make the addition of specialised function template easier.
20  - Added documentation of the library, generated by doxygen.
21 */
22 
23 #ifndef BAS_HPP_
24 #define BAS_HPP_
25 
26 #include <cstring>
27 #include <memory>
28 #include <tuple>
29 #include <vector>
30 
31 #include <iostream>
32 
34 
35 #define _BAS_SIZE_BYTES_ 2 // Number of bytes used to store var size and array size in the payload,
36 #define _BAS_ARRAY_SIZE_ 2 // MAX is your OS size_t byte size (4 is max recommended as it will be compatible in most cases)
37 #define _BAS_CHECKSUM_SIZE_ 4 // If this lib is used for networking, make sure that these 3 numbers are the same on both ends
38 
40 
41 namespace bas {
42 
43 template <typename T>
44 class Helper;
45 
46 template <typename T>
48 
55 public:
57  {
58  prepareChecksum();
59  }
60 
67  inline SerializedObject(const char* data)
68  {
69  constructFromPayload(data);
70  }
71 
72 
77  inline SerializedObject(const SerializedObject& other)
78  : _data(other._data)
79  , _isChecksumRemoved(other._isChecksumRemoved)
80  {
81  }
82 
86  ~SerializedObject() = default;
87 
96  template <typename T>
97  inline void pushData(const T& data)
98  {
99  Helper<T> helper;
100  helper.pushData(*this, data);
101  }
102 
112  template <typename T>
113  inline void pushData(const T* data, size_t array_size)
114  {
115  const char* data_ = (const char*)data;
116  size_t size = sizeof(T);
117 
118  pushSizes(size, array_size);
119 
120  pushRawData(size, array_size, data_);
121 
122  checksumUpdate();
123  }
124 
130  template <typename T>
131  inline T popData(void)
132  {
133  Helper<T> helper;
134  return helper.popData(*this);
135  }
136 
143  template <typename T>
145  {
146  size_t size = 0;
147  size_t array_size = 0;
148  T* var;
149 
150  std::tie(size, array_size) = getSizes();
151 
152  var = new T[array_size];
153 
154  copyMem(size, array_size, var);
155 
156  return PoppedArray<T>(array_size, var);
157  }
158 
165  inline const char* payload(void) const
166  {
167  return _data.data();
168  }
169 
176  inline std::vector<char>& vector(void)
177  {
178  return _data;
179  }
180 
187  inline size_t size(void) const
188  {
189  return _data.size();
190  }
191 
195  inline void clear()
196  {
197  _data.clear();
198  _isChecksumRemoved = false;
199  }
200 
206  {
207  _data = other._data;
208  _isChecksumRemoved = other._isChecksumRemoved;
209  return *this;
210  }
211 
220  inline SerializedObject& operator=(const char* data)
221  {
222  constructFromPayload(data);
223  return *this;
224  }
225 
232  inline void removeChecksum(void)
233  {
234  if (_isChecksumRemoved)
235  return;
236  _isChecksumRemoved = true;
237  for (size_t i = 0; i < _BAS_CHECKSUM_SIZE_; i++)
238  _data.erase(_data.begin());
239  }
240 
247  inline void addChecksum(void)
248  {
249  if (!_isChecksumRemoved)
250  return;
251  for (size_t i = 0; i < _BAS_CHECKSUM_SIZE_; i++)
252  _data.insert(_data.begin(), 0);
253  checksumUpdate();
254  }
255 
256 private:
257  inline void prepareChecksum(void)
258  {
259  for (size_t i = 0; i < _BAS_CHECKSUM_SIZE_; i++)
260  _data.push_back(0);
261  }
262 
263  inline void constructFromPayload(const char* data)
264  {
265  size_t size = 0;
266  for (size_t i = 0; i < _BAS_CHECKSUM_SIZE_; i++) {
267  size |= (data[i] << (i * 8));
268  }
269  _data.assign(data, data + size + _BAS_CHECKSUM_SIZE_);
270  }
271 
272  inline void checksumUpdate(void)
273  {
274  size_t size = _data.size();
275 
276  if (_isChecksumRemoved)
277  addChecksum();
278 
279  for (size_t i = 0; i < _BAS_CHECKSUM_SIZE_; i++) {
280  _data[i] = (size >> (i * 8));
281  }
282  }
283 
284  inline void pushRawData(size_t size, size_t array_size, const char* data)
285  {
286  for (size_t j = 0; j < array_size; j++) {
287  for (size_t i = 0; i < size; i++) {
288  _data.push_back(data[j * size + i]);
289  }
290  }
291  }
292 
293  inline void pushSizes(size_t size, size_t array_size)
294  {
295  for (size_t i = 0; i < _BAS_SIZE_BYTES_; i++)
296  _data.push_back((size >> (i * 8)) & 0xFF);
297 
298  for (size_t i = 0; i < _BAS_ARRAY_SIZE_; i++)
299  _data.push_back((array_size >> (i * 8)) & 0xFF);
300  }
301 
302  inline std::tuple<int, int> getSizes(void)
303  {
304  size_t size = 0, array_size = 0;
305 
306  if (!_isChecksumRemoved) {
307  removeChecksum();
308  }
309 
310  for (size_t i = 0; i < _BAS_SIZE_BYTES_; i++) {
311  size |= (_data.front() << (i * 8));
312  _data.erase(_data.begin());
313  }
314 
315  for (size_t i = 0; i < _BAS_ARRAY_SIZE_; i++) {
316  array_size |= (_data.front() << (i * 8));
317  _data.erase(_data.begin());
318  }
319  return std::make_tuple(size, array_size);
320  }
321 
322  template <typename T>
323  inline void copyMem(size_t size, size_t array_size, T* var)
324  {
325  for (size_t j = 0; j < array_size; j++) {
326  std::memcpy(var + j, _data.data(), size);
327  for (size_t i = 0; i < size; i++) {
328  _data.erase(_data.begin());
329  }
330  }
331  }
332 
333  std::vector<char> _data;
334  bool _isChecksumRemoved = false;
335 
336  template <typename T>
337  friend class Helper;
338  friend class Helper<std::string>;
339 };
340 
348 public:
349 
353  virtual ~Serializable() = default;
354 
363  virtual void makeSerialization(SerializedObject& obj) = 0;
364 
373  virtual void makeUnserialization(SerializedObject& obj) = 0;
374 
383  inline void unserialize(SerializedObject obj)
384  {
385  makeUnserialization(obj);
386  }
387 
396  {
397  SerializedObject obj;
398  makeSerialization(obj);
399  return obj;
400  }
401 
402 private:
403 };
404 
411 template <typename T>
412 class PoppedArray {
413 
414 public:
416  PoppedArray() = default;
417 
418  inline PoppedArray(size_t size, T* ptr)
419  : _size(size)
420  , _ptr(ptr)
421  {
422  }
424 
431  inline size_t size(void)
432  {
433  return _size;
434  }
435 
443  inline T* get(void)
444  {
445  return _ptr.get();
446  }
447 
448 private:
449  size_t _size;
450  std::shared_ptr<T> _ptr;
451 };
452 
453 
459 template <typename T>
460 class Helper {
461 public:
462  inline void pushData(SerializedObject& obj, const T& data_)
463  {
464  const char* data = (const char*)&data_;
465  size_t size = sizeof(T);
466  size_t array_size = 1;
467 
468  obj.pushSizes(size, array_size);
469 
470  obj.pushRawData(size, array_size, data);
471 
472  obj.checksumUpdate();
473  }
474 
475  inline T popData(SerializedObject& obj)
476  {
477  size_t size = 0;
478  size_t array_size = 0;
479  T var;
480 
481  std::tie(size, array_size) = obj.getSizes();
482 
483  obj.copyMem(size, array_size, &var);
484 
485  return var;
486  }
487 
488 };
490 template <>
491 class Helper<std::string> {
492 public:
493  inline void pushData(SerializedObject& obj, const std::string& data_)
494  {
495  const char* data = data_.c_str();
496  size_t size = sizeof(char);
497  size_t array_size = data_.size() + 1;
498 
499  obj.pushSizes(size, array_size);
500 
501  obj.pushRawData(size, array_size, data);
502 
503  obj.checksumUpdate();
504  }
505 
506  inline std::string popData(SerializedObject &obj)
507  {
508  size_t size = 0;
509  size_t array_size = 0;
510  std::string str;
511  char* var;
512 
513  std::tie(size, array_size) = obj.getSizes();
514 
515  var = new char[array_size];
516 
517  obj.copyMem(size, array_size, var);
518 
519  str = var;
520 
521  delete var;
522 
523  return str;
524  }
525 
526 };
527 
528 template <typename T>
529 class Helper<std::vector<T>> {
530 public:
531  inline void pushData(SerializedObject& obj, const std::vector<T>& data_)
532  {
533  const char* data = (char*)data_.data();
534  size_t size = sizeof(T);
535  size_t array_size = data_.size();
536 
537  obj.pushSizes(size, array_size);
538 
539  obj.pushRawData(size, array_size, data);
540 
541  obj.checksumUpdate();
542  }
543 
544  inline std::vector<T> popData(SerializedObject &obj)
545  {
546  size_t size = 0;
547  size_t array_size = 0;
548  std::vector<T> vector;
549  T* var;
550 
551  std::tie(size, array_size) = obj.getSizes();
552 
553  var = new T[array_size];
554 
555  obj.copyMem(size, array_size, var);
556 
557  vector.assign(var, var + array_size);
558 
559  delete var;
560 
561  return vector;
562  }
563 
564 };
565 
566 template <>
567 class Helper<SerializedObject> {
568 public:
569  inline void pushData(SerializedObject& obj, SerializedObject& data_)
570  {
571  const char* data = (char*)data_.payload();
572  size_t size = sizeof(char);
573  size_t array_size = data_.size();
574 
575  obj.pushSizes(size, array_size);
576 
577  obj.pushRawData(size, array_size, data);
578 
579  obj.checksumUpdate();
580  }
581 
583  {
584  size_t size = 0;
585  size_t array_size = 0;
586  SerializedObject obj;
587  char* var;
588 
589  std::tie(size, array_size) = obj.getSizes();
590 
591  var = new char[array_size];
592 
593  obj.copyMem(size, array_size, var);
594 
595  obj = var;
596 
597  delete var;
598 
599  return obj;
600  }
601 
602 };
604 }
605 
686 #endif /* !BAS_HPP_ */
687 
void removeChecksum(void)
Remove Checksum at the beggining of payload.
Definition: bas.hpp:232
SerializedObject serialize(void)
Serialize the class.
Definition: bas.hpp:395
Definition: bas.hpp:41
T * get(void)
Returns the pointer to the array popped by popDataArray<>().
Definition: bas.hpp:443
SerializedObject & operator=(const SerializedObject &other)
Assign values to the SerializedObject.
Definition: bas.hpp:205
SerializedObject()
Definition: bas.hpp:56
size_t size(void) const
Returns the size of the payload in Bytes.
Definition: bas.hpp:187
SerializedObject(const char *data)
Construct object from payload.
Definition: bas.hpp:67
#define _BAS_ARRAY_SIZE_
Definition: bas.hpp:36
void clear()
Clear the SerializedObject and resets it to its default values.
Definition: bas.hpp:195
~SerializedObject()=default
Default destructor.
The PoppedArray is the return value of SerializedObject::popDataArray<>().
Definition: bas.hpp:47
The Serializable class, when herited from, allows you to serialize your own classes.
Definition: bas.hpp:347
const char * payload(void) const
Returns a pointer to the payload of the object.
Definition: bas.hpp:165
void addChecksum(void)
Add Checksum at the beggining of payload.
Definition: bas.hpp:247
#define _BAS_CHECKSUM_SIZE_
Definition: bas.hpp:37
void unserialize(SerializedObject obj)
Reconstruct object from a SerializedObject.
Definition: bas.hpp:383
void pushData(const T *data, size_t array_size)
Pushes an array of data into the object payload.
Definition: bas.hpp:113
size_t size(void)
Returns the size of the array popped by popDataArray<>().
Definition: bas.hpp:431
T popData(void)
Pop the next data in the payload.
Definition: bas.hpp:131
SerializedObject(const SerializedObject &other)
Copy-construct object from another SerializedObject.
Definition: bas.hpp:77
std::vector< char > & vector(void)
Returns a reference to the std::vector containing the payload.
Definition: bas.hpp:176
void pushData(const T &data)
Pushes data into the payload.
Definition: bas.hpp:97
The SerializedObject contains and manages the payload of your serializations.
Definition: bas.hpp:54
SerializedObject & operator=(const char *data)
Assign a copy of data to the payload.
Definition: bas.hpp:220
PoppedArray< T > popDataArray(void)
Pop the next array in the payload and returns a PoppedArray.
Definition: bas.hpp:144
#define _BAS_SIZE_BYTES_
Definition: bas.hpp:35