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
来防止截断长行。
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
来防止截断长行。