I have encountered a problem in sending a QByteArray
through QDataStream
.
I have created a class for storing serialized message to send it through socket, class looks like this:
#ifndef SERIALIZEDMESSAGE_H
#define SERIALIZEDMESSAGE_H
#include <QByteArray>
#include <QDataStream>
#include <QDebug>
class SerializedMessage
{
public:
SerializedMessage();
SerializedMessage(int type, QByteArray data);
friend QDataStream &operator<<(QDataStream &out, const SerializedMessage &message);
friend QDataStream &operator >>(QDataStream &in, SerializedMessage &message);
QByteArray dataArray() const;
void setDataArray(const QByteArray &dataArray);
private:
int m_type;
QByteArray m_dataArray;
};
SerializedMessage::SerializedMessage()
{
}
SerializedMessage::SerializedMessage(int type, QByteArray data)
:m_type(type), m_dataArray(data)
{
}
QByteArray SerializedMessage::dataArray() const
{
return m_dataArray;
}
void SerializedMessage::setDataArray(const QByteArray &dataArray)
{
m_dataArray = dataArray;
}
QDataStream &operator<<(QDataStream &out, const SerializedMessage &message)
{
out << qint32(message.m_type);
qDebug() << "write type" << message.m_type;
out << qint32(message.m_dataArray.length());
qDebug() << "write data lenght" << message.m_dataArray.length();
int bytesWritten = out.writeRawData(message.m_dataArray.data(), message.m_dataArray.length());
qDebug() << bytesWritten << "bytes written";
qDebug() << "write data base64" << message.m_dataArray.toBase64();
return out;
}
QDataStream &operator >>(QDataStream &in, SerializedMessage &message)
{
qint32 type;
qint32 dataLength;
QByteArray dataArray;
char * tmp;
in >> type >> dataLength;
qDebug() << "read type" << type;
qDebug() << "read data lenght" << dataLength;
int bytesRead = in.readRawData(dataArray.data(), dataLength);
qDebug() << bytesRead << "bytes read";
qDebug() << "read data length" << dataArray.length();
qDebug() << "read data base64" << dataArray.toBase64();
message = SerializedMessage(type, dataArray);
return in;
}
To test operators QDataStream & operator>>
and QDataStream &operator<<
I have created testunit.
#include <QString>
#include <QtTest>
#include <QDebug>
#include "textmessage.h"
#include "serializedmessage.h"
class Serialization_test : public QObject
{
Q_OBJECT
public:
Serialization_test();
private Q_SLOTS:
void textMessageSerialization();
void serializedMessageFromArray();
};
Serialization_test::Serialization_test()
{
}
void Serialization_test::textMessageSerialization()
{
TextMessage txtMessage(QString("sender"), QString("message"));
SerializedMessage serialized = txtMessage.toSerializedMessage();
TextMessage deserialized = TextMessage::fromSerializedMessage(serialized);
QCOMPARE(txtMessage.sender(), deserialized.sender());
QCOMPARE(txtMessage.message(), deserialized.message());
}
void Serialization_test::serializedMessageFromArray()
{
TextMessage txtMessage(QString("sender"), QString("message"));
SerializedMessage serialized = txtMessage.toSerializedMessage();
SerializedMessage outSerialized;
QByteArray sr;
QDataStream inputStream(&sr, QIODevice::WriteOnly);
inputStream << serialized;
QDataStream outputStream(sr);
outputStream >> outSerialized;
TextMessage outText = TextMessage::fromSerializedMessage(outSerialized);
QCOMPARE(txtMessage.sender(), outText.sender());
QCOMPARE(outText.message(), outText.message());
}
QTEST_APPLESS_MAIN(Serialization_test)
#include "tst_serialization_test.moc"
To run this test properly you'll need TextMessage
class
#ifndef TEXTMESSAGE_H
#define TEXTMESSAGE_H
#include <QString>
#include <QDataStream>
#include "serializedmessage.h"
class TextMessage
{
public:
TextMessage(const QString & sender, const QString & message);
SerializedMessage toSerializedMessage();
static TextMessage fromSerializedMessage(const SerializedMessage &msg);
QString sender() const;
void setSender(const QString &sender);
QString message() const;
void setMessage(const QString &message);
private:
QString m_sender;
QString m_message;
};
#include "textmessage.h"
TextMessage::TextMessage(const QString &sender, const QString &message)
: m_sender(sender), m_message(message)
{
}
TextMessage TextMessage::fromSerializedMessage(const SerializedMessage & msg)
{
QString sender;
QString message;
QDataStream stream(msg.dataArray());
stream >> sender >> message;
TextMessage textMessge(sender, message);
return textMessge;
}
QString TextMessage::sender() const
{
return m_sender;
}
void TextMessage::setSender(const QString &sender)
{
m_sender = sender;
}
QString TextMessage::message() const
{
return m_message;
}
void TextMessage::setMessage(const QString &message)
{
m_message = message;
}
SerializedMessage TextMessage::toSerializedMessage()
{
QByteArray data;
QDataStream stream(&data, QIODevice::WriteOnly);
stream << m_sender << m_message;
SerializedMessage message(1, data);
return message;
}
#endif // TEXTMESSAGE_H
My problem is when I run void Serialization_test::serializedMessageFromArray()
the test hangs up or fails with message
FAIL! : Serialization_test::serializedMessageFromArray() Compared values are not the same
Actual (txtMessage.sender()): "sender"
Expected (outText.sender()) : ""
What is more, debugging read and write operators shows that out.writeRawData(message.m_dataArray.data(), message.m_dataArray.length());
writes the same amout of data, that is read in in.readRawData(dataArray.data(), dataLength);
. I'm not pretty sure what is going on. I would be grateful for any help
Your problem is in
operator >>(QDataStream&, SerializedMessage&)
. You must ensure that theQByteArray
is sized properly before you attempt to write bytes into it usingQDataStream::readRawData
: