C-SERVER:HTTP 响应缺少新行

C-SERVER : HTTP response missing new line

C-服务器:
执行外部脚本:/script
来自存储在“vvv”和“vvv”中的脚本的响应被发送到客户端。

fp = popen("/script", "r");
fgets(vvv, 500, fp);

write(client_fd, vvv, sizeof(vvv) - 1);
close(client_fd);

/脚本的内容

#!/bin/sh
echo -e 'HTTP/1.1 200 OK\r\n';

使用 netcat 测试:

echo TEST | nc <ip> <port>
HTTP/1.1 200 OK

问题:它无法包含一个新行,因此它可以与网络浏览器一起使用。

C-服务器:

#include <arpa/inet.h>
#include <err.h>
#include <string.h>

char inn[100];

FILE *fp;
char vvv[100];

int main()
{
  int one = 1, client_fd;
  struct sockaddr_in svr_addr, cli_addr;
  socklen_t sin_len = sizeof(cli_addr);

  int sock = socket(AF_INET, SOCK_STREAM, 0);
  if (sock < 0)
    err(1, "can't open socket");

  setsockopt(sock, SOL_SOCKET, SO_REUSEADDR, &one, sizeof(int));

  int port = 85;
  svr_addr.sin_family = AF_INET;
  svr_addr.sin_addr.s_addr = INADDR_ANY;
  svr_addr.sin_port = htons(port);

  if (bind(sock, (struct sockaddr *) &svr_addr, sizeof(svr_addr)) == -1) {
    close(sock);
    err(1, "Can't bind");
  }

  listen(sock, 5);
  while (1) {
    client_fd = accept(sock, (struct sockaddr *) &cli_addr, &sin_len);
    size_t bytesRead = read(client_fd,inn,sizeof(inn) - 1);

    fp = popen("/script", "r");
    fgets(vvv, 500, fp);
    printf(vvv);


    write(client_fd, vvv, sizeof(vvv) - 1);
    close(client_fd);
  }
}

再次,请阅读文档。 fgets 读取:

at most one less than the number of characters specified by n from the stream pointed to by stream into the array pointed to by s. No additional characters are read after a new-line character (which is retained) or after end-of-file. A null character is written immediately after the last character read into the array.

HTTP 响应必须包含 2 个 CRLF 换行符 - 第一个终止状态行,第二个将状态行与正文分开:

HTTP/1.0 200 OK\r\n
\r\n

您的脚本同样应该生成这些换行符:

echo -e 'HTTP/1.1 200 OK\r\n\r\n';

但是您的 fgets 呼叫停止很短 - 它只读到第一个 \n 字符。实际上你根本不应该使用 fgets,而是重复使用 fread 直到满足 EOF,比如

fp = popen("/script", "r");
while (!feof(fp) && !ferror(fp)) {
    size_t bytes_read = fread(vvv, 1, sizeof(vvv), fp);
    if (read) {
        write(client_fd, vvv, bytes_read);
    }
}
pclose(fp); // must *absolutely* close the pipe with pclose

(P.S.HTTP/1.1协议有点复杂,如果你声称支持它,那么你还需要支持各种额外的东西)

请注意,HTTP header 和内容由一个空行分隔。如果您根本不提供任何内容,也是如此。

另外 echo 已经在该行之后附加了一个换行符。使用 printf 代替:

#!/bin/sh
printf 'HTTP/1.1 200 OK\r\n';
printf '\r\n';
printf 'Content'; # provide some content

但这需要您现在需要从 fp 中读取多行。添加一个额外的循环以读取 fp:

中的每一行
...

while (1) {
  client_fd = accept(sock, (struct sockaddr *) &cli_addr, &sin_len);
  size_t bytesRead = read(client_fd,inn,sizeof(inn) - 1);

  fp = popen("/script", "r");

  while (!feof(fp)) {
    // be careful here to use `sizeof` instead of the harcoded `500`
    fgets(vvv, sizeof(vvv), fp);
    // only write the number of bytes read
    write(client_fd, vvv, strlen(vvv));
    printf(vvv);
  }

  close(client_fd);
}

...

同时考虑使用 fread 而不是 fgets 来防止截断长行。