C-SERVER:HTTP响应缺少新行
C-SERVER: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
执行外部脚本:/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哈帕拉,我甚至不明白那些事。只是在我让脚本发挥作用之前给一些额外的空间。这些都是对未来的次要担忧。