我正在开发FUSE驱动程序,当我将其作为守护程序运行时(没有-f或-d标志),通过libcurl发出的所有https请求都会失败。通过发出https请求,派生并返回父进程,然后从新进程中发出第二个请求,我能够重现该错误。如果我删除fork
调用或发出http请求,则没有错误。
我现在正在发布正式的错误报告,但是有人知道我该如何使其工作吗?
这是我的代码和(日志文件)输出:
注意:如果您运行我的程序,请通过管道传递到/dev/null,因为libcurl默认情况下会将接收到的缓冲区发送到stdout。
#include <curl/curl.h>
#include <string>
#include <unistd.h>
#include <iostream>
using namespace std;
void log(string str)
{ //TODO: remove
static const char logfile[] = "/home/austin/megalog";
FILE *f = fopen(logfile, "a");
fwrite(str.data(), str.size(), 1, f);
fclose(f);
cout << str;
}
int main(int argc, char *argv[])
{
string url = "https://www.google.com/";
char errBuf[1024];
CURLcode err;
curl_global_init(CURL_GLOBAL_DEFAULT);
CURL *handle = curl_easy_init();
curl_easy_setopt(handle, CURLOPT_URL, url.c_str());
curl_easy_setopt(handle, CURLOPT_ERRORBUFFER, errBuf);
if ((err = curl_easy_perform(handle)))
{
log("first request failed\n");
return 1;
}
curl_easy_cleanup(handle);
if(fork())
return 0;
handle = curl_easy_init();
curl_easy_setopt(handle, CURLOPT_URL, url.c_str());
curl_easy_setopt(handle, CURLOPT_ERRORBUFFER, errBuf);
if ((err = curl_easy_perform(handle)))
{
log(string("curl error while sending: (") + to_string(err) + ") " + curl_easy_strerror(err) + "\n");
log(errBuf);
}
else
log("no error\n");
return 0;
}
...以及输出:
$ g++ -std=c++11 main.cpp -lcurl
$ rm -f log
$ ./a.out > /dev/null
$ cat log
curl error while sending: (35) SSL connect error
A PKCS #11 module returned CKR_DEVICE_ERROR, indicating that a problem has occurred with the token or slot.
我正在使用(最新)libcurl版本7.29.0,(最新)openssl版本1.0.1e,并且我正在运行Fedora 18和内核版本3.7.4。
最佳答案
要使此工作有效,您需要先在curl_global_cleanup
之前调用fork
,在curl_global_init
之后再调用fork
。 libcurl邮件列表中的某人提到,在初始化需要初始化的库之后调用fork
时,这是一个常见问题。
关于linux - fork()之后libCurl SSL错误,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/15466809/