最近用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"); } }
这样就万无一失了。