正在添加自定义邮件头

Adding custom message header

本文关键字:自定义 添加      更新时间:2023-10-16

有人能告诉我:"如何构造标头?"我正在编写FTP客户端-服务器代码。我想在客户端为我必须发送到服务器的所有数据包添加一个Header,其中包含我的数据大小等信息。

我在SO见过一些例子,但不知道具体的方法。任何指向示例/示例代码的指针都将受到赞赏。

查看scapy,一个用于构建数据包的库。我认为这正是你所需要的。

建立协议数据的最佳方法是在某个地方有一个uint8_t缓冲区,并按照协议中的指定填充它。

对于FTP本身,这相当简单,因为控制连接中只有ASCII字符串。像这样的东西可能会起作用:

bool append_literal(uint8_t **cursor, uint8_t *end, uint8_t const *data, size_t size) {
    assert(cursor);
    assert(*cursor);
    assert(end);
    assert(*cursor < end);
    assert(data);
    if((end - *cursor) < size)    // Check buffer limit
        return false;
    memcpy(*cursor, data, size);  // Copy
    *cursor += size;              // Increment cursor
    return true;
}
bool append_number(uint8_t **cursor, uint8_t *end, uint64_t number) {
    assert(cursor);
    assert(*cursor);
    assert(end);
    assert(*cursor < end);
    if((end - *cursor) < 17)     // Max number of base 10 digits in uint64_t
        return false;
    *cursor += sprintf(*cursor, "%" PRIu64, number);
    return true;
}
bool append_uint32_binary(uint8_t **cursor, uint8_t *end, uint32_t value) {
    assert(cursor);
    assert(*cursor);
    assert(end);
    assert(*cursor < end);
    if((end - *cursor) < 4)      // Bytes needed for uint32 value
        return false;
    *(*cursor)++ = (value >> 24) & 0xff;
    *(*cursor)++ = (value >> 16) & 0xff;
    *(*cursor)++ = (value >> 8) & 0xff;
    *(*cursor)++ = value & 0xff;
    return true;
}
// Specific FTP commands
bool append_size_command(uint8_t **cursor, uint8_t *end, uint64_t size) {
    return append_literal(cursor, end, "SIZE ", 5) &&
        append_number(cursor, end, size) &&
        append_literal(cursor, end, "rn", 2);
}
// [...]
bool send_size_command(int fd, uint64_t size) {
    uint8_t buffer[32];
    uint8_t *cursor = buffer;
    uint8_t *const end = buffer + sizeof buffer;
    if(!append_size_command(&cursor, end, size))
        return false;
    size_t length = cursor - buffer;
    ssize_t sent = send(fd, buffer, length, 0);
    return (sent == length);
}

如果程序编写正确,即指针有效,那么"assert"语句测试的内容将始终为true。任何函数在成功时返回true,如果失败则返回false

我已经为套接字使用了标准的POSIX函数,因为WinSock实现了这些函数,而且它们是相当可移植的。

您的作业是找到PRIu64需要包含的标题:)

在两个客户端服务器的Header中定义一个成员结构(最好在两者中都包含一个结构的Header文件)。在原始消息之前发送标题。服务器上具有结构的固定长度参数的recv。

typedef struct  
{
    DWORD   dwMsgType ; 
    DWORD   dwMsgSize ;
}MSG_HEADER, *PMSG_HEADER ;