C代码中握手WebSockets

时间:2012-03-17 03:23:41

标签: html5 websocket

我是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

我通过在谷歌上运行做了一点但直到现在它还没有用!

你可以让我知道我的错误吗?

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

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

我在ubuntu工作,所以现在使用File指针进行套接字编程 现在我的代码通过使用带有客户端套接字描述符的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`

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

1 个答案:

答案 0 :(得分:2)

您不能依赖客户端按定义的顺序发送标头。你应该遍历parsingHeader中的所有项目,直到你找到“Sec-WebSocket-Key”,而不是获取parsingHeader [5]的值。

之后,您需要更改您回复的密钥。 RFC 6455描述了这一点。将magicKey附加到saveKey之后,应该使用sha-1哈希,然后对其进行base64编码。 RFC 3174包含生成sha-1哈希的代码。有许多开源base64编码器可用,如this one