使用curlpp出现链接错误

8
我编写了一个C++程序,尝试使用curlpp安全访问一个基于SSL的HTTP服务器。问题是,我无法链接该程序。
操作系统:Ubuntu 15.10
错误信息:
/usr/bin/cmake -E cmake_link_script CMakeFiles/prtg_probe.dir/link.txt --verbose=1 Wird gelinkt prtg_probe (c++) CMakeFiles/prtg_probe.dir/probe.cpp.o: In function curlpp::internal::OptionContainer, std::allocator >, std::allocator, std::allocator > > >

::OptionContainer(std::__cxx11::list, std::allocator >, std::allocator, std::allocator > > > const&)': /usr/include/curlpp/internal/OptionContainer.inl:38: undefined reference to curlpp::internal::SList::SList(std::__cxx11::list, std::allocator >, std::allocator, std::allocator > > > const&)' CMakeFiles/prtg_probe.dir/probe.cpp.o: In function curlpp::internal::OptionContainer, std::allocator >, std::allocator, std::allocator > > > ::setValue(std::__cxx11::list, std::allocator >, std::allocator, std::allocator > > > const&)': /usr/include/curlpp/internal/OptionContainer.inl:52: undefined reference to `curlpp::internal::SList::operator=(std::__cxx11::list, std::allocator >, std::allocator, std::allocator > > > const&)' ....
这是导致错误的代码:
#include <iostream>
#include <sstream>
#include <string>
#include <cstring>
#include <cctype>
#include <cstddef>
#include <cstdio>
#include <cstdlib>
#include <fstream>
#include <cerrno>
#include <boost/bind.hpp>
#include <curlpp/cURLpp.hpp>
#include <curlpp/Easy.hpp>
#include <curlpp/Options.hpp>
#include <curlpp/Exception.hpp>

....

