File Coverage

blib/lib/Net/APNS/Simple.pm
Criterion Covered Total %
statement 57 91 62.6
branch 6 24 25.0
condition 3 20 15.0
subroutine 15 18 83.3
pod 2 3 66.6
total 83 156 53.2


line stmt bran cond sub pod time code
1             package Net::APNS::Simple;
2 3     3   136926 use 5.008001;
  3         26  
3 3     3   16 use strict;
  3         5  
  3         71  
4 3     3   16 use warnings;
  3         5  
  3         80  
5 3     3   14 use Carp ();
  3         6  
  3         65  
6 3     3   1927 use JSON;
  3         32904  
  3         15  
7 3     3   1964 use Moo;
  3         34047  
  3         13  
8 3     3   5907 use Protocol::HTTP2::Client;
  3         117001  
  3         113  
9 3     3   1494 use IO::Select;
  3         5056  
  3         146  
10 3     3   2489 use IO::Socket::SSL qw();
  3         258748  
  3         3582  
11              
12             our $VERSION = "0.05";
13              
14             has [qw/auth_key key_id team_id bundle_id development/] => (
15             is => 'rw',
16             );
17              
18             has [qw/cert_file key_file passwd_cb/] => (
19             is => 'rw',
20             );
21              
22             has [qw/proxy/] => (
23             is => 'rw',
24             default => $ENV{https_proxy},
25             );
26              
27             has [qw/apns_id apns_expiration apns_collapse_id/] => (
28             is => 'rw',
29             );
30              
31             has apns_priority => (
32             is => 'rw',
33             default => 10,
34             );
35              
36 2     2 0 6791 sub algorithm {'ES256'}
37              
38             sub _host {
39 3     3   8 my ($self) = @_;
40 3 100       40 return 'api.' . ($self->development ? 'sandbox.' : '') . 'push.apple.com'
41             }
42              
43 2     2   20 sub _port {443}
44              
45             sub _socket {
46 0     0   0 my ($self) = @_;
47 0 0 0     0 if (!$self->{_socket} || !$self->{_socket}->opened){
48 0         0 my %ssl_opts = (
49             # openssl 1.0.1 support only NPN
50             SSL_npn_protocols => ['h2'],
51             # openssl 1.0.2 also have ALPN
52             SSL_alpn_protocols => ['h2'],
53             SSL_version => 'TLSv1_2',
54             );
55 0         0 for (qw/cert_file key_file passwd_cb/) {
56 0 0       0 $ssl_opts{"SSL_$_"} = $self->{$_} if defined $self->{$_};
57             }
58              
59 0         0 my ($host,$port) = ($self->_host, $self->_port);
60              
61 0         0 my $socket;
62 0 0       0 if ( my $proxy = $self->proxy ) {
63 0 0       0 $proxy =~ s|^http://|| or die "Invalid proxy $proxy - only http proxy is supported!\n";
64 0         0 require Net::HTTP;
65 0   0     0 $socket = Net::HTTP->new(PeerAddr => $proxy) || die $@;
66 0         0 $socket->write_request(
67             CONNECT => "$host:$port",
68             Host => "$host:$port",
69             Connection => "Keep-Alive",
70             'Proxy-Connection' => "Keep-Alive",
71             );
72 0         0 my ($code, $mess, %h) = $socket->read_response_headers;
73 0 0       0 $code eq '200' or die "Proxy error: $code $mess";
74              
75 0 0 0     0 IO::Socket::SSL->start_SSL(
76             $socket,
77             # explicitly set hostname we should use for SNI
78             SSL_hostname => $host,
79             %ssl_opts,
80             ) or die $! || $IO::Socket::SSL::SSL_ERROR;
81             }
82             else {
83             # TLS transport socket
84 0 0 0     0 $socket = IO::Socket::SSL->new(
85             PeerHost => $host,
86             PeerPort => $port,
87             %ssl_opts,
88             ) or die $! || $IO::Socket::SSL::SSL_ERROR;
89             }
90 0         0 $self->{_socket} = $socket;
91              
92             # non blocking
93 0         0 $self->{_socket}->blocking(0);
94             }
95 0         0 return $self->{_socket};
96             }
97              
98             sub _client {
99 1     1   3 my ($self) = @_;
100 1   33     16 $self->{_client} ||= Protocol::HTTP2::Client->new(keepalive => 1);
101 1         110 return $self->{_client};
102             }
103              
104             sub prepare {
105 1     1 1 2589 my ($self, $device_token, $payload, $cb) = @_;
106 1         9 my @headers = (
107             'apns-topic' => $self->bundle_id,
108             );
109              
110 1         4 for (qw/apns_id apns_priority apns_expiration apns_collapse_id/) {
111 4         21 my $v = $self->$_;
112 4 100       13 next unless defined $v;
113 2         3 my $k = $_;
114 2         7 $k =~ s/_/-/g;
115 2         6 push @headers, $k => $v;
116             }
117              
118 1 50 33     11 if ($self->team_id and $self->auth_key and $self->key_id) {
      33        
119 1         631 require Crypt::PK::ECC;
120             # require for treat pkcs#8 private key
121 1         18421 Crypt::PK::ECC->VERSION(0.059);
122 1         629 require Crypt::JWT;
123 1         28443 my $claims = {
124             iss => $self->team_id,
125             iat => time,
126             };
127 1         9 my $jwt = Crypt::JWT::encode_jwt(
128             payload => $claims,
129             key => [$self->auth_key],
130             alg => $self->algorithm,
131             extra_headers => {
132             kid => $self->key_id,
133             },
134             );
135 1         9393 push @headers, authorization => sprintf('bearer %s', $jwt);
136             }
137 1         7 my $path = sprintf '/3/device/%s', $device_token;
138 1         2 push @{$self->{_request}}, {
  1         7  
139             ':scheme' => 'https',
140             ':authority' => join(":", $self->_host, $self->_port),
141             ':path' => $path,
142             ':method' => 'POST',
143             headers => \@headers,
144             data => JSON::encode_json($payload),
145             on_done => $cb,
146             };
147 1         10 return $self;
148             }
149              
150             sub _make_client_request_single {
151 1     1   4 my ($self) = @_;
152 1 50       2 if (my $req = shift @{$self->{_request}}){
  1         5  
153 1         4 my $done_cb = delete $req->{on_done};
154             $self->_client->request(
155             %$req,
156             on_done => sub {
157 0 0   0     ref $done_cb eq 'CODE'
158             and $done_cb->(@_);
159 0           $self->_make_client_request_single();
160             },
161 1         5 );
162             }
163             else {
164 0           $self->_client->close;
165             }
166             }
167              
168             sub notify {
169 0     0 1   my ($self) = @_;
170             # request one by one as APNS server returns SETTINGS_MAX_CONCURRENT_STREAMS = 1
171 0           $self->_make_client_request_single();
172 0           my $io = IO::Select->new($self->_socket);
173             # send/recv frames until request is done
174 0           while ( !$self->_client->shutdown ) {
175 0           $io->can_write;
176 0           while ( my $frame = $self->_client->next_frame ) {
177 0           syswrite $self->_socket, $frame;
178             }
179 0           $io->can_read;
180 0           while ( sysread $self->_socket, my $data, 4096 ) {
181 0           $self->_client->feed($data);
182             }
183             }
184 0           undef $self->{_client};
185 0           $self->_socket->close(SSL_ctx_free => 1);
186             }
187              
188             1;
189             __END__