计算机网络课程设计——http请求实现代理服务器的设计——代理服务器端

上代码
看主函数
int main(int argc, char **argv)
知道这个程序该怎么运行了吗??
答案是:打开cmd进入到该项目bin/debug目录下执行proxy-server.exe 3000
proxy-server.exe是可执行文件,你创建工程用的什么名就写什么(Codeblocks,VC那就是文件名)
3000代表端口号,要接收客户端的访问,进而实现代理功能
客户端代码参考下篇
https://blog.csdn.net/dayexiaofan/article/details/85257804

#include <stdio.h>
#include <winsock2.h>
#pragma comment(lib, "ws2_32.lib")  //WinSock使用的库函数

//定义常量
#define HTTP_DEF_PORT        80     //连接的缺省端口
#define HTTP_BUFFER_SIZE      1024     //缓冲区的大小
#define HTTP_FILENAME_LEN   256     //文件名长度

//定义文件类型对应的 Content-Type
struct doc_type
{
    char *suffix; //文件后缀
    char *type;   //Content-Type
};

struct doc_type file_type[] =
{
    {"html","text/html"}, //html文件
    {"gif","image/gif"},  //图片
    {"jpeg","image/jpeg"},//图像
    {NULL,NULL}
};

//http请求
char *http_res_hdr_tmpl =
    "HTTP/1.1 200 OK\r\nServer: YeFanServer <0.1>\r\n"
    "Accept-Ranges: bytes\r\nContent-Length: %d\r\nConnection: close\r\n"
    "Content-Type: %s;charset=UTF-8\r\n";

//根据文件后缀查找对应的 Content-Type.失败返回null
char *http_get_type_by_suffix(const char *suffix)
{
    struct doc_type *type;
    for (type = file_type; type->suffix; type++)
    {
        if (strcmp(type->suffix, suffix) == 0)
            return type->type;
    }
    return NULL;
}

//解析请求行,得到文件名及其后缀,请求行格式【get http://localhost:3000/index.html HTTP/1.1】
//buf代表字符串指针长度、buflen代表buf的长度、file_name文件名、suffix文件名后缀,由上面的函数判断
void http_parse_request_cmd(char *buf, int buflen, char *file_name, char *suffix)
{
    int length = 0;
    char *begin, *end, *bias;
    //查找URL的开始位置
    begin = strchr(buf, ' '); //在字符串buf中搜索第一次出现’空格‘的位置
    begin + = 1;
    //查找URL的结束位置
    end = strchr(begin, ' ');
    *end = 0;

    bias = strrchr(begin, '/'); //找到一个字符'/'在另一个字符串begin中末次出现的位置
    length = end - bias;

    //找到文件名的开始位置
    if ((*bias == '/') || (*bias == '\\'))
    {
        bias++;
        length--;
    }

    //得到文件名
    if (length > 0)
    {
        memcpy(file_name, bias, length);
        file_name[length] = 0;
        begin = strchr(file_name, '.');
        if (begin)
            strcpy(suffix, begin + 1);
    }
}

//向客户端发送HTTP响应
int http_send_response(SOCKET h_socket, char *buf, int buf_len)
{
    int read_len, file_len, hdr_len, send_len;
    char *type;
    char read_buf[HTTP_BUFFER_SIZE];
    char http_header[HTTP_BUFFER_SIZE];
    char file_name[HTTP_FILENAME_LEN] = "index.html", suffix[16] = "html";
    FILE *res_file;

    //得到文件名和后缀
    http_parse_request_cmd(buf, buf_len, file_name, suffix);

    res_file = fopen(file_name, "rb+"); //用读/写方式打开一个二进制文件,只允许读/写数据
    if (res_file == NULL)
    {
        printf("[Web] The file [%s] is not existed\n", file_name);
        return 0;
    }

    fseek(res_file, 0, SEEK_END);
    file_len = ftell(res_file);
    fseek(res_file, 0, SEEK_SET);

    type = http_get_type_by_suffix(suffix); //文件对应的 Content-Type
    if (type == NULL)
    {
        printf("[Web] There is not the related content type\n");
        return 0;
    }

    //构造 HTTP 首部,并发送
    hdr_len = sprintf(http_header, http_res_hdr_tmpl, file_len, type);
    send_len = send(h_socket, http_header, hdr_len, 0);
    //send_len=1;
    if (send_len == SOCKET_ERROR)
    {
        fclose(res_file);
        printf("[Web] Fail to send, error = %d\n", WSAGetLastError());
        return 0;
    }
    //发送文件, HTTP 的消息体
    do
    {
        read_len = fread(read_buf, sizeof(char), HTTP_BUFFER_SIZE, res_file);
        if (read_len > 0)
        {
            send_len = send(h_socket, read_buf, read_len, 0);
            file_len -= read_len;
        }
    } while ((read_len > 0) && (file_len > 0));
    fclose(res_file);
    return 1;
}


int main(int argc, char **argv)
{
    WSADATA wsa_data;
    SOCKET  srv_socket = 0, acpt_soc;  //socket 句柄
    struct sockaddr_in serv_addr;   // 服务器地址
    struct sockaddr_in client_addr;   // 客户端地址
    char recv_buf[HTTP_BUFFER_SIZE];
    unsigned short port = HTTP_DEF_PORT;
    int from_len = sizeof(client_addr);
    int result = 0, recv_len;
    if (argc == 2) // 端口号
        port = atoi(argv[1]);
    WSAStartup(MAKEWORD(2,0), &wsa_data); //初始化 WinSock 资源
    srv_socket = socket(AF_INET, SOCK_STREAM, 0); //创建 socket
    if (srv_socket == INVALID_SOCKET)
    {
        printf("[Web] socket() Fails, error = %d\n", WSAGetLastError());
        return -1;
    }

    //服务器地址
    serv_addr.sin_family = AF_INET;
    serv_addr.sin_port = htons(port);
    serv_addr.sin_addr.s_addr = htonl(INADDR_ANY);

    result = bind(srv_socket, (struct sockaddr *) &serv_addr, sizeof(serv_addr));
    if (result == SOCKET_ERROR) //绑定失败
    {
        closesocket(srv_socket);
        printf("[Web] Fail to bind, error = %d\n", WSAGetLastError());
        return -1;
    }

    result = listen(srv_socket, 5);
    printf("[Web] The server is running ... ...\n");

    while (1)
    {
        acpt_soc = accept(srv_socket, (struct sockaddr *) &client_addr, &from_len);
        if (acpt_soc == INVALID_SOCKET) // 接受失败
        {
            printf("[Web] Fail to accept, error = %d\n", WSAGetLastError());
            break;
        }

        printf("[Web] Accepted address:[%s], port:[%d]\n",inet_ntoa(client_addr.sin_addr), ntohs(client_addr.sin_port));

        recv_len = recv(acpt_soc, recv_buf, HTTP_BUFFER_SIZE, 0);
        if (recv_len == SOCKET_ERROR) // 接收失败
        {
            closesocket(acpt_soc);
            printf("[Web] Fail to recv, error = %d\n", WSAGetLastError());
            break;
        }

        recv_buf[recv_len] = 0;

        //向客户端发送响应数据
        result = http_send_response(acpt_soc, recv_buf, recv_len);
        closesocket(acpt_soc);
    }

    closesocket(srv_socket);
    WSACleanup();
    printf("[Web] The server is stopped.\n");

    return 0;
}

有问题评论区留言,看到会帮忙解决
在这里插入图片描述

评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值