AnyEvent-ReverseHTTP
view release on metacpan or search on metacpan
lib/AnyEvent/ReverseHTTP.pm view on Meta::CPAN
use Scalar::Util;
use base qw(Exporter);
our @EXPORT = qw(reverse_http);
use Any::Moose;
has endpoint => (
is => 'rw', isa => 'Str',
required => 1, default => "http://www.reversehttp.net/reversehttp",
);
has label => (
is => 'rw', isa => 'Str',
required => 1,
lazy => 1, default => sub {
require Digest::SHA;
require Time::HiRes;
return Digest::SHA::sha1_hex($$ . Time::HiRes::gettimeofday() . {});
},
);
has token => (
is => 'rw', isa => 'Str',
default => '-',
);
has on_register => (
is => 'rw', isa => 'CodeRef',
default => sub { sub { warn "Public Application URL: $_[0]\n" } },
);
has on_error => (
is => 'rw', isa => 'CodeRef',
default => sub { sub { Carp::croak(@_) } },
);
has on_request => (
is => 'rw', isa => 'CodeRef',
default => sub { sub { Carp::croak("on_request handler is not defined!") } },
);
sub reverse_http {
my $cb = pop;
my @args =
@_ == 1 ? qw(label) :
@_ == 2 ? qw(label token) :
@_ >= 3 ? qw(endpoint label token) : ();
my %args; @args{@args} = @_;
return __PACKAGE__->new(%args, on_request => $cb)->connect;
}
sub connect {
my $self = shift;
my %query = (name => $self->label);
$query{token} = $self->token if $self->token;
my $body = join "&", map "$_=" . URI::Escape::uri_escape($query{$_}), keys %query;
http_post $self->endpoint, $body, sub {
my($body, $hdr) = @_;
if ($hdr->{Status} eq '201' || $hdr->{Status} eq '204') {
my $app_url = _extract_link($hdr, 'related');
$self->on_register->($app_url);
} else {
return $self->on_error->("$hdr->{Status}: $hdr->{Reason}");
}
my $poller; $poller = sub {
my($body, $hdr) = @_;
if ($hdr->{Status} eq '200') {
my $req = HTTP::Request->parse($body);
$req->header('Requesting-Client', $hdr->{'requesting-client'});
my $res = $self->on_request->($req);
my $postback = sub {
my $res = shift;
# Duck typing for as_string, but accepts plaintext too for 200
unless (Scalar::Util::blessed($res) && $res->can('as_string')) {
my $content = $res;
$res = HTTP::Response->new(200);
$res->content_type('text/plain');
$res->content($content);
}
$res->protocol("HTTP/1.1"); # Upgrade since reversehttp.net requires so
# HTTP::Response->as_string by default adds a new line which could be harmful
my $res_body = $res->as_string;
chomp $res_body if $res->content_type eq 'text/plain';
http_post $hdr->{URL}, $res_body,
headers => { 'content-type' => 'message/http' },
sub {
my($body, $hdr) = @_;
if ($hdr->{Status} ne '202') {
$self->on_error->("$hdr->{Status}: $hdr->{Reason}");
}
};
};
# Return condvar to pass back to event loop
if (Scalar::Util::blessed($res) && $res->isa('AnyEvent::CondVar')) {
$res->cb(sub { $postback->($res->recv) });
} else {
$postback->($res);
}
}
my $next = _extract_link($hdr, 'next');
http_get $next, $poller;
};
my $url = _extract_link($hdr, 'first');
http_get $url, $poller;
( run in 1.264 second using v1.01-cache-2.11-cpan-5b529ec07f3 )