File Coverage

blib/lib/Blockchain/Contract/Solidity/ABI/Type.pm
Criterion Covered Total %
statement 115 117 98.2
branch 33 34 97.0
condition 8 8 100.0
subroutine 27 29 93.1
pod 3 22 13.6
total 186 210 88.5


line stmt bran cond sub pod time code
1             package Blockchain::Contract::Solidity::ABI::Type;
2              
3 5     5   60 use v5.26;
  5         20  
4 5     5   27 use strict;
  5         11  
  5         101  
5 5     5   23 use warnings;
  5         10  
  5         137  
6 5     5   26 no indirect;
  5         11  
  5         25  
7              
8 5     5   251 use Carp;
  5         19  
  5         310  
9 5     5   2322 use Module::Load;
  5         5737  
  5         29  
10 5     5   320 use constant NOT_IMPLEMENTED => 'Method not implemented';
  5         10  
  5         10862  
11              
12             sub new {
13 164     164 0 561 my ($class, %params) = @_;
14              
15 164         392 my $self = bless {}, $class;
16 164         466 $self->{signature} = $params{signature};
17 164         323 $self->{data} = $params{data};
18              
19 164         512 $self->configure();
20              
21 164         778 return $self;
22             }
23              
24       111 0   sub configure { }
25              
26             sub encode {
27 0     0 0 0 croak NOT_IMPLEMENTED;
28             }
29              
30             sub decode {
31 0     0 0 0 croak NOT_IMPLEMENTED;
32             }
33              
34             sub static {
35 663   100 663 0 2131 return shift->{static} //= [];
36             }
37              
38             sub push_static {
39 122     122 0 232 my ($self, $data) = @_;
40 122 100       235 push($self->static->@*, ref $data eq 'ARRAY' ? $data->@* : $data);
41             }
42              
43             sub dynamic {
44 607   100 607 0 1815 return shift->{dynamic} //= [];
45             }
46              
47             sub push_dynamic {
48 66     66 0 126 my ($self, $data) = @_;
49 66 100       119 push($self->dynamic->@*, ref $data eq 'ARRAY' ? $data->@* : $data);
50             }
51              
52             sub signature {
53 733     733 1 4320 return shift->{signature};
54             }
55              
56             sub data {
57 234     234 1 1328 return shift->{data};
58             }
59              
60             sub fixed_length {
61 42     42 0 104 my $self = shift;
62 42 100       88 if ($self->signature =~ /[a-z](\d+)/) {
63 28         129 return $1;
64             }
65 14         43 return undef;
66             }
67              
68             sub pad_right {
69 16     16 0 42 my ($self, $data) = @_;
70              
71 16         27 my @chunks;
72 16         147 push(@chunks, $_ . '0' x (64 - length $_)) for unpack("(A64)*", $data);
73              
74 16         63 return \@chunks;
75             }
76              
77             sub pad_left {
78 42     42 0 3395 my ($self, $data) = @_;
79              
80 42         68 my @chunks;
81 42         285 push(@chunks, sprintf("%064s", $_)) for unpack("(A64)*", $data);
82              
83 42         190 return \@chunks;
84              
85             }
86              
87             sub encode_length {
88 23     23 0 54 my ($self, $length) = @_;
89 23         150 return sprintf("%064s", sprintf("%x", $length));
90             }
91              
92             sub encode_offset {
93 25     25 0 71 my ($self, $offset) = @_;
94 25         133 return sprintf("%064s", sprintf("%x", $offset * 32));
95             }
96              
97             sub encoded {
98 541     541 0 850 my $self = shift;
99 541         964 my @data = ($self->static->@*, $self->dynamic->@*);
100 541 100       1923 return scalar @data ? \@data : undef;
101             }
102              
103             sub is_dynamic {
104 359 100   359 0 719 return shift->signature =~ /(bytes|string)(?!\d+)|(\[\])/ ? 1 : 0;
105             }
106              
107             sub new_type {
108 144     144 1 443 my (%params) = @_;
109              
110 144         286 my $signature = $params{signature};
111              
112 144         225 my $module;
113 144 100       1077 if ($signature =~ /\[(\d+)?\]$/gm) {
    100          
    100          
    100          
    100          
    100          
114 20         48 $module = "Array";
115             } elsif ($signature =~ /^\(.*\)/) {
116 12         27 $module = "Tuple";
117             } elsif ($signature =~ /^address$/) {
118 22         60 $module = "Address";
119             } elsif ($signature =~ /^(u)?(int|bool)(\d+)?$/) {
120 63         131 $module = "Int";
121             } elsif ($signature =~ /^(?:bytes)(\d+)?$/) {
122 21         47 $module = "Bytes";
123             } elsif ($signature =~ /^string$/) {
124 5         13 $module = "String";
125             } else {
126 1         92 croak "Module not found for the given parameter signature $signature";
127             }
128              
129             # this is just to avoid `use module` for every new type included
130 143         238 my $_package = __PACKAGE__;
131 143         493 my $package = sprintf("%s::%s", $_package, $module);
132 143         476 load $package;
133             return $package->new(
134             signature => $signature,
135 143         8061 data => $params{data});
136             }
137              
138             sub instances {
139 391   100 391 0 7994 return shift->{instances} //= [];
140             }
141              
142             sub get_initial_offset {
143 38     38 0 72 my $self = shift;
144 38         68 my $offset = 0;
145 38         82 for my $param ($self->instances->@*) {
146 88         284 my $encoded = $param->encode;
147 82 100       221 if ($param->is_dynamic) {
148 25         96 $offset += 1;
149             } else {
150 57         122 $offset += scalar $param->encoded->@*;
151             }
152             }
153              
154 32         109 return $offset;
155             }
156              
157             sub static_size {
158 66     66 0 141 return 1;
159             }
160              
161             sub read_stack_set_data {
162 14     14 0 26 my $self = shift;
163              
164 14         31 my @data = $self->data->@*;
165 14         30 my @offsets;
166 14         27 my $current_offset = 0;
167              
168             # Since at this point we don't information about the chunks of data it is_dynamic
169             # needed to get all the offsets in the static header, so the dynamic values can
170             # be retrieved based in between the current and the next offsets
171 14         30 for my $instance ($self->instances->@*) {
172 39 100       89 if ($instance->is_dynamic) {
173 8         30 push @offsets, hex($data[$current_offset]) / 32;
174             }
175              
176 39         108 my $size = 1;
177 39 100       71 $size = $instance->static_size unless $instance->is_dynamic;
178 39         83 $current_offset += $size;
179             }
180              
181 14         28 $current_offset = 0;
182 14         20 my %response;
183             # Dynamic data must to be set first since the full_size method
184             # will need to use the data offset related to the size of the item
185 14         37 for (my $i = 0; $i < $self->instances->@*; $i++) {
186 39         74 my $instance = $self->instances->[$i];
187 39 100       69 next unless $instance->is_dynamic;
188 8         21 my $offset_start = shift @offsets;
189 8   100     38 my $offset_end = $offsets[0] // scalar @data - 1;
190 8         35 my @range = @data[$offset_start .. $offset_end];
191 8         19 $instance->{data} = \@range;
192 8         16 $current_offset += scalar @range;
193 8         28 $response{$i} = $instance->decode();
194             }
195              
196 14         31 $current_offset = 0;
197              
198 14         33 for (my $i = 0; $i < $self->instances->@*; $i++) {
199 39         78 my $instance = $self->instances->[$i];
200              
201 39 100       79 if ($instance->is_dynamic) {
202 8         13 $current_offset++;
203 8         21 next;
204             }
205              
206 31         56 my $size = 1;
207 31 50       57 $size = $instance->static_size unless $instance->is_dynamic;
208 31         114 my @range = @data[$current_offset .. $current_offset + $size - 1];
209 31         69 $instance->{data} = \@range;
210 31         46 $current_offset += $size;
211              
212 31         118 $response{$i} = $instance->decode();
213             }
214              
215 14         22 my @array_response;
216             # the given order of type signatures needs to be strict followed
217 14         29 push(@array_response, $response{$_}) for 0 .. scalar $self->instances->@* - 1;
218 14         75 return \@array_response;
219             }
220              
221             1;
222              
223             __END__
224              
225             =pod
226              
227             =encoding UTF-8
228              
229             =head1 NAME
230              
231             Blockchain::Contract::Solidity::ABI::Type - Interface for solidity variable types
232              
233             =head1 SYNOPSIS
234              
235             Allows you to define and instantiate a solidity variable type:
236              
237             my $type = Blockchain::Contract::Solidity::ABI::Type::new_type(
238             signature => $signature,
239             data => $value
240             );
241              
242             $type->encode();
243             ...
244              
245             In most cases you don't want to use this directly, use instead:
246              
247             =over 4
248              
249             =item * B<Encoder>: L<Blockchain::Contract::Solidity::ABI::Encoder>
250              
251             =item * B<Decoder>: L<Blockchain::Contract::Solidity::ABI::Decoder>
252              
253             =back
254              
255             =head1 METHODS
256              
257             =head2 new_type
258              
259             Create a new L<Blockchain::Contract::Solidity::ABI::Type> instance based
260             in the given signature.
261              
262             Usage:
263              
264             new_type(signature => signature, data => value) -> L<Blockchain::Contract::Solidity::ABI::Type::*>
265              
266             =over 4
267              
268             =item * C<%params> signature and data key values
269              
270             =back
271              
272             Returns an new instance of one of the sub modules for L<Blockchain::Contract::Solidity::ABI::Type>
273              
274             =head1 AUTHOR
275              
276             Reginaldo Costa, C<< <refeco at cpan.org> >>
277              
278             =head1 BUGS
279              
280             Please report any bugs or feature requests to L<https://github.com/refeco/perl-ABI>
281              
282             =head1 SUPPORT
283              
284             You can find documentation for this module with the perldoc command.
285              
286             perldoc Blockchain::Contract::Solidity::ABI::Type
287              
288             =head1 LICENSE AND COPYRIGHT
289              
290             This software is Copyright (c) 2022 by REFECO.
291              
292             This is free software, licensed under:
293              
294             The MIT License
295              
296             =cut
297