[ZBX-23838] Cannot build server with libcurl in non-standard location Created: 2023 Dec 13  Updated: 2024 Apr 10  Resolved: 2024 Jan 05

Status: Closed
Project: ZABBIX BUGS AND ISSUES
Component/s: Installation (I)
Affects Version/s: 7.0.0alpha9
Fix Version/s: 7.0.0beta1, 7.0 (plan)

Type: Problem report Priority: Trivial
Reporter: Alex Kalimulin Assignee: dimir
Resolution: Workaround proposed Votes: 0
Labels: None
Remaining Estimate: Not Specified
Time Spent: Not Specified
Original Estimate: Not Specified

Attachments: PNG File image-2024-01-04-16-11-07-048.png    
Issue Links:
Duplicate
Team: Team A
Sprint: Sprint candidates
Story Points: 2

 Description   

Steps to reproduce:

  1. Get system with old libcurl that cannot be upgraded
  2. Install new curl in other location (e.g. /usr/local)
  3. Build server with ./configure --enable-server --with-libcurl=/usr/local/bin/curl-config

Result:

../../src/libs/zbxhttp/libzbxhttp.a(http.o): In function `zbx_http_convert_to_utf8':
/usr2/alex/home/devel/zabbix/src/libs/zbxhttp/http.c:957: undefined reference to `curl_easy_header'

Expected:

Successful build

Analysis:

Presumably, even though curl-config returns path to libcurl to use for linking, ./configure does not account the proper path so make links against old lib in standard location.

 



 Comments   
Comment by Vladislavs Sokurenko [ 2023 Dec 14 ]

It appears that include file is from new version while actually compiled with old version.

Comment by dimir [ 2023 Dec 14 ]

Reproduced.

Prerequisites

2 libraries installed:

/usr/lib/x86_64-linux-gnu/libcurl.so (7.64.0)
/usr/local/lib/libcurl.so            (8.5.0)

There is a libcurl function Zabbix needs: curl_easy_header, introduced in cURL 7.83.0 (as you can see, not available in older installed version).

What happens

That's a linker error, not a compiler, so the header file is not an issue. What seems to be an issue is the order of -L options passed to the linker. And this order depends on the features enabled when running ./configure .

I was able to reproduce the issue when configuring Zabbix server with MySQL:

$ ./configure --enable-server --with-mysql --with-libcurl=/usr/local/bin/curl-config
[...]
Linker flags:                -L/usr/lib/x86_64-linux-gnu/ -L/usr/local/lib -rdynamic

Running make, you get the linker error (note the order of -L options):

cc <OPTS> -L/usr/lib/x86_64-linux-gnu/ -L/usr/local/lib <OPTS>
/usr/bin/ld: ../../src/libs/zbxhttp/libzbxhttp.a(http.o): in function `zbx_http_convert_to_utf8':
/root/git/zabbix-master/src/libs/zbxhttp/http.c:957: undefined reference to `curl_easy_header'
collect2: error: ld returned 1 exit status 

This is because it will find the "old" curl library in

/usr/lib/x86_64-linux-gnu/libcurl.so 

which doesn't have the required function.

Why is /usr/lib/x86_64-linux-gnu/ listed first?

This is because in ./configure we first check for MySQL things:

$ mariadb_config --libs  
-L/usr/lib/x86_64-linux-gnu/ -lmariadb 

remember those and later, the libcurl things:

$ /usr/local/bin/curl-config --libs  
-L/usr/local/lib -lcurl

which results in the following string:

-L/usr/lib/x86_64-linux-gnu/ -L/usr/local/lib

If we just build the agent, it's all good, because we don't get that first part from MySQL config and correct cURL library is used (/usr/local/lib/libcurl.so):

$ ./configure --enable-agent --with-libcurl=/usr/local/bin/curl-config
[...]
Linker flags:                -L/usr/local/lib -rdynamic

$ make
[...]

$ echo $?
0
Comment by Oleksii Zagorskyi [ 2023 Dec 18 ]

Have a look to ZBX-23861.

Comment by dimir [ 2023 Dec 19 ]

Solution 1

The simplest solution is to specify the path to the desired library to use manually in LDFLAGS:

$ LDFLAGS="/usr/local/lib/libcurl.so" ./configure --enable-server --with-mysql --with-libcurl=/usr/local/bin/curl-config

Solution 2

More complex solution would require to rewrite the way we work with pkg-config and generate direct path to the libraries instead of using a combination of -L<dir> and -l<lib>. E. g. if <pkg-config utility> --libs returns

-L/usr/local/lib -lcurl

translate this to

/usr/local/lib/libcurl.so

Solution 3

Tell user to remove the older library installed in standard path.

Comment by dimir [ 2023 Dec 19 ]

I vote for Solution #1. User still has control over compilation process even in this corner case. Perhaps we could even document this.

Comment by dimir [ 2023 Dec 20 ]

Was decided with sasha to go for #1 and document the workaround.

Comment by dimir [ 2023 Dec 20 ]

It appears that there is a possibility that user might be unaware that Zabbix gets compiled with older library even when asked to use the newer one.

Let's document the workaround but keep this one open until proper solution.

Follow up on documentation: ZBX-23873

Comment by Alex Kalimulin [ 2023 Dec 22 ]

This workaround:

LDFLAGS="/usr/local/lib/libcurl.so" ./configure --enable-server --with-mysql --with-libcurl=/usr/local/bin/curl-config

may not work because LDFLAGS are placed before the list of server files and libraries.

This one works:

LIBCURL_LIBS="/usr/local/lib/libcurl.so" ./configure --enable-server --with-mysql --with-libcurl=/usr/local/bin/curl-config

Because it disables curl-config --libs logic in configure and injects the library directly into the right place.

<dimir> I have checked the cc command that links zabbix_server binary and it's absolutely identical in my and your cases. Though, it works in my case but fails in yours.

But your version also works in my case. Looks like it's more reliable as it completely overrides LIBCURL_LIBS:

Comment by dimir [ 2024 Jan 05 ]

Proposed workaround

$ LDFLAGS="-Wl,--no-as-needed /usr/local/lib/libcurl.so" ./configure --enable-server --with-mysql --with-libcurl=/usr/local/bin/curl-config

will be documented in ZBX-23873

 

Generated at Fri Apr 04 12:17:07 EEST 2025 using Jira 9.12.4#9120004-sha1:625303b708afdb767e17cb2838290c41888e9ff0.