Я написал программу на C ++ и пытаюсь использовать curlpp для безопасного доступа к http-серверу через SSL. Проблема в том, что я не могу связать программу.

Операционная система: 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: в функции 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: неопределенная ссылка на curlpp :: internal :: SList :: SList (std :: __ cxx11 :: list, std :: allocator>, std :: allocator, std :: allocator>>> const &) ' CMakeFiles / prtg_probe.dir / probe.cpp.o: в функции 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: неопределенная ссылка на `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::... выдает одну ошибку во время связывания. Я смотрел вокруг и искал в Google несколько часов, но все, что я нашел, - это связать 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

Для меня это похоже на ту же проблему.

8
A.T. 29 Окт 2015 в 20:14

3 ответа

Лучший ответ

Проблема в том, что связь ccurlcpp::UnsetOption::UnsetOption частично неисправен в двоичном файле lipcurlcpp.so.

Жалоба компоновщика на:

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

Является:

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 по какой-то причине не был деформирован должным образом. Если Я получаю исходный файл, в котором этот конструктор определен из 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

Я получил:

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&)

- это подпись, которую компилятор говорит компоновщику искать, но это не подпись в библиотеке. Краткое объяснение ошибки: библиотека не работает.

Однако мы видим, что такое несоответствие не влияет на другую перегрузку конструктора:

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

И не может, поскольку char const * является встроенным типом.

Это позволяет исправить взлом. Файл, в котором находится вызов неопределенной ссылки скомпилировано (как установлено) /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 и соберите и установите сами. Это правильное средство.

9
Mike Kinghan 3 Ноя 2015 в 18:18

Для меня в 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;
}
0
Stepan Yakovenko 30 Окт 2019 в 09:35

Вы можете попробовать скомпилировать свой проект с использованием старого ABI:
g ++ -o example00 example00.cpp -D_GLIBCXX_USE_CXX11_ABI = 0 -lm -lcurl -lcurlpp

4
mborows2 1 Июн 2017 в 11:47