【问题标题】:Handshaking WebSockets in C codeC 代码中的握手 WebSockets
【发布时间】:2012-04-02 13:40:06
【问题描述】:

我是 html5 及其 websocket 的新手!

现在我一直在尝试用 c 制作自己的 websocket 服务器,但是

对我来说很难

我只想从 c 服务器向 html 客户端发送“hello, world”,但是

我在握手方面遇到了麻烦

onMessage 和 onOpen 都不能在客户端工作...!!

据我所知,服务器只需要发送

HTTP/1.1 101 切换协议\r\n

升级:websocket\r\n

连接:升级\r\n

Sec-WebSocket-Accept:我收到的密钥 + 258EAFA5-E914-47DA-95CA-C5AB0DC85B11

而且必须完成! // 就我所知...T_T

我通过在 google 上运行做了一点,但直到现在它都不起作用!

你能告诉我我的错误是什么吗??

===========总结=================

我想知道的是服务器如何发送响应客户端 html,甚至还没有连接!

我在 ubuntu 中工作,所以现在使用文件指针进行套接字编程 现在我的代码使用带有客户端套接字描述符的 write() 函数发送上面的响应

    write(clnt_sock, query, sizeof(query));

但我不确定它是否正确

===================================

这是我的 C 服务器

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <sys/socket.h>

void error_handling(char *message);

int main(int argc, char *argv[])
{
    int serv_sock;
    int clnt_sock;
    int i=0;
    char readHeader[500]={0};
    char* parsingHeader[10];
    char* saveOrigin;
    char* saveKey;
    char* magicKey="258EAFA5-E914-47DA-95CA-C5AB0DC85B11";
    char query[105]="HTTP/1.1 101 Switching Protocols\r\nUpgrade: websocket\r\nConnection: Upgrade\r\nSec-WebSocket-Accept: ";

    struct sockaddr_in serv_addr;
    struct sockaddr_in clnt_addr;
    socklen_t clnt_addr_size;

    char message[]="Hello World!";

    if(argc!=2){
        printf("Usage : %s <port>\n", argv[0]);
        exit(1);
    }

    serv_sock=socket(PF_INET, SOCK_STREAM, 0);
    if(serv_sock == -1)
        error_handling("socket() error");

    memset(&serv_addr, 0, sizeof(serv_addr));
    serv_addr.sin_family=AF_INET;
    serv_addr.sin_addr.s_addr=htonl(INADDR_ANY);
    serv_addr.sin_port=htons(atoi(argv[1]));

    if(bind(serv_sock, (struct sockaddr*) &serv_addr, sizeof(serv_addr))==-1 )
        error_handling("bind() error"); 

    if(listen(serv_sock, 5)==-1)
        error_handling("listen() error");

    clnt_addr_size=sizeof(clnt_addr);  
    clnt_sock=accept(serv_sock, (struct sockaddr*)&clnt_addr,&clnt_addr_size);
    if(clnt_sock==-1)
        error_handling("accept() error");  
    read(clnt_sock,readHeader,355);

        parsingHeader[0]=strtok(readHeader,"\r\n");

    for(i=1;i<6;i++)
        parsingHeader[i]=strtok(NULL,"\r\n");

    saveOrigin=strtok(parsingHeader[4]," ");
    saveOrigin=strtok(NULL," ");
    saveKey=strtok(parsingHeader[5]," ");
    saveKey=strtok(NULL," ");

    strcat(saveKey,magicKey);
    strcat(query,saveKey);
    strcat(query,"\r\n");
    strcat(query,"\r\n");
    printf("%s",query);


    write(clnt_sock, query, sizeof(query));
    close(clnt_sock);   
    close(serv_sock);
    return 0;
}

void error_handling(char *message)
{
    fputs(message, stderr);
    fputc('\n', stderr);
    exit(1);
}`enter code here`

感谢您阅读我的问题!!!

【问题讨论】:

  • 不确定是不是问题,但是写调用应该是 write(clnt_sock, query, sizeof(char) * strlen(query)); 因为查询是一个 char 数组, sizeof() 将是指针的大小,而不是您可能期望的字符串的大小。换句话说,它可能会写 4 个字符而不是整个字符串。
  • 感谢科宾教我!但现在是问题 T_T...我需要知道为什么当服务器发送响应时 opOpen 在客户端不起作用

标签: html websocket


【解决方案1】:

您不能依赖客户端以定义的顺序发送标头。与其取 parsingHeader[5] 的值,不如遍历 parsingHeader 中的所有项目,直到找到“Sec-WebSocket-Key”。

之后,您需要更改响应的密钥。 RFC 6455 对此进行了描述。在您将magicKey 附加到saveKey 之后,您应该获取它的sha-1 哈希,然后对其进行base64 编码。 RFC 3174 包含生成 sha-1 哈希的代码。有很多可用的开源 base64 编码器,例如 this one

【讨论】:

  • 哇,我认识你 simonc!实际上,我在握手后看到了您的回复,但是我看到的您的其他回复使我完成了握手!真的很感谢你帮助我,虽然这是别人的回答!! :D 很高兴在我的问题中看到你!!
  • 你能用工作示例代码更新你的问题吗? :-)
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2021-04-09
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多