Исправление ошибки IO::Socket::SSL 1.56 must be installed for https support

В языке программирования Perl есть несколько библиотек с помощью которых можно загружать данные из интернета по протоколу HTTP. Саме известные — это LWP и HTTP::Tiny. Библиотека HTTP::Tiny поставляется вместе с Perl, ее не нужно устанавливать дополнительно, поэтому часто удобно скачивать данные с ее помощью.

Успешная загрузка HTML страницы по протоколу http

Вот пример кода, который загружает главную страницу сайта http://example.com. В этом примере используется протокол http, а не https:

#!/usr/bin/perl

use strict;
use warnings;
use feature qw(say);

use HTTP::Tiny;
use Data::Dumper;

my $response = HTTP::Tiny->new()->get('http://example.com');

warn Dumper $response;

Если запустить этот код, то видно что все работает. Получили статус 200. У поля success значение является истиной (в этом тексте для удобства отображения структуры данных у поля content было изменено значение, вместо html кода страницы написаны три точки):

$ perl script.pl
$VAR1 = {
          'reason' => 'OK',
          'protocol' => 'HTTP/1.1',
          'content' => '...',
          'headers' => {
                         'etag' => '"3147526947+gzip+ident"',
                         'x-cache' => 'HIT',
                         'connection' => 'close',
                         'cache-control' => 'max-age=604800',
                         'content-length' => '1256',
                         'server' => 'ECS (nyb/1D2A)',
                         'expires' => 'Mon, 30 Dec 2019 12:38:04 GMT',
                         'vary' => 'Accept-Encoding',
                         'content-type' => 'text/html; charset=UTF-8',
                         'date' => 'Mon, 23 Dec 2019 12:38:04 GMT',
                         'last-modified' => 'Thu, 17 Oct 2019 07:18:26 GMT'
                       },
          'url' => 'http://example.com',
          'success' => 1,
          'status' => '200'
        };

Ошибка при использовании протокола https

Но если немного изменить этот код и загружать данные по протоколу https, то данные не загрузятся. Вот код программы:

#!/usr/bin/perl

use strict;
use warnings;
use feature qw(say);

use HTTP::Tiny;
use Data::Dumper;

my $response = HTTP::Tiny->new()->get('https://example.com');

warn Dumper $response;

Вот результат работы этой программы:

$ perl script.pl
$VAR1 = {
          'url' => 'https://example.com',
          'reason' => 'Internal Exception',
          'status' => 599,
          'success' => '',
          'headers' => {
                         'content-type' => 'text/plain',
                         'content-length' => 57
                       },
          'content' => 'IO::Socket::SSL 1.56 must be installed for https support
'
        };

Значение поля status 599 (это специальный статус придуманный в библиотеке HTTP::Tiny, он означает что проблема с самой библиотекой). Значение success ложно, а в поле content есть пояснение из-за чего произошла ошибка: IO::Socket::SSL 1.56 must be installed for https support.

Попытка решения

Ага. Чтобы HTTP::Tiny мог работать по протоколу https нужно чтобы была библиотека IO::Socket::SSL.

Пытаемся поставить ее с помощью cpanm, но получаем ошибку:

root@faf6a4b66b08:/app# cpanm IO::Socket::SSL
--> Working on IO::Socket::SSL
Fetching http://www.cpan.org/authors/id/S/SU/SULLR/IO-Socket-SSL-2.066.tar.gz ... OK
==> Found dependencies: Net::SSLeay
--> Working on Net::SSLeay
Fetching http://www.cpan.org/authors/id/C/CH/CHRISN/Net-SSLeay-1.88.tar.gz ... OK
Configuring Net-SSLeay-1.88 ... OK
Building and testing Net-SSLeay-1.88 ... FAIL
! Installing Net::SSLeay failed. See /root/.cpanm/work/1577105006.3284/build.log for details. Retry with --force to force install it.
! Installing the dependencies failed: Module 'Net::SSLeay' is not installed
! Bailing out the installation for IO-Socket-SSL-2.066.
root@faf6a4b66b08:/app#

Как видно из лога для библиотеки IO::Socket::SSL нужно установить Net::SSLeay, но Net::SSLeay не устанавливается. В файле build.log можно увидеть подробности. Вот фрагмент этого файла:

cp lib/Net/SSLeay.pod blib/lib/Net/SSLeay.pod
/usr/bin/perl /usr/share/perl/5.18/ExtUtils/xsubpp  -typemap /usr/share/perl/5.18/ExtUtils/typemap -typemap typemap  SSLeay.xs > SSLeay.xsc && mv SSLeay.xsc SSLeay.c
cc -c   -D_REENTRANT -D_GNU_SOURCE -DDEBIAN -fstack-protector -fno-strict-aliasing -pipe -I/usr/local/include -D_LARGEFILE_SOURCE -D_FILE_OFFSET_BITS=64 -O2 -g   -DVERSION=\"1.88\" -DXS_VERSION=\"1.88\" -fPIC "-I/usr/lib/perl/5.18/CORE"   SSLeay.c
SSLeay.xs:163:25: fatal error: openssl/err.h: No such file or directory
 #include  openssl/err.h
                         ^
compilation terminated.
make: *** [SSLeay.o] Error 1

Успешное решение

Чтобы HTTP::Tiny мог работать по протоколу https нужно установить библиотеку IO::Socket::SSL. Чтобы установить IO::Socket::SSL нужно установить Net::SSLeay. А вот чтобы установить библиотеку Net::SSLeay нужно поставить дополнительную библиотеку в систему.

На Ubuntu это делается с помощью команды:

$ apt-get update && apt-get install -y libssl-dev

После этого нужно установить IO::Socket::SSL:

# cpanm IO::Socket::SSL

Эта команда успешно установит библиотеки Net::SSLeay и IO::Socket::SSL. И после этого HTTP::Tiny сможет работать по протоколу https.

Другие статьи