package LWP::UserAgent::Patch::HTTPSHardTimeout;
use 5.010001;
use strict;
no warnings;
use Log::Any '$log';
use Module::Patch 0.12 qw();
use base qw(Module::Patch);
our $VERSION = '0.04'; # VERSION
our %config;
my $p_send_request = sub {
my $ctx = shift;
my $orig = $ctx->{orig};
my ($self, $request, $arg, $size) = @_;
my $url = $request->uri;
my $scheme = $url->scheme;
if ($scheme eq 'https' && $config{-timeout} > 0) {
my $resp;
eval {
$log->tracef("Wrapping send_request() with alarm timeout (%d)",
$config{-timeout});
local $SIG{ALRM} = sub { die "alarm\n" };
alarm $config{-timeout};
$resp = $orig->(@_);
alarm 0;
};
if ($@) {
die unless $@ eq "alarm\n";
return LWP::UserAgent::_new_response(
$request, &HTTP::Status::RC_HTTP_REQUEST_TIMEOUT, $@);
} else {
return $resp;
}
} else {
return $orig->(@_);
}
};
sub patch_data {
return {
v => 3,
config => {
-timeout => {
schema => 'int*',
default => 3600,
},
},
patches => [
{
action => 'wrap',
mod_version => qr/^6\.0.+/,
sub_name => 'send_request',
code => $p_send_request,
},
],
};
}
1;
# ABSTRACT: Add hard timeout to HTTPS requests
__END__
=pod
=head1 NAME
LWP::UserAgent::Patch::HTTPSHardTimeout - Add hard timeout to HTTPS requests
=head1 VERSION
version 0.04
=head1 SYNOPSIS
use LWP::UserAgent::Patch::HTTPSHardTimeout -timeout => 300;
=head1 DESCRIPTION
This module contains a simple workaround for hanging issue with HTTPS requests.
It wraps send_request() with an alarm() timeout.
=head1 FAQ
=head2 Why not subclass?
By patching, you do not need to replace all the client code which uses
LWP::UserAgent (or WWW::Mechanize, and so on).
=head1 SEE ALSO
http://stackoverflow.com/questions/9400068/make-timeout-work-for-lwpuseragent-https
L<LWPx::ParanoidAgent>
=head1 AUTHOR
Steven Haryanto <stevenharyanto@gmail.com>
=head1 COPYRIGHT AND LICENSE
This software is copyright (c) 2012 by Steven Haryanto.
This is free software; you can redistribute it and/or modify it under
the same terms as the Perl 5 programming language system itself.
=cut