class myclass
{

....

try
{
    curlpp::Cleanup cleaner;
    curlpp::Easy request;
    ostringstream os;

    request.setOpt(new curlpp::options::Url(&url_announce[0]));
    request.setOpt(new curlpp::options::SslEngineDefault());
    list<string> header;
    header.push_back("Content-Type: text/*");
    request.setOpt(new curlpp::options::HttpHeader(header));
    request.setOpt(new curlpp::options::PostFields(data_announce));
    request.setOpt(new curlpp::options::PostFieldSize((long)data_announce.length()));
    request.setOpt(new curlpp::options::WriteStream(&os));
    request.perform();
    request_announce = os.str();
}
catch (curlpp::LogicError & e)
{
    syslog(LOG_DAEMON, "Error accessing PRTG server: %s", e.what());
}
catch (curlpp::RuntimeError & e)
{
    syslog(LOG_DAEMON, "Error accessing PRTG server: %s", e.what());
}

每一行使用curlpp::options::...都会在链接时产生一个错误。我已经搜索了谷歌几个小时,但是找到的只是将libcurl与libcurl一起链接。我尝试过这样做,但仍然出现这些错误。

以下是完整的链接命令:

/usr/bin/c++   -g    CMakeFiles/prtg_probe.dir/sensors.cpp.o 
CMakeFiles/prtg_probe.dir/mini_probe.cpp.o CMakeFiles/prtg_probe.dir/probe.cpp.o
CMakeFiles/prtg_probe.dir/helper.cpp.o 
CMakeFiles/prtg_probe.dir/config.cpp.o CMakeFiles/prtg_probe.dir/main.cpp.o
-o prtg_probe -rdynamic -lm -lpthread -lcrypto -lssl -lcurlpp -lcurl -lboost_system -lboost_filesystem -ljsoncpp -luuid

有人知道我缺少什么吗?我需要添加另一个库吗?如果是的话,是哪一个?


我尝试使用以下命令编译example00.cpp

g++ -o example00 example00.cpp -lm -lcurl -lcurlpp

结果如下:

/tmp/cc3pcvDc.o: In Funktion `curlpp::OptionTrait<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, (CURLoption)10002>::updateHandleToMe(curlpp::internal::CurlHandle*) const':
example00.cpp: (.text._ZNK6curlpp11OptionTraitINSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEEL10CURLoption10002EE16updateHandleToMeEPNS_8internal10CurlHandleE[_ZNK6curlpp11OptionTraitINSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEEL10CURLoption10002EE16updateHandleToMeEPNS_8internal10CurlHandleE]+0x68):
Nicht definierter Verweis auf `curlpp::UnsetOption::UnsetOption(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)'
/tmp/cc3pcvDc.o: In Funktion `curlpp::Option<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >::getValue() const':
example00.cpp: (.text._ZNK6curlpp6OptionINSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEEE8getValueEv[_ZNK6curlpp6OptionINSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEEE8getValueEv]+0x68):
Nicht definierter Verweis auf `curlpp::UnsetOption::UnsetOption(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)'
collect2: error: ld returned 1 exit status

这对我来说似乎是同样的问题。

尝试创建一个最小化的示例来解决你的问题。删除与单个curlpp::options行无关的所有行,删除所有额外的库。如果最小集合有效,则添加更多行和库,直到出现问题。 - usr1234567
正如您在上面所看到的,当我尝试编译cURLpp中包含的一个示例时,我会遇到相同的错误。 - A.T.
你尝试过倒序排列吗?-lm -lcurl -lcurlpp - usr1234567
3个回答

9
问题在于ccurlcpp::UnsetOption::UnsetOption的链接在lipcurlcpp.so二进制文件中存在部分缺陷。
链接器对以下内容提出了抱怨:
g++ -o example00 example00.cpp -lm -lcurl -lcurlpp

is:

undefined reference to `curlpp::UnsetOption::UnsetOption(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)'

但如果我在libcurlpp.so中解析构造函数的签名:

nm -D -C libcurlpp.so | grep UnsetOption::UnsetOption

我明白:

0000000000021776 T curlpp::UnsetOption::UnsetOption(char const*)
000000000002173e T curlpp::UnsetOption::UnsetOption(std::string const&)

出于某些原因,std::string 没有被正确地去除 typedef。如果从 curlpp 0.7.3 源代码包的 Exception.cpp 获取定义此构造函数的源文件,则进行编译:

curlpp-0.7.3/src/curlpp$ g++ -I../../include -I. -c Exception.cpp

接下来从目标文件中去除构造函数的符号重载:

nm -C Exception.o | grep UnsetOption::UnsetOption

I get:

00000000000003f4 T curlpp::UnsetOption::UnsetOption(char const*)
00000000000003c2 T curlpp::UnsetOption::UnsetOption(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)

所以:
curlpp::UnsetOption::UnsetOption(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)

“signature”是编译器告诉链接器要查找的标记,但该库中并不包含此标记。这个错误的简短解释是:该库有问题。

然而,我们可以看到另一个构造函数重载没有受到这种不一致的影响:

curlpp::UnsetOption::UnsetOption(char const*)

由于char const *是内置类型,因此它也无法实现。这使得一种hack修复成为可能。在编译未定义引用调用的文件(如安装后的)/usr/include/curlpp/Option.inl中,位于以下行:

throw UnsetOption(std::string("You are trying to set an unset option to a handle"));

以root身份编辑此文件,您会发现它(不一致地)包含两个实例:

throw UnsetOption(std::string("blah blah"));

并且还有一个实例:

throw UnsetOption("blah blah");

UnsetOption(std::string("blah blah"))的出现更改为UnsetOption("blah blah")。然后,只有这个文件中的良好构造函数才会被调用,至少example00将编译和链接。
如果您不喜欢这种方法,或者发现问题在其他地方重新出现,则可以下载Ubuntu源包curlpp_0.7.3.orig.tar.gz并自行构建和安装。这是正确的解决方法。

非常感谢您的出色解释。我已经获取了源代码并编译了库。现在我能够链接示例并使它们正常工作。 - A.T.
@A.T. 不用客气。如果这个答案解决了你的问题,你应该接受它,这样它就不会再出现在SO上未回答的问题列表中了。你会因此获得一些声望分,我也会获得一些。 - Mike Kinghan

5
您可以尝试使用旧版ABI编译您的项目:
g++ -o example00 example00.cpp -D_GLIBCXX_USE_CXX11_ABI=0 -lm -lcurl -lcurlpp

1
简单的方法!太棒了! - choujayyl

0

对于我,在Ubuntu 18.04上可以尝试以下方法:

apt-get install libcurl4-openssl-dev
apt-get install libcurlpp-dev

Cmake:

...
add_executable(HTTPPostTest HTTPPostTest.cpp)
TARGET_LINK_LIBRARIES(HTTPPostTest -lcurl -lcurlpp stdc++fs )
...

HTTPPostTest.cpp:

#include <curlpp/cURLpp.hpp>
#include <curlpp/Easy.hpp>
#include <curlpp/Options.hpp>
#include <curlpp/Exception.hpp>

int main(int argc, char *argv[]) {
    try {
        curlpp::Cleanup cleaner;
        curlpp::Easy request;
        request.setOpt(new curlpp::options::Url("https://postman-echo.com/post"));
        request.setOpt(new curlpp::options::Verbose(true));
        std::list<std::string> header;
        header.push_back("Content-Type: application/octet-stream");
        request.setOpt(new curlpp::options::HttpHeader(header));
        request.setOpt(new curlpp::options::PostFields("abcd"));
        request.setOpt(new curlpp::options::PostFieldSize(5));
        request.perform();
    } catch ( curlpp::LogicError & e ) {
        std::cout << e.what() << std::endl;
    } catch ( curlpp::RuntimeError & e ) {
        std::cout << e.what() << std::endl;
    }
    return EXIT_SUCCESS;
}

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接