试图制作AMF3数据包

时间:2014-01-27 00:58:00

标签: c++ flex amf

我正在尝试制作AMF数据包。我正在使用AMF3的https://github.com/Ventero/amf-cpp C ++实现,但它不包含所有需要的变量。 AMF0文档描述了如何构建AMF数据包(http://download.macromedia.com/pub/labs/amf/amf0_spec_121207.pdf) 前两个字节指定数据包版本:

  

版本= U16   它可以是0或3

U16是大端的无符号16位整数 (网络)字节顺序

所以我创造了

typedef unsigned int u16;

在amf.hpp中 我现在正在尝试将所有头信息添加到单个std :: vector变量中。 我写道:

std::vector<u16> buf = std::vector<u16>{3};  //packet version
std::vector<u16> buf2 = std::vector<u16>{0}; //header-count
std::vector<u16> buf3 = std::vector<u16>{1}; //message count

std::vector<u16> data;

buffermain.insert(data.end(), buf.begin(), buf.end());
buffermain.insert(data.end(), buf2.begin(), buf2.end());
buffermain.insert(data.end(), buf3.begin(), buf3.end());

在数据结果中我刚刚插入了第一个矢量(buf)。

//编辑 我取得了一些进展。

Serializer serializer;
QByteArray outArray; //to insert amf bytes, and send it later

AmfArray Content;
AmfObject Object;

Object.addSealedProperty("Source", AmfNull());
Object.addSealedProperty("operation", AmfNull());
Object.addSealedProperty("clientId", AmfNull());
Object.addSealedProperty("destination", AmfNull());
Object.addSealedProperty("messageId", AmfNull());
Object.addSealedProperty("timestamp", AmfNull());
Object.addSealedProperty("timeToLive", AmfNull());
Object.addSealedProperty("timeToLive", AmfNull());
Object.addSealedProperty("body", AmfNull());
Object.addSealedProperty("headers", AmfNull());

Content.push_back(Object);
serializer << Content;

std::vector<uint8_t> data2 = serializer.data();

char* datas = reinterpret_cast<char*>(data2.data());//

std::vector<unsigned __int32> v;
v.reserve(data.size());

char* sizes = reinterpret_cast<char*>(v.data()); //teoretical size of message in U32

char null = 0;
outArray.append(null); //version first byte
outArray.append(3); //version second byte
outArray.append(null); //header count first byte
outArray.append(null); //header count second byte
outArray.append(null); //messages count first byte
outArray.append(1); //messages count second byte

outArray.append(null); //"Target" lenght first byte 
outArray.append(4); //"Target" lenght second byte 

outArray.append(QByteArray::fromHex("6e756c6c")); // "Target" value

outArray.append(null); // "Response" length first byte
outArray.append(2); // "Response" length second byte

outArray.append(QByteArray::fromHex("2f31"));

outArray.append(sizes); //insert theoretical length of message
outArray.append(datas); //insert message

2 个答案:

答案 0 :(得分:2)

(我是amf-cpp的作者)

我刚刚为amf-cpp添加了AMF3数据包支持。要使用它,您必须创建AmfPacket对象,添加标题(PacketHeader)或消息(PacketMessage),然后序列化AmfPacket。这是一个简单的例子(为简洁起见假定using namespace amf;之类的东西):

AmfPacket packet;
// first, we construct a simple header in-place
packet.headers.emplace_back(
    "SomeHeader", // header name
    false, // must understand?
    AmfString("Value") // header value
);

// set up the message value
AmfObject object;
// add some properties
object.addSealedProperty("prop", AmfString("val"));
AmfArray content;
content.push_back(object);
// now construct the message in-place
packet.messages.emplace_back(
    "com/example/Object.method", // target uri
    "/1/onResult", // response-uri
    content // value
);

// amf::v8 is a typedef for std::vector<uint8_t>
v8 data = packet.serialize();

生成的data对象可以与您的QDataStream一起使用。

将来,如果您发现amf-cpp缺少您想要使用的功能,请随时在GitHub issue tracker上打开错误报告。

答案 1 :(得分:0)

您可以使用QDataStream以网络字节顺序输出数据。

QDataStream stream(&outArray, QIODevice::WriteOnly);
stream.setByteOrder(QDataStream::BigEndian);
stream << (qint16)3; // version
stream << (qint16)0; // header count
stream << (qint16)1; // message count
stream << (qint16)4; // target length
stream << (qint32)0x6e756c6c; // target
stream << (qint16)2; // response length
stream << (qint16)0x2f31; // response
stream << (qint32)data.size(); // message size
stream.writeRawData(data.data(), data.size()); // message data

我建议切换所有I / O代码,使用QDataStream与QByteArray连接,而不是单独写出每个字节。