Html C代码中的握手WebSocket

Html C代码中的握手WebSocket,html,websocket,Html,Websocket,我是html5及其websocket的新手 现在我一直在尝试用c制作自己的websocket服务器,但是 这对我来说很难 我只想从c服务器向html客户端发送“hello,world”,但是 我在握手方面遇到了麻烦 onMessage和onOpen都不在客户端工作 据我所知,服务器只需要发送 HTTP/1.1 101交换协议\r\n 升级:websocket\r\n 连接:升级\r\n Sec WebSocket接受:密钥I接收+258EAFA5-E914-47DA-95CA-C5AB0DC85

我是html5及其websocket的新手

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

这对我来说很难

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

我在握手方面遇到了麻烦

onMessage和onOpen都不在客户端工作

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

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

升级:websocket\r\n

连接:升级\r\n

Sec WebSocket接受:密钥I接收+258EAFA5-E914-47DA-95CA-C5AB0DC85B11

这是必须完成的!//就我所知…T

我通过在谷歌上运行赚了一点钱,但直到现在它还不起作用

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

============摘要================

我想知道的是服务器如何发送响应到客户端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`
#包括
#包括
#包括
#包括
#包括
#包括
无效错误处理(字符*消息);
int main(int argc,char*argv[])
{
int serv_sock;
int clnt_袜子;
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交换协议\r\n升级:websocket\r\n连接:升级\r\nSec websocket接受:”;
服务地址中的结构sockaddr\u;
clnt\u addr中的结构sockaddr\u;
袜子的尺寸;
char message[]=“你好,世界!”;
如果(argc!=2){
printf(“用法:%s\n”,argv[0]);
出口(1);
}
serv_sock=socket(PF_INET,sock_STREAM,0);
如果(serv_sock==-1)
错误处理(“套接字()错误”);
memset(&serv_addr,0,sizeof(serv_addr));
服务地址sin_family=AF_INET;
服务地址sin\U addr.s\U addr=htonl(INADDR\U ANY);
服务地址sin_port=htons(atoi(argv[1]);
if(bind(serv_sock,(struct sockaddr*)和serv_addr,sizeof(serv_addr))=-1)
错误处理(“bind()错误”);
if(监听(serv_sock,5)=-1)
错误处理(“侦听()错误”);
clnt\u addr\u size=sizeof(clnt\u addr);
clnt\u sock=accept(serv\u sock,(struct sockaddr*)和clnt\u addr,以及clnt\u addr\u size);
如果(clnt_sock==-1)
错误处理(“接受()错误”);
读取(clnt_sock,读取头,355);
parsingHeader[0]=strtok(readHeader,“\r\n”);

对于(i=1;i,您不能依赖于客户端以定义的顺序发送标题。您不应该使用parsingHeader[5]的值,而应该循环parsingHeader中的所有项目,直到找到“Sec WebSocket Key”


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

不确定这是否是问题,但写入调用应该是write(clnt_sock,query,sizeof(char)*strlen(query));因为query是一个char数组,所以sizeof()将是指针的大小,而不是您可能期望的字符串的大小。换句话说,它可能会写入4个字符而不是整个字符串。感谢您教我Corbin!但现在这是一个问题T_T…我需要知道为什么当服务器发送响应时opOpen无法在客户端工作哇,我知道您是simonc!实际上我看到了您的答复在我握手之后,除了你的其他回复,我看到的其他回复让我完成了握手!!非常感谢你的帮助,尽管这是其他人的回答!!:D我很高兴在我的问题中看到你!!你能用工作示例代码更新你的问题吗?:-)