本文仅记录研究rabbitmq-c客户端源码时,对RabbitMQ帧数据构成进行总结。
先上一张图:
上图是rabbitmq-c客户端与服务器进行初始信令交互时的抓包,在此仅此作为示例使用。其中图中所示为第一条Connection.Start信令的具体内容。请关注以下几个值:
1.第一行显示的Length值为389
2.协议内部的Length值为327
那么389-327=62字节的内容都包括了什么?其包括了:14(Ethernet头)+20(IP头)+20(TCP头)=54字节,以及AMQP的帧数据 7(帧头)+1(帧尾)=8字节。
源码中有如下结构:
struct amqp_connection_state_t_ {
amqp_pool_t frame_pool;
amqp_pool_t decoding_pool;
amqp_connection_state_enum state;
int channel_max;
int frame_max;
int heartbeat;
amqp_bytes_t inbound_buffer;
size_t inbound_offset;
size_t target_size;
amqp_bytes_t outbound_buffer;
int sockfd;
amqp_bytes_t sock_inbound_buffer;
size_t sock_inbound_offset;
size_t sock_inbound_limit;
amqp_link_t *first_queued_frame;
amqp_link_t *last_queued_frame;
amqp_rpc_reply_t most_recent_api_result;
};
其中 target_size 是指什么呢?我们可以从另外三段代码中找到答案:
在amqp_new_connection中
state->target_size = 8;
在return_to_idle中
state->target_size = HEADER_SIZE;
在amqp_handle_input中
state->target_size = amqp_d32(raw_frame, 3) + HEADER_SIZE + FOOTER_SIZE;
哈哈,原来该值正是AMQP的实际要处理的帧的长度(在上例中其实就是327+7+1)。
有了上述说明,再看代码时就不会被其中的各种帧解析代码搞晕了!