最近用fastcgi的c语言api写服务,发现一个问题。我用nginx来接收请求,并通过fastcgi_pass传递到c程序。在用curl测试请求的时候,发现c程序是有被调用的,但是nginx返回的响应一直是502 "upstream closed prematurely FastCGI stdout while reading response header from upstream"。在网上找了很久,也有同样的问题,但是一般都是说原因是printf的时候没有按照http协议,比如说一定要printf("Content-type: text/html \r\n\r\n")。我按照这个格式写了,但还是出错。最后发现是由于include了c和c++的一些头文件,比如,下面的代码就有可能出现这样的问题(也有可能不出现问题):
#include "fcgi_stdio.h" #include <iostream> #include <string> #include <map> #include <stdlib.h> #include <stdio.h> int main() { while(FCGI_Accept() >= 0) { // //get content type and length // printf("Content-type: text/*\r\n\r\n"); printf("<title>FastCGI Hello! (C, fcgi_stdio library)</title>\n"); } }
在官方的api使用例子里面,只有include “fcgi_stdio.h”和“stdlib.h”这两个头文件。于是我看了一下fcgi_stdio.h这个文件,发现它定义了一个宏,也叫printf,而事实上是调用了FCGI_printf这个函数。 而include其他的头文件可能会导致链接的时候链接到原来的printf实现,所以最保险的做法是:
#include "fcgi_stdio.h"
#include <iostream>
#include <string>
#include <map>
#include <stdlib.h>
#include <stdio.h>
int main()
{
while(FCGI_Accept() >= 0)
{
//
//get content type and length
//
FCGI_printf("Content-type: text/*\r\n\r\n");
FCGI_printf("<title>FastCGI Hello! (C, fcgi_stdio library)</title>\n");
}
}
这样就万无一失了。