/*! \file pichunkstream.h * \ingroup Serialization * \~\brief * \~english Binary markup de/serializator stream * \~russian Бинарный поток для де/сериализации с разметкой */ /* PIP - Platform Independent Primitives Binary markup serializator Ivan Pelipenko peri4ko@yandex.ru This program is free software: you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with this program. If not, see . */ #ifndef PICHUNKSTREAM_H #define PICHUNKSTREAM_H #include "pibytearray.h" //! \ingroup Serialization //! \~\brief //! \~english Class for binary de/serialization. //! \~russian Класс для бинарной де/сериализации. class PIP_EXPORT PIChunkStream { public: //! \~english //! Version of data packing. Read-access %PIChunkStream automatic detect version, but write-access //! %PIChunkStream by default write in new version, be careful! //! \~russian //! Версия хранения данных. %PIChunkStream на чтение автоматически определяет версию, но для записи //! использует по умолчанию новую, осторожно! enum Version { Version_1 /*! \~english First, old version \~russian Первая, старая версия */, Version_2 /*! \~english Second, more optimized version \~russian Вторая, более оптимизированная версия */ = 2, }; //! \~english Contructs stream for read from "data" //! \~russian Создает поток на чтение из "data" PIChunkStream(const PIByteArray & data): version_(Version_2) {setSource(data);} //! \~english Contructs stream for read or write to/from "data", or empty stream for write if "data" = 0 //! \~russian Создает поток на чтение или запись из/в "data", или пустой поток на запись если "data" = 0 PIChunkStream(PIByteArray * data = 0, Version v = Version_2): version_(v) {setSource(data);} //! \~english Contructs empty stream for write with version \"v\" //! \~russian Создает пустой поток на запись с версией \"v\" PIChunkStream(Version v): version_(v) {setSource(0);} ~PIChunkStream(); template struct Chunk { Chunk(int i, const T & d): id(i), data(d) {} int id; T data; }; template struct ChunkConst { ChunkConst(int i, const T & d): id(i), data(d) {} int id; const T & data; }; //! \~english Returns chunk with ID "id" and value "data" for write to stream //! \~russian Возвращает чанк с ID "id" и значением "data" для записи в поток template static ChunkConst chunk(int id, const T & data) {return ChunkConst(id, data);} //! \~english Add to this stream chunk with ID "id" and value "data" //! \~russian Добавляет в этот поток чанк с ID "id" и значением "data" template PIChunkStream & add(int id, const T & data) {*this << ChunkConst(id, data); return *this;} //! \~english //! Extract %PIByteArray from "data" and set it current stream. //! If "read_all" then call \a readAll() after extract. //! Returns if has data to read. //! \~russian //! Извлекает %PIByteArray из "data" и инициализирует им поток. //! Если указан "read_all", то вызывает \a readAll() после инициализации. //! Возвращает если ли данные для чтения. bool extract(PIByteArray & data, bool read_all = false); void setSource(const PIByteArray & data); void setSource(PIByteArray * data); //! \~english Returns internal buffer with written data //! \~russian Возвращает внутренний буфер с записанными данными PIByteArray data() const; //! \~english Returns if there is end of stream //! \~russian Возвращает достигнут ли конец потока bool atEnd() const {return data_->size_s() <= 1;} //! \~english Returns stream version //! \~russian Возвращает версию потока Version version() const {return (Version)version_;} //! \~english Read one chunk from stream and returns its ID //! \~russian Читает один чанк из потока и возвращает его ID int read(); //! \~english Read all chunks from stream. This function just index input data //! \~russian Читает все чанки из потока. Данный метод лишь индексирует данные void readAll(); //! \~english Returns last readed chunk ID //! \~russian Возвращает ID последнего прочитанного чанка int getID() {return last_id;} //! \~english Returns value of last readed chunk //! \~russian Возвращает значение последнего прочитанного чанка template T getData() const {T ret{}; PIByteArray s(last_data); s >> ret; return ret;} //! \~english Place value of last readed chunk into \"v\" //! \~russian Записывает значение последнего прочитанного чанка в \"v\" template void get(T & v) const {v = getData();} //! \~english Place value of chunk with ID \"id\" into \"v\". You should call \a readAll() before using this function! //! \~russian Записывает значение чанка с ID \"id\" в \"v\". Необходимо вызвать \a readAll() перед использованием этого метода! template const PIChunkStream & get(int id, T & v) const { CacheEntry pos = data_map.value(id); if (pos.start < 0 || pos.length == 0) return *this; PIByteArray ba(data_->data(pos.start), pos.length); if (!ba.isEmpty()) ba >> v; return *this; } //! \~english Replace value of chunk with ID \"id\" to \"v\". You should call \a readAll() before using this function! //! \~russian Заменяет значение чанка с ID \"id\" на \"v\". Необходимо вызвать \a readAll() перед использованием этого метода! template PIChunkStream & set(int id, const T & v) { PIByteArray ba; ba << v; replaceChunk(id, ba); return *this; } private: void _init(); struct CacheEntry { CacheEntry(int s = 0, int l = 0, int b = 0): start(s), length(l), size_bytes(b) {} int start; int length; int size_bytes; }; static uint readVInt(PIByteArray & s, uchar * bytes = 0); static void writeVInt(PIByteArray & s, uint val); static int peekVInt(Version version_, uchar * data_, int sz, uint & ret); void replaceChunk(int id, const PIByteArray & v); int last_id; uchar version_; mutable PIByteArray * data_, last_data, tmp_data; PIMap data_map; bool first_byte_taken; template friend PIChunkStream & operator <<(PIChunkStream & s, const PIChunkStream::Chunk & c); template friend PIChunkStream & operator <<(PIChunkStream & s, const PIChunkStream::ChunkConst & c); }; template PIChunkStream & operator <<(PIChunkStream & s, const PIChunkStream::Chunk & c) { PIByteArray ba; ba << c.data; switch (s.version_) { case PIChunkStream::Version_1: (*(s.data_)) << c.id << ba; break; case PIChunkStream::Version_2: if (s.data_->isEmpty()) (*(s.data_)) << uchar(uchar(s.version_) | 0x80); PIChunkStream::writeVInt(*(s.data_), c.id); PIChunkStream::writeVInt(*(s.data_), ba.size()); s.data_->append(ba); break; default: break; } return s; } template PIChunkStream & operator <<(PIChunkStream & s, const PIChunkStream::ChunkConst & c) { PIByteArray ba; ba << c.data; switch (s.version_) { case PIChunkStream::Version_1: (*(s.data_)) << c.id << ba; break; case PIChunkStream::Version_2: if (s.data_->isEmpty()) (*(s.data_)) << uchar(uchar(s.version_) | 0x80); PIChunkStream::writeVInt(*(s.data_), c.id); PIChunkStream::writeVInt(*(s.data_), ba.size()); s.data_->append(ba); break; default: break; } return s; } #endif // PICHUNKSTREAM_H