C-SERVER:HTTP响应缺少新行

C-SERVER:HTTP响应缺少新行,c,bash,shell,webserver,C,Bash,Shell,Webserver,C-SERVER: 执行外部脚本:/script 存储在“vvv”和“vvv”中的脚本的响应被发送到客户端 fp = popen("/script", "r"); fgets(vvv, 500, fp); write(client_fd, vvv, sizeof(vvv) - 1); close(client_fd); /script的内容 #!/bin/sh echo -e 'HTTP/1.1 200 OK\r\n'; 使用netcat进行测试: echo TEST | nc <i

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

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

write(client_fd, vvv, sizeof(vvv) - 1);
close(client_fd);
/script的内容

#!/bin/sh
echo -e 'HTTP/1.1 200 OK\r\n';
使用netcat进行测试:

echo TEST | nc <ip> <port>
HTTP/1.1 200 OK
echo测试| nc
HTTP/1.1200ok
问题:它无法包含新行,因此无法使用web浏览器

C-SERVER:

#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);
  }
}
#包括
#包括
#包括
查尔酒店[100];
文件*fp;
char-vvv[100];
int main()
{
int one=1,客户端\u fd;
svr\u addr、cli\u addr中的结构sockaddr\u;
socklen\u t sin\u len=sizeof(cli\u addr);
intsock=socket(AF_INET,sock_STREAM,0);
if(sock<0)
错误(1,“无法打开插座”);
setsockopt(sock,SOL_SOCKET,SO_REUSEADDR,&one,sizeof(int));
int端口=85;
svr\u addr.sin\u family=AF\u 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){
关闭(袜子);
错误(1,“无法绑定”);
}
听(短袜,5);
而(1){
client\u fd=accept(sock,(struct sockaddr*)&cli\u addr和sin\u len);
size_t bytesRead=read(客户、客栈、sizeof(客栈)-1);
fp=popen(“/script”,“r”);
fgets(vvv,500,fp);
printf(vvv);
写入(客户机fd、vvv、sizeof(vvv)-1);
关闭(客户_fd);
}
}

请再次阅读文档。内容如下:

从stream指向的流到s指向的数组中,最多比n指定的字符数少一个在新行字符(保留)之后或在文件结尾之后不读取其他字符。。在读取到数组中的最后一个字符之后立即写入空字符

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

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头和内容之间用一条空行隔开。如果您根本不提供任何内容,情况也是如此

另外,
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);
}

...

也要考虑使用<代码> Frad 而不是<代码> fgs<代码>以防止截断长线。

我对C太新了。如果这是答案的话,我甚至不知道如何做VVV=VVV+/N”。当然,这不是你的缓冲区<代码> VVV 是100字节,但是你给了限制500到<代码> FGES为什么?如果脚本确实回显了“HTTP/1.1 200 OK\r\n”;(否-e标志)仍然不起作用。结果是:HTTP/1.1 200正常\r\n\n@Antti哈帕拉,我甚至不明白那些事。只是在我让脚本发挥作用之前给一些额外的空间。这些都是对未来的次要担忧。