Poprawka błędu IO::Socket::SSL 1.56 must be installed for https support

W języku programowania Perl jest kilka bibliotek za pomocą których można pobierać dane z internetu za pośrednictwem protokołu HTTP. Mianowicie znane — to LWP i HTTP::Tiny. Biblioteka HTTP::Tiny jest dostarczany wraz z Perl, jej nie trzeba instalować dodatkowo, dlatego często wygodnie pobierać dane z jej pomocą.

Wczytywanie HTML strony protokołu http

Oto przykład kodu, który pobiera główną stronę witryny http://example.com. W tym przykładzie wykorzystywany jest protokół http, a nie 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;

Jeśli uruchomić ten kod, to widać, że wszystko działa. Otrzymałeś status 200. U pola successwartość jest prawdą (w tym tekście dla ułatwienia wyświetlania struktury danych pola content wartość została zmieniona, zamiast kodu html strony są napisane trzy punkty):

$ 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'
        };

Błąd podczas korzystania z protokołu https

Ale jeśli trochę zmienić ten kod i pobierać dane za pomocą protokołu https, dane nie zostaną załadowane. Oto kod programu:

#!/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;

Oto wynik działania tego programu:

$ 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
'
        };

Wartość pola status 599 (jest to specjalny status wymyślone w bibliotece HTTP::Tiny, oznacza to, że problem z samą biblioteką). Wartość success fałszywie, a w polu content jest wyjaśnienie, o co wystąpił błąd: IO::Socket::SSL 1.56 must be installed for https support.

Próba rozwiązania

Tak. Aby HTTP::Tiny mógł działać za pośrednictwem protokołu https trzeba aby była biblioteka IO::Socket::SSL.

Staramy się dostarczać jej z pomocą cpanm, ale mamy błąd:

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#

Jak wynika z logu dla biblioteki IO::Socket::SSL należy ustawić Net::SSLeay, ale Net::SSLeay nie jest zainstalowany. W pliku build.log można zobaczyć szczegóły. Oto fragment tego pliku:

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

Pomyślne rozwiązanie

Aby HTTP::Tiny może pracować za pomocą protokołu https należy zainstalować bibliotekę IO::Socket::SSL. Aby zainstalować IO::Socket::SSL należy ustawić Net::SSLeay. A oto aby zainstalować bibliotekę Net::SSLeay należy umieścić dodatkowe biblioteki do systemu.

Na Ubuntu to zrobić za pomocą polecenia:

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

Wtedy trzeba ustawić IO::Socket::SSL:

# cpanm IO::Socket::SSL

Zespół ten z powodzeniem instaluje biblioteki Net::SSLeay i IO::Socket::SSL. I wtedy HTTP::Tiny może działać za pośrednictwem protokołu https.

Inne artykuły