line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
1
|
|
|
|
|
|
|
package HAL::Resource; |
2
|
10
|
|
|
10
|
|
107
|
use Moo; |
|
10
|
|
|
|
|
24
|
|
|
10
|
|
|
|
|
91
|
|
3
|
10
|
|
|
10
|
|
3824
|
use JSON 'decode_json'; |
|
10
|
|
|
|
|
21
|
|
|
10
|
|
|
|
|
89
|
|
4
|
10
|
|
|
10
|
|
1342
|
use Filter::signatures; |
|
10
|
|
|
|
|
27
|
|
|
10
|
|
|
|
|
89
|
|
5
|
10
|
|
|
10
|
|
327
|
no warnings 'experimental::signatures'; |
|
10
|
|
|
|
|
28
|
|
|
10
|
|
|
|
|
450
|
|
6
|
10
|
|
|
10
|
|
76
|
use feature 'signatures'; |
|
10
|
|
|
|
|
23
|
|
|
10
|
|
|
|
|
1001
|
|
7
|
10
|
|
|
10
|
|
7051
|
use Future; |
|
10
|
|
|
|
|
103618
|
|
|
10
|
|
|
|
|
391
|
|
8
|
|
|
|
|
|
|
|
9
|
10
|
|
|
10
|
|
83
|
use Carp qw(croak); |
|
10
|
|
|
|
|
25
|
|
|
10
|
|
|
|
|
6499
|
|
10
|
|
|
|
|
|
|
|
11
|
|
|
|
|
|
|
our $VERSION = '0.57'; |
12
|
|
|
|
|
|
|
|
13
|
|
|
|
|
|
|
=head1 NAME |
14
|
|
|
|
|
|
|
|
15
|
|
|
|
|
|
|
HAL::Resource - wrap a HAL resource |
16
|
|
|
|
|
|
|
|
17
|
|
|
|
|
|
|
=head1 SYNOPSIS |
18
|
|
|
|
|
|
|
|
19
|
|
|
|
|
|
|
my $ua = WWW::Mechanize->new(); |
20
|
|
|
|
|
|
|
my $res = $ua->get('https://api.example.com/'); |
21
|
|
|
|
|
|
|
my $r = HAL::Resource->new( |
22
|
|
|
|
|
|
|
ua => $ua, |
23
|
|
|
|
|
|
|
%{ decode_json( $res->decoded_content ) }, |
24
|
|
|
|
|
|
|
); |
25
|
|
|
|
|
|
|
|
26
|
|
|
|
|
|
|
=head1 ABOUT |
27
|
|
|
|
|
|
|
|
28
|
|
|
|
|
|
|
This module is just a very thin wrapper for HAL resources. If you find this |
29
|
|
|
|
|
|
|
module useful, I'm very happy to spin it off into its own distribution. |
30
|
|
|
|
|
|
|
|
31
|
|
|
|
|
|
|
=head1 SEE ALSO |
32
|
|
|
|
|
|
|
|
33
|
|
|
|
|
|
|
L - similar to this module, but lacks a HTTP transfer facility and |
34
|
|
|
|
|
|
|
currently fails its test suite |
35
|
|
|
|
|
|
|
|
36
|
|
|
|
|
|
|
L - a module to generate HAL JSON |
37
|
|
|
|
|
|
|
|
38
|
|
|
|
|
|
|
L - an adapter to export DBIx::Class structures |
39
|
|
|
|
|
|
|
as HAL |
40
|
|
|
|
|
|
|
|
41
|
|
|
|
|
|
|
Hypertext Application Language - L |
42
|
|
|
|
|
|
|
|
43
|
|
|
|
|
|
|
=cut |
44
|
|
|
|
|
|
|
|
45
|
|
|
|
|
|
|
has ua => ( |
46
|
|
|
|
|
|
|
weaken => 1, |
47
|
|
|
|
|
|
|
is => 'ro', |
48
|
|
|
|
|
|
|
); |
49
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
has _links => ( |
51
|
|
|
|
|
|
|
is => 'ro', |
52
|
|
|
|
|
|
|
); |
53
|
|
|
|
|
|
|
|
54
|
|
|
|
|
|
|
has _external => ( |
55
|
|
|
|
|
|
|
is => 'ro', |
56
|
|
|
|
|
|
|
); |
57
|
|
|
|
|
|
|
|
58
|
|
|
|
|
|
|
has _embedded => ( |
59
|
|
|
|
|
|
|
is => 'ro', |
60
|
|
|
|
|
|
|
); |
61
|
|
|
|
|
|
|
|
62
|
29
|
|
|
29
|
0
|
116
|
sub resource_url( $self, $name ) { |
|
29
|
|
|
|
|
83
|
|
|
29
|
|
|
|
|
83
|
|
|
29
|
|
|
|
|
61
|
|
63
|
29
|
|
|
|
|
157
|
my $l = $self->_links; |
64
|
29
|
50
|
|
|
|
206
|
if( exists $l->{$name} ) { |
65
|
|
|
|
|
|
|
$l->{$name}->{href} |
66
|
29
|
|
|
|
|
275
|
} |
67
|
|
|
|
|
|
|
} |
68
|
|
|
|
|
|
|
|
69
|
0
|
|
|
0
|
0
|
0
|
sub resources( $self ) { |
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
70
|
0
|
|
|
|
|
0
|
sort keys %{ $self->_links } |
|
0
|
|
|
|
|
0
|
|
71
|
|
|
|
|
|
|
} |
72
|
|
|
|
|
|
|
|
73
|
18
|
|
|
18
|
0
|
45
|
sub fetch_resource_future( $self, $name, %options ) { |
|
18
|
|
|
|
|
42
|
|
|
18
|
|
|
|
|
51
|
|
|
18
|
|
|
|
|
53
|
|
|
18
|
|
|
|
|
59
|
|
74
|
18
|
|
66
|
|
|
202
|
my $class = $options{ class } || ref $self; |
75
|
18
|
|
|
|
|
116
|
my $ua = $self->ua; |
76
|
|
|
|
|
|
|
my $url = $self->resource_url( $name ) |
77
|
18
|
50
|
|
|
|
138
|
or croak "Couldn't find resource '$name' in " . join ",", sort keys %{$self->_links}; |
|
0
|
|
|
|
|
0
|
|
78
|
18
|
|
|
18
|
|
58
|
Future->done( $ua->get( $url ))->then( sub( $res ) { |
|
18
|
|
|
|
|
1556250
|
|
|
18
|
|
|
|
|
67
|
|
79
|
18
|
|
|
|
|
62
|
Future->done( bless { ua => $ua, %{ decode_json( $res->content )} } => $class ); |
|
18
|
|
|
|
|
177
|
|
80
|
18
|
|
|
|
|
206
|
}); |
81
|
|
|
|
|
|
|
} |
82
|
|
|
|
|
|
|
|
83
|
1
|
|
|
1
|
0
|
5
|
sub fetch_resource( $self, $name, %options ) { |
|
1
|
|
|
|
|
4
|
|
|
1
|
|
|
|
|
4
|
|
|
1
|
|
|
|
|
3
|
|
|
1
|
|
|
|
|
3
|
|
84
|
1
|
|
|
|
|
9
|
$self->fetch_resource_future( $name, %options )->get |
85
|
|
|
|
|
|
|
} |
86
|
|
|
|
|
|
|
|
87
|
3
|
|
|
3
|
0
|
17
|
sub navigate_future( $self, %options ) { |
|
3
|
|
|
|
|
9
|
|
|
3
|
|
|
|
|
10
|
|
|
3
|
|
|
|
|
7
|
|
88
|
3
|
|
33
|
|
|
12
|
$options{ class } ||= ref $self; |
89
|
3
|
|
50
|
|
|
16
|
my $path = delete $options{ path } || []; |
90
|
3
|
|
|
|
|
31
|
my $resource = Future->done( $self ); |
91
|
3
|
|
|
|
|
91
|
for my $item (@$path) { |
92
|
6
|
|
|
|
|
320
|
my $i = $item; |
93
|
6
|
|
|
6
|
|
31
|
$resource = $resource->then( sub( $r ) { |
|
6
|
|
|
|
|
535
|
|
|
6
|
|
|
|
|
14
|
|
94
|
6
|
|
|
|
|
44
|
$r->fetch_resource_future( $i, %options ); |
95
|
6
|
|
|
|
|
52
|
}); |
96
|
|
|
|
|
|
|
}; |
97
|
3
|
|
|
|
|
1573
|
$resource |
98
|
|
|
|
|
|
|
} |
99
|
|
|
|
|
|
|
|
100
|
3
|
|
|
3
|
0
|
9
|
sub navigate( $self, %options ) { |
|
3
|
|
|
|
|
9
|
|
|
3
|
|
|
|
|
12
|
|
|
3
|
|
|
|
|
7
|
|
101
|
3
|
|
|
|
|
18
|
$self->navigate_future( %options )->get |
102
|
|
|
|
|
|
|
} |
103
|
|
|
|
|
|
|
|
104
|
5
|
|
|
5
|
0
|
16
|
sub inflate_list( $self, $class, $list ) { |
|
5
|
|
|
|
|
13
|
|
|
5
|
|
|
|
|
13
|
|
|
5
|
|
|
|
|
10
|
|
|
5
|
|
|
|
|
10
|
|
105
|
5
|
|
|
|
|
23
|
my $ua = $self->ua; |
106
|
|
|
|
|
|
|
map { |
107
|
23
|
|
|
|
|
14056
|
$class->new( ua => $ua, %$_ ) |
108
|
5
|
|
|
|
|
11
|
} @{ $list }; |
|
5
|
|
|
|
|
65
|
|
109
|
|
|
|
|
|
|
} |
110
|
|
|
|
|
|
|
|
111
|
|
|
|
|
|
|
1; |
112
|
|
|
|
|
|
|
|
113
|
|
|
|
|
|
|
=head1 AUTHOR |
114
|
|
|
|
|
|
|
|
115
|
|
|
|
|
|
|
Max Maischein, Ecorion@cpan.orgE |
116
|
|
|
|
|
|
|
|
117
|
|
|
|
|
|
|
=head1 SEE ALSO |
118
|
|
|
|
|
|
|
|
119
|
|
|
|
|
|
|
L, L. |
120
|
|
|
|
|
|
|
|
121
|
|
|
|
|
|
|
=head1 REPOSITORY |
122
|
|
|
|
|
|
|
|
123
|
|
|
|
|
|
|
The public repository of this module is |
124
|
|
|
|
|
|
|
L. |
125
|
|
|
|
|
|
|
|
126
|
|
|
|
|
|
|
=head1 SUPPORT |
127
|
|
|
|
|
|
|
|
128
|
|
|
|
|
|
|
The public support forum of this module is |
129
|
|
|
|
|
|
|
L. |
130
|
|
|
|
|
|
|
|
131
|
|
|
|
|
|
|
=head1 BUG TRACKER |
132
|
|
|
|
|
|
|
|
133
|
|
|
|
|
|
|
Please report bugs in this module via the RT CPAN bug queue at |
134
|
|
|
|
|
|
|
L |
135
|
|
|
|
|
|
|
or via mail to L. |
136
|
|
|
|
|
|
|
|
137
|
|
|
|
|
|
|
=head1 COPYRIGHT (c) |
138
|
|
|
|
|
|
|
|
139
|
|
|
|
|
|
|
Copyright 2003-2019 by Max Maischein C. |
140
|
|
|
|
|
|
|
|
141
|
|
|
|
|
|
|
=head1 LICENSE |
142
|
|
|
|
|
|
|
|
143
|
|
|
|
|
|
|
This module is released under the same terms as Perl itself. |
144
|
|
|
|
|
|
|
|
145
|
|
|
|
|
|
|
=cut |