line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
1
|
|
|
|
|
|
|
package BankAccount::Validator::UK; |
2
|
|
|
|
|
|
|
|
3
|
|
|
|
|
|
|
$BankAccount::Validator::UK::VERSION = '0.63'; |
4
|
|
|
|
|
|
|
$BankAccount::Validator::UK::AUTHORITY = 'cpan:MANWAR'; |
5
|
|
|
|
|
|
|
|
6
|
|
|
|
|
|
|
=head1 NAME |
7
|
|
|
|
|
|
|
|
8
|
|
|
|
|
|
|
BankAccount::Validator::UK - Interface to validate UK bank account. |
9
|
|
|
|
|
|
|
|
10
|
|
|
|
|
|
|
=head1 VERSION |
11
|
|
|
|
|
|
|
|
12
|
|
|
|
|
|
|
Version 0.63 |
13
|
|
|
|
|
|
|
|
14
|
|
|
|
|
|
|
=cut |
15
|
|
|
|
|
|
|
|
16
|
3
|
|
|
3
|
|
164124
|
use 5.006; |
|
3
|
|
|
|
|
24
|
|
17
|
3
|
|
|
3
|
|
1675
|
use Data::Dumper; |
|
3
|
|
|
|
|
16658
|
|
|
3
|
|
|
|
|
175
|
|
18
|
3
|
|
|
3
|
|
1138
|
use BankAccount::Validator::UK::Rule; |
|
3
|
|
|
|
|
8
|
|
|
3
|
|
|
|
|
112
|
|
19
|
|
|
|
|
|
|
|
20
|
3
|
|
|
3
|
|
1409
|
use Moo; |
|
3
|
|
|
|
|
26895
|
|
|
3
|
|
|
|
|
13
|
|
21
|
3
|
|
|
3
|
|
4454
|
use namespace::autoclean; |
|
3
|
|
|
|
|
31618
|
|
|
3
|
|
|
|
|
12
|
|
22
|
|
|
|
|
|
|
|
23
|
|
|
|
|
|
|
has sc => (is => 'rw'); |
24
|
|
|
|
|
|
|
has an => (is => 'rw'); |
25
|
|
|
|
|
|
|
has mod => (is => 'rw'); |
26
|
|
|
|
|
|
|
has attempt => (is => 'rw'); |
27
|
|
|
|
|
|
|
has last_ex => (is => 'rw'); |
28
|
|
|
|
|
|
|
has trace => (is => 'rw'); |
29
|
|
|
|
|
|
|
has debug => (is => 'ro', default => sub { 0 }); |
30
|
|
|
|
|
|
|
has last_check => (is => 'rw', default => sub { 0 }); |
31
|
|
|
|
|
|
|
has multi_rule => (is => 'ro', default => sub { 0 }); |
32
|
|
|
|
|
|
|
has sort_code => (is => 'ro', default => sub { BankAccount::Validator::UK::Rule::get_sort_codes() }); |
33
|
|
|
|
|
|
|
|
34
|
|
|
|
|
|
|
=head1 DESCRIPTION |
35
|
|
|
|
|
|
|
|
36
|
|
|
|
|
|
|
The module uses the algorithm provided by VOCALINK to validate the bank sort code |
37
|
|
|
|
|
|
|
and account number. It is done by modulus checking method as specified in the |
38
|
|
|
|
|
|
|
document which is available on their website L |
39
|
|
|
|
|
|
|
It currently supports the document L drafted 3rd Mar 2022. |
40
|
|
|
|
|
|
|
|
41
|
|
|
|
|
|
|
Institutions covered by this document are below: |
42
|
|
|
|
|
|
|
|
43
|
|
|
|
|
|
|
=over 4 |
44
|
|
|
|
|
|
|
|
45
|
|
|
|
|
|
|
=item * Allied Irish |
46
|
|
|
|
|
|
|
|
47
|
|
|
|
|
|
|
=item * Bank of England |
48
|
|
|
|
|
|
|
|
49
|
|
|
|
|
|
|
=item * Bank of Ireland |
50
|
|
|
|
|
|
|
|
51
|
|
|
|
|
|
|
=item * Bank of Scotland |
52
|
|
|
|
|
|
|
|
53
|
|
|
|
|
|
|
=item * Barclays |
54
|
|
|
|
|
|
|
|
55
|
|
|
|
|
|
|
=item * Bradford and Bingley Building Society |
56
|
|
|
|
|
|
|
|
57
|
|
|
|
|
|
|
=item * Charity Bank |
58
|
|
|
|
|
|
|
|
59
|
|
|
|
|
|
|
=item * Citibank |
60
|
|
|
|
|
|
|
|
61
|
|
|
|
|
|
|
=item * Clear Bank |
62
|
|
|
|
|
|
|
|
63
|
|
|
|
|
|
|
=item * Clydesdale |
64
|
|
|
|
|
|
|
|
65
|
|
|
|
|
|
|
=item * Contis Financial Services |
66
|
|
|
|
|
|
|
|
67
|
|
|
|
|
|
|
=item * Co-Operative Bank |
68
|
|
|
|
|
|
|
|
69
|
|
|
|
|
|
|
=item * Coutts |
70
|
|
|
|
|
|
|
|
71
|
|
|
|
|
|
|
=item * First Trust |
72
|
|
|
|
|
|
|
|
73
|
|
|
|
|
|
|
=item * Halifax |
74
|
|
|
|
|
|
|
|
75
|
|
|
|
|
|
|
=item * Hoares Bank |
76
|
|
|
|
|
|
|
|
77
|
|
|
|
|
|
|
=item * HSBC |
78
|
|
|
|
|
|
|
|
79
|
|
|
|
|
|
|
=item * Lloyds |
80
|
|
|
|
|
|
|
|
81
|
|
|
|
|
|
|
=item * Metro Bank |
82
|
|
|
|
|
|
|
|
83
|
|
|
|
|
|
|
=item * NatWest |
84
|
|
|
|
|
|
|
|
85
|
|
|
|
|
|
|
=item * Nationwide Building Society |
86
|
|
|
|
|
|
|
|
87
|
|
|
|
|
|
|
=item * Northern |
88
|
|
|
|
|
|
|
|
89
|
|
|
|
|
|
|
=item * Orwell Union Ltd. |
90
|
|
|
|
|
|
|
|
91
|
|
|
|
|
|
|
=item * Royal Bank of Scotland |
92
|
|
|
|
|
|
|
|
93
|
|
|
|
|
|
|
=item * Santander |
94
|
|
|
|
|
|
|
|
95
|
|
|
|
|
|
|
=item * Secure Trust |
96
|
|
|
|
|
|
|
|
97
|
|
|
|
|
|
|
=item * Starling Bank |
98
|
|
|
|
|
|
|
|
99
|
|
|
|
|
|
|
=item * Tesco Bank |
100
|
|
|
|
|
|
|
|
101
|
|
|
|
|
|
|
=item * TSB |
102
|
|
|
|
|
|
|
|
103
|
|
|
|
|
|
|
=item * Ulster Bank |
104
|
|
|
|
|
|
|
|
105
|
|
|
|
|
|
|
=item * Unity Trust Bank |
106
|
|
|
|
|
|
|
|
107
|
|
|
|
|
|
|
=item * Virgin Bank |
108
|
|
|
|
|
|
|
|
109
|
|
|
|
|
|
|
=item * Williams & Glyn |
110
|
|
|
|
|
|
|
|
111
|
|
|
|
|
|
|
=item * Woolwich |
112
|
|
|
|
|
|
|
|
113
|
|
|
|
|
|
|
=item * Yorkshire Bank |
114
|
|
|
|
|
|
|
|
115
|
|
|
|
|
|
|
=back |
116
|
|
|
|
|
|
|
|
117
|
|
|
|
|
|
|
=head2 NOTE |
118
|
|
|
|
|
|
|
|
119
|
|
|
|
|
|
|
If the modulus check shows the account number as valid this means that the account |
120
|
|
|
|
|
|
|
number is a possible account number for the sorting code but does'nt necessarily |
121
|
|
|
|
|
|
|
mean that it's an account number being used at that sorting code. Any account |
122
|
|
|
|
|
|
|
details found as invalid should be checked with the account holder where possible. |
123
|
|
|
|
|
|
|
|
124
|
|
|
|
|
|
|
=head1 CONSTRUCTOR |
125
|
|
|
|
|
|
|
|
126
|
|
|
|
|
|
|
The constructor simply expects debug flag, which is optional. By the default the |
127
|
|
|
|
|
|
|
debug flag is off. |
128
|
|
|
|
|
|
|
|
129
|
|
|
|
|
|
|
use strict; use warnings; |
130
|
|
|
|
|
|
|
use BankAccount::Validator::UK; |
131
|
|
|
|
|
|
|
|
132
|
|
|
|
|
|
|
# Debug is turned off. |
133
|
|
|
|
|
|
|
my $account1 = BankAccount::Validator::UK->new; |
134
|
|
|
|
|
|
|
|
135
|
|
|
|
|
|
|
# Debug is turned on. |
136
|
|
|
|
|
|
|
my $account2 = BankAccount::Validator::UK->new(debug => 1); |
137
|
|
|
|
|
|
|
|
138
|
|
|
|
|
|
|
=head1 METHODS |
139
|
|
|
|
|
|
|
|
140
|
|
|
|
|
|
|
=head2 is_valid($sort_code, $account_number) |
141
|
|
|
|
|
|
|
|
142
|
|
|
|
|
|
|
It expects two parameters i.e. the sort code and the account number.The sort code |
143
|
|
|
|
|
|
|
can be either nn-nn-nn or nnnnnn format. If the account number starts with 0 then |
144
|
|
|
|
|
|
|
its advisable to pass in as string i.e. '0nnnnnnn'. |
145
|
|
|
|
|
|
|
|
146
|
|
|
|
|
|
|
use strict; use warnings; |
147
|
|
|
|
|
|
|
use BankAccount::Validator::UK; |
148
|
|
|
|
|
|
|
|
149
|
|
|
|
|
|
|
my $account = BankAccount::Validator::UK->new; |
150
|
|
|
|
|
|
|
print "[10-79-99][88837491] is valid.\n" |
151
|
|
|
|
|
|
|
if $account->is_valid(107999, 88837491); |
152
|
|
|
|
|
|
|
|
153
|
|
|
|
|
|
|
print "[18-00-02][00000190] is valid.\n" |
154
|
|
|
|
|
|
|
if $account->is_valid('18-00-02', '00000190'); |
155
|
|
|
|
|
|
|
|
156
|
|
|
|
|
|
|
=cut |
157
|
|
|
|
|
|
|
|
158
|
|
|
|
|
|
|
sub is_valid { |
159
|
41
|
|
|
41
|
1
|
2568
|
my ($self, $sc, $an) = @_; |
160
|
|
|
|
|
|
|
|
161
|
41
|
100
|
|
|
|
139
|
die("ERROR: Missing bank sort code.\n") unless defined $sc; |
162
|
40
|
100
|
|
|
|
103
|
die("ERROR: Missing bank account number.\n") unless defined $an; |
163
|
|
|
|
|
|
|
|
164
|
39
|
|
|
|
|
125
|
($sc, $an) = _prepare($sc, $an); |
165
|
37
|
100
|
|
|
|
113
|
die("ERROR: Invalid sort code.\n") unless (length($sc) == 6); |
166
|
36
|
100
|
|
|
|
118
|
die("ERROR: Invalid account number.\n") unless (length($an) == 8); |
167
|
|
|
|
|
|
|
|
168
|
35
|
|
|
|
|
108
|
my $_sort_code = _init('u', $sc); |
169
|
35
|
|
|
|
|
69
|
my $_account_number = _init('a', $an); |
170
|
35
|
|
|
|
|
91
|
my $_rules = _get_rules($sc); |
171
|
|
|
|
|
|
|
|
172
|
35
|
50
|
|
|
|
149
|
next if (scalar(@{$_rules}) == 0); |
|
35
|
|
|
|
|
130
|
|
173
|
|
|
|
|
|
|
|
174
|
35
|
|
|
|
|
125
|
$self->{sc} = $sc; |
175
|
35
|
|
|
|
|
90
|
$self->{an} = $an; |
176
|
35
|
100
|
|
|
|
54
|
$self->{multi_rule} = (scalar(@{$_rules}) > 1)?(1):(0); |
|
35
|
|
|
|
|
139
|
|
177
|
35
|
|
|
|
|
73
|
foreach my $_rule (@{$_rules}) { |
|
35
|
|
|
|
|
86
|
|
178
|
45
|
|
|
|
|
104
|
$self->{attempt}++; |
179
|
|
|
|
|
|
|
_init('u', '090126', $_sort_code) |
180
|
45
|
100
|
|
|
|
129
|
if ($_rule->{ex} == 8); |
181
|
|
|
|
|
|
|
|
182
|
45
|
100
|
66
|
|
|
253
|
if (($_rule->{ex} == 6) |
|
|
|
100
|
|
|
|
|
183
|
|
|
|
|
|
|
&& |
184
|
|
|
|
|
|
|
($_account_number->{a} =~ /^[4|5|6|7|8]$/) |
185
|
|
|
|
|
|
|
&& |
186
|
|
|
|
|
|
|
($_account_number->{g} == $_account_number->{h})) { |
187
|
|
|
|
|
|
|
|
188
|
2
|
|
|
|
|
16
|
$self->{last_ex} = $_rule->{ex}; |
189
|
2
|
|
|
|
|
4
|
$self->{last_check} = 1; |
190
|
2
|
|
|
|
|
10
|
push @{$self->{trace}}, {'ex' => $_rule->{ex}, |
191
|
|
|
|
|
|
|
'mod' => $_rule->{mod}, |
192
|
2
|
|
|
|
|
4
|
'res' => 'VALID'}; |
193
|
2
|
|
|
|
|
5
|
next; |
194
|
|
|
|
|
|
|
} |
195
|
|
|
|
|
|
|
|
196
|
43
|
100
|
66
|
|
|
429
|
if (($_rule->{ex} == 7) && ($_account_number->{g} == 9)) { |
|
|
100
|
|
|
|
|
|
|
|
100
|
|
|
|
|
|
|
|
100
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
100
|
|
|
|
|
|
197
|
1
|
|
|
|
|
51
|
_init('u','000000', $_rule); |
198
|
1
|
|
|
|
|
3
|
_init('a','00', $_rule); |
199
|
|
|
|
|
|
|
} |
200
|
|
|
|
|
|
|
elsif ($_rule->{ex} == 8) { |
201
|
1
|
|
|
|
|
4
|
_init('u', '090126', $_sort_code); |
202
|
|
|
|
|
|
|
} |
203
|
|
|
|
|
|
|
elsif ($_rule->{ex} =~ /^[2|9]$/) { |
204
|
5
|
100
|
|
|
|
29
|
if ($_rule->{ex} == 9) { |
|
|
100
|
|
|
|
|
|
205
|
1
|
|
|
|
|
3
|
_init('u', '309634', $_sort_code); |
206
|
|
|
|
|
|
|
} |
207
|
|
|
|
|
|
|
elsif ($_account_number->{a} != 0) { |
208
|
3
|
100
|
|
|
|
34
|
if ($_account_number->{g} != 9) { |
|
|
50
|
|
|
|
|
|
209
|
2
|
|
|
|
|
9
|
_init('u','001253', $_rule); |
210
|
2
|
|
|
|
|
5
|
_init('a','6,4,8,7,10,9,3,1', $_rule); |
211
|
|
|
|
|
|
|
} |
212
|
|
|
|
|
|
|
elsif ($_account_number->{g} == 9) { |
213
|
1
|
|
|
|
|
4
|
_init('u','000000', $_rule); |
214
|
1
|
|
|
|
|
3
|
_init('a','0,0,8,7,10,9,3,1', $_rule); |
215
|
|
|
|
|
|
|
} |
216
|
|
|
|
|
|
|
} |
217
|
|
|
|
|
|
|
} |
218
|
|
|
|
|
|
|
elsif ($_rule->{ex} == 10) { |
219
|
5
|
|
|
|
|
37
|
my $_ab = sprintf("%s%s", $_account_number->{a}, $_account_number->{b}); |
220
|
5
|
100
|
100
|
|
|
118
|
if ((($_ab eq "09") or ($_ab eq "99")) && ($_account_number->{g} == 9)) { |
|
|
|
66
|
|
|
|
|
221
|
3
|
|
|
|
|
23
|
_init('u', '000000', $_rule); |
222
|
3
|
|
|
|
|
10
|
_init('a', '00', $_rule); |
223
|
|
|
|
|
|
|
} |
224
|
|
|
|
|
|
|
} |
225
|
|
|
|
|
|
|
elsif ($_rule->{ex} == 3) { |
226
|
0
|
|
|
|
|
0
|
$self->{last_ex} = 3; |
227
|
0
|
0
|
|
|
|
0
|
next if ($_account_number->{c} =~ /^[6|9]$/); |
228
|
|
|
|
|
|
|
} |
229
|
|
|
|
|
|
|
elsif ($_rule->{ex} == 5) { |
230
|
|
|
|
|
|
|
_init('u', $self->{sort_code}->{$sc}, $_sort_code) |
231
|
10
|
100
|
|
|
|
877
|
if (exists $self->{sort_code}->{$sc}); |
232
|
|
|
|
|
|
|
} |
233
|
|
|
|
|
|
|
|
234
|
43
|
|
|
|
|
80
|
my $_status; |
235
|
43
|
100
|
|
|
|
233
|
if ($_rule->{mod} =~ /MOD(\d+)/i) { |
|
|
50
|
|
|
|
|
|
236
|
35
|
|
|
|
|
188
|
$_status = $self->_standard_check($_sort_code, $_account_number, $_rule); |
237
|
|
|
|
|
|
|
} |
238
|
|
|
|
|
|
|
elsif ($_rule->{mod} =~ /DBLAL/i) { |
239
|
8
|
|
|
|
|
33
|
$_status = $self->_double_alternate_check($_sort_code, $_account_number, $_rule); |
240
|
|
|
|
|
|
|
} |
241
|
|
|
|
|
|
|
|
242
|
43
|
50
|
|
|
|
131
|
if (defined $_status) { |
243
|
43
|
|
|
|
|
128
|
$self->{last_ex} = $_status->{ex}; |
244
|
43
|
100
|
|
|
|
135
|
$self->{last_check} = ($_status->{res} eq 'PASS')?(1):(0);; |
245
|
43
|
|
|
|
|
66
|
push @{$self->{trace}}, $_status; |
|
43
|
|
|
|
|
114
|
|
246
|
|
|
|
|
|
|
} |
247
|
|
|
|
|
|
|
|
248
|
43
|
|
|
|
|
128
|
my $_result = $self->_check_result(); |
249
|
43
|
100
|
|
|
|
351
|
return $_result if defined $_result; |
250
|
|
|
|
|
|
|
} |
251
|
|
|
|
|
|
|
|
252
|
|
|
|
|
|
|
return $self->{last_check} |
253
|
1
|
50
|
33
|
|
|
24
|
if ((defined $self->{last_ex}) && ($self->{last_ex} =~ /^6$/) && ($self->{multi_rule})); |
|
|
|
33
|
|
|
|
|
254
|
|
|
|
|
|
|
|
255
|
0
|
|
|
|
|
0
|
return; |
256
|
|
|
|
|
|
|
} |
257
|
|
|
|
|
|
|
|
258
|
|
|
|
|
|
|
=head2 get_trace() |
259
|
|
|
|
|
|
|
|
260
|
|
|
|
|
|
|
Returns the trace information about each rule that applied to the given sort code |
261
|
|
|
|
|
|
|
and account number. |
262
|
|
|
|
|
|
|
|
263
|
|
|
|
|
|
|
use strict; use warnings; |
264
|
|
|
|
|
|
|
use Data::Dumper; |
265
|
|
|
|
|
|
|
use BankAccount::Validator::UK; |
266
|
|
|
|
|
|
|
|
267
|
|
|
|
|
|
|
my $account = BankAccount::Validator::UK->new; |
268
|
|
|
|
|
|
|
print "[87-14-27][09123496] is valid.\n" |
269
|
|
|
|
|
|
|
if $account->is_valid('871427', '09123496'); |
270
|
|
|
|
|
|
|
|
271
|
|
|
|
|
|
|
print "Trace information:\n" . Dumper($account->get_trace); |
272
|
|
|
|
|
|
|
|
273
|
|
|
|
|
|
|
=cut |
274
|
|
|
|
|
|
|
|
275
|
|
|
|
|
|
|
sub get_trace { |
276
|
1
|
|
|
1
|
1
|
33
|
my ($self) = @_; |
277
|
|
|
|
|
|
|
|
278
|
1
|
50
|
|
|
|
10
|
return $self->{trace} if scalar(@{$self->{trace}}); |
|
1
|
|
|
|
|
5
|
|
279
|
|
|
|
|
|
|
} |
280
|
|
|
|
|
|
|
|
281
|
|
|
|
|
|
|
# |
282
|
|
|
|
|
|
|
# |
283
|
|
|
|
|
|
|
# PRIVATE METHODS |
284
|
|
|
|
|
|
|
|
285
|
|
|
|
|
|
|
sub _standard_check { |
286
|
35
|
|
|
35
|
|
104
|
my ($self, $_sort_code, $_account_number, $_rule) = @_; |
287
|
|
|
|
|
|
|
|
288
|
35
|
|
|
|
|
53
|
my $total = 0; |
289
|
35
|
50
|
|
|
|
157
|
$total += 27 if ($_rule->{ex} == 1); |
290
|
|
|
|
|
|
|
|
291
|
35
|
50
|
|
|
|
164
|
if ($_rule->{mod} =~ /MOD(\d+)/i) { |
292
|
35
|
|
|
|
|
61
|
foreach (keys %{$_sort_code}) { |
|
35
|
|
|
|
|
176
|
|
293
|
|
|
|
|
|
|
print "KEY: [$_] SC: [$_sort_code->{$_}] WEIGHTING: [$_rule->{$_}]\n" |
294
|
210
|
50
|
|
|
|
296
|
if $self->{debug}; |
295
|
210
|
|
|
|
|
339
|
$total += $_sort_code->{$_} * $_rule->{$_}; |
296
|
|
|
|
|
|
|
} |
297
|
|
|
|
|
|
|
|
298
|
35
|
|
|
|
|
61
|
foreach (keys %{$_account_number}) { |
|
35
|
|
|
|
|
431
|
|
299
|
|
|
|
|
|
|
print "KEY: [$_] AN: [$_account_number->{$_}] WEIGHTING: [$_rule->{$_}]\n" |
300
|
280
|
50
|
|
|
|
393
|
if $self->{debug}; |
301
|
280
|
|
|
|
|
399
|
$total += $_account_number->{$_} * $_rule->{$_}; |
302
|
|
|
|
|
|
|
} |
303
|
|
|
|
|
|
|
|
304
|
35
|
|
|
|
|
172
|
my $remainder = $total % $1; |
305
|
35
|
100
|
66
|
|
|
179
|
if ($_rule->{ex} == 4) { |
|
|
100
|
|
|
|
|
|
|
|
100
|
|
|
|
|
|
306
|
1
|
|
|
|
|
10
|
my $_gh = sprintf("%d%d", $_account_number->{g}, $_account_number->{h}); |
307
|
1
|
50
|
|
|
|
5
|
if ($remainder == $_gh) { |
308
|
|
|
|
|
|
|
return {'ex' => $_rule->{ex}, |
309
|
|
|
|
|
|
|
'mod' => $_rule->{mod}, |
310
|
1
|
|
|
|
|
8
|
'rem' => $remainder, |
311
|
|
|
|
|
|
|
'tot' => $total, |
312
|
|
|
|
|
|
|
'res' => 'PASS'}; |
313
|
|
|
|
|
|
|
} |
314
|
|
|
|
|
|
|
} |
315
|
|
|
|
|
|
|
elsif (($_rule->{ex} == 5) && ($1 == 11)) { |
316
|
6
|
100
|
|
|
|
20
|
if ($remainder == 0) { |
|
|
100
|
|
|
|
|
|
317
|
2
|
50
|
|
|
|
8
|
if ($_account_number->{g} == 0) { |
318
|
|
|
|
|
|
|
return {'ex' => $_rule->{ex}, |
319
|
|
|
|
|
|
|
'mod' => $_rule->{mod}, |
320
|
2
|
|
|
|
|
15
|
'rem' => $remainder, |
321
|
|
|
|
|
|
|
'tot' => $total, |
322
|
|
|
|
|
|
|
'res' => 'PASS'}; |
323
|
|
|
|
|
|
|
} |
324
|
|
|
|
|
|
|
else { |
325
|
|
|
|
|
|
|
return {'ex' => $_rule->{ex}, |
326
|
|
|
|
|
|
|
'mod' => $_rule->{mod}, |
327
|
0
|
|
|
|
|
0
|
'rem' => $remainder, |
328
|
|
|
|
|
|
|
'tot' => $total, |
329
|
|
|
|
|
|
|
'res' => 'FAIL'}; |
330
|
|
|
|
|
|
|
} |
331
|
|
|
|
|
|
|
} |
332
|
|
|
|
|
|
|
elsif ($remainder == 1) { |
333
|
|
|
|
|
|
|
return {'ex' => $_rule->{ex}, |
334
|
|
|
|
|
|
|
'mod' => $_rule->{mod}, |
335
|
1
|
|
|
|
|
7
|
'rem' => $remainder, |
336
|
|
|
|
|
|
|
'tot' => $total, |
337
|
|
|
|
|
|
|
'res' => 'FAIL'}; |
338
|
|
|
|
|
|
|
} |
339
|
|
|
|
|
|
|
else { |
340
|
3
|
|
|
|
|
7
|
$remainder = 11 - $remainder; |
341
|
3
|
100
|
|
|
|
10
|
if ($_account_number->{g} == $remainder) { |
342
|
|
|
|
|
|
|
return {'ex' => $_rule->{ex}, |
343
|
|
|
|
|
|
|
'mod' => $_rule->{mod}, |
344
|
2
|
|
|
|
|
13
|
'rem' => $remainder, |
345
|
|
|
|
|
|
|
'tot' => $total, |
346
|
|
|
|
|
|
|
'res' => 'PASS'}; |
347
|
|
|
|
|
|
|
} |
348
|
|
|
|
|
|
|
else { |
349
|
|
|
|
|
|
|
return {'ex' => $_rule->{ex}, |
350
|
|
|
|
|
|
|
'mod' => $_rule->{mod}, |
351
|
1
|
|
|
|
|
13
|
'rem' => $remainder, |
352
|
|
|
|
|
|
|
'tot' => $total, |
353
|
|
|
|
|
|
|
'res' => 'FAIL'}; |
354
|
|
|
|
|
|
|
} |
355
|
|
|
|
|
|
|
} |
356
|
|
|
|
|
|
|
} |
357
|
|
|
|
|
|
|
elsif ($remainder == 0) { |
358
|
|
|
|
|
|
|
return {'ex' => $_rule->{ex}, |
359
|
|
|
|
|
|
|
'mod' => $_rule->{mod}, |
360
|
21
|
|
|
|
|
155
|
'rem' => $remainder, |
361
|
|
|
|
|
|
|
'tot' => $total, |
362
|
|
|
|
|
|
|
'res' => 'PASS'}; |
363
|
|
|
|
|
|
|
} |
364
|
|
|
|
|
|
|
else { |
365
|
7
|
100
|
|
|
|
22
|
if ($_rule->{ex} == 14) { |
366
|
1
|
50
|
|
|
|
5
|
if ($_account_number->{h} =~ /^[0|1|9]$/) { |
367
|
1
|
|
|
|
|
3
|
my $an = substr($self->{an}, 0, 7); |
368
|
1
|
|
|
|
|
4
|
$an = sprintf("%s%s", '0', $an); |
369
|
1
|
|
|
|
|
4
|
_init('a', $an, $_account_number); |
370
|
|
|
|
|
|
|
|
371
|
1
|
|
|
|
|
1
|
$total = 0; |
372
|
1
|
|
|
|
|
2
|
foreach (keys %{$_sort_code}) { |
|
1
|
|
|
|
|
3
|
|
373
|
|
|
|
|
|
|
print "KEY: [$_] SC: [$_sort_code->{$_}] WEIGHTING: [$_rule->{$_}]\n" |
374
|
6
|
50
|
|
|
|
10
|
if $self->{debug}; |
375
|
6
|
|
|
|
|
9
|
$total += $_sort_code->{$_} * $_rule->{$_}; |
376
|
|
|
|
|
|
|
} |
377
|
|
|
|
|
|
|
|
378
|
1
|
|
|
|
|
2
|
foreach (keys %{$_account_number}) { |
|
1
|
|
|
|
|
3
|
|
379
|
|
|
|
|
|
|
print "KEY: [$_] AN: [$_account_number->{$_}] WEIGHTING: [$_rule->{$_}]\n" |
380
|
8
|
50
|
|
|
|
79
|
if $self->{debug}; |
381
|
8
|
|
|
|
|
12
|
$total += $_account_number->{$_} * $_rule->{$_}; |
382
|
|
|
|
|
|
|
} |
383
|
|
|
|
|
|
|
|
384
|
1
|
|
|
|
|
4
|
$remainder = $total % 11; |
385
|
1
|
50
|
|
|
|
3
|
if ($remainder == 0) { |
386
|
|
|
|
|
|
|
return {'ex' => $_rule->{ex}, |
387
|
|
|
|
|
|
|
'mod' => $_rule->{mod}, |
388
|
1
|
|
|
|
|
7
|
'rem' => $remainder, |
389
|
|
|
|
|
|
|
'tot' => $total, |
390
|
|
|
|
|
|
|
'res' => 'PASS'}; |
391
|
|
|
|
|
|
|
} |
392
|
|
|
|
|
|
|
else { |
393
|
|
|
|
|
|
|
return {'ex' => $_rule->{ex}, |
394
|
|
|
|
|
|
|
'mod' => $_rule->{mod}, |
395
|
0
|
|
|
|
|
0
|
'rem' => $remainder, |
396
|
|
|
|
|
|
|
'tot' => $total, |
397
|
|
|
|
|
|
|
'res' => 'FAIL'}; |
398
|
|
|
|
|
|
|
} |
399
|
|
|
|
|
|
|
} |
400
|
|
|
|
|
|
|
else { |
401
|
|
|
|
|
|
|
return {'ex' => $_rule->{ex}, |
402
|
|
|
|
|
|
|
'mod' => $_rule->{mod}, |
403
|
0
|
|
|
|
|
0
|
'rem' => $remainder, |
404
|
|
|
|
|
|
|
'tot' => $total, |
405
|
|
|
|
|
|
|
'res' => 'FAIL'}; |
406
|
|
|
|
|
|
|
} |
407
|
|
|
|
|
|
|
} |
408
|
|
|
|
|
|
|
else { |
409
|
|
|
|
|
|
|
return {'ex' => $_rule->{ex}, |
410
|
|
|
|
|
|
|
'mod' => $_rule->{mod}, |
411
|
6
|
|
|
|
|
39
|
'rem' => $remainder, |
412
|
|
|
|
|
|
|
'tot' => $total, |
413
|
|
|
|
|
|
|
'res' => 'FAIL'}; |
414
|
|
|
|
|
|
|
} |
415
|
|
|
|
|
|
|
} |
416
|
|
|
|
|
|
|
} |
417
|
|
|
|
|
|
|
|
418
|
0
|
|
|
|
|
0
|
return; |
419
|
|
|
|
|
|
|
} |
420
|
|
|
|
|
|
|
|
421
|
|
|
|
|
|
|
sub _double_alternate_check { |
422
|
8
|
|
|
8
|
|
24
|
my ($self, $_sort_code, $_account_number, $_rule) = @_; |
423
|
|
|
|
|
|
|
|
424
|
8
|
|
|
|
|
12
|
my $total = 0; |
425
|
8
|
100
|
|
|
|
291
|
$total += 27 if ($_rule->{ex} == 1); |
426
|
|
|
|
|
|
|
|
427
|
8
|
|
|
|
|
27
|
foreach (keys %{$_sort_code}) { |
|
8
|
|
|
|
|
40
|
|
428
|
48
|
|
|
|
|
96
|
$total += _dbal_total($_sort_code->{$_} * $_rule->{$_}); |
429
|
|
|
|
|
|
|
} |
430
|
|
|
|
|
|
|
|
431
|
8
|
|
|
|
|
13
|
foreach (keys %{$_account_number}) { |
|
8
|
|
|
|
|
35
|
|
432
|
64
|
|
|
|
|
90
|
$total += _dbal_total($_account_number->{$_} * $_rule->{$_}); |
433
|
|
|
|
|
|
|
} |
434
|
|
|
|
|
|
|
|
435
|
8
|
|
|
|
|
21
|
my $remainder = $total % 10; |
436
|
8
|
100
|
|
|
|
29
|
if ($_rule->{ex} == 1) { |
|
|
100
|
|
|
|
|
|
|
|
100
|
|
|
|
|
|
437
|
2
|
100
|
|
|
|
6
|
if ($remainder == 0) { |
438
|
|
|
|
|
|
|
return {'ex' => $_rule->{ex}, |
439
|
|
|
|
|
|
|
'mod' => $_rule->{mod}, |
440
|
1
|
|
|
|
|
8
|
'rem' => $remainder, |
441
|
|
|
|
|
|
|
'tot' => $total, |
442
|
|
|
|
|
|
|
'res' => 'PASS'}; |
443
|
|
|
|
|
|
|
} |
444
|
|
|
|
|
|
|
else { |
445
|
|
|
|
|
|
|
return {'ex' => $_rule->{ex}, |
446
|
|
|
|
|
|
|
'mod' => $_rule->{mod}, |
447
|
1
|
|
|
|
|
7
|
'rem' => $remainder, |
448
|
|
|
|
|
|
|
'tot' => $total, |
449
|
|
|
|
|
|
|
'res' => 'FAIL'}; |
450
|
|
|
|
|
|
|
} |
451
|
|
|
|
|
|
|
} |
452
|
|
|
|
|
|
|
elsif ($_rule->{ex} == 5) { |
453
|
4
|
100
|
|
|
|
11
|
if ($remainder == 0) { |
454
|
1
|
50
|
|
|
|
5
|
if ($_account_number->{h} == 0) { |
455
|
|
|
|
|
|
|
return {'ex' => $_rule->{ex}, |
456
|
|
|
|
|
|
|
'mod' => $_rule->{mod}, |
457
|
1
|
|
|
|
|
6
|
'rem' => $remainder, |
458
|
|
|
|
|
|
|
'tot' => $total, |
459
|
|
|
|
|
|
|
'res' => 'PASS'}; |
460
|
|
|
|
|
|
|
} |
461
|
|
|
|
|
|
|
} |
462
|
|
|
|
|
|
|
else { |
463
|
3
|
|
|
|
|
7
|
$remainder = 10 - $remainder; |
464
|
3
|
100
|
|
|
|
11
|
if ($_account_number->{h} == $remainder) { |
465
|
|
|
|
|
|
|
return {'ex' => $_rule->{ex}, |
466
|
|
|
|
|
|
|
'mod' => $_rule->{mod}, |
467
|
2
|
|
|
|
|
12
|
'rem' => $remainder, |
468
|
|
|
|
|
|
|
'tot' => $total, |
469
|
|
|
|
|
|
|
'res' => 'PASS'}; |
470
|
|
|
|
|
|
|
} |
471
|
|
|
|
|
|
|
else { |
472
|
|
|
|
|
|
|
return {'ex' => $_rule->{ex}, |
473
|
|
|
|
|
|
|
'mod' => $_rule->{mod}, |
474
|
1
|
|
|
|
|
5
|
'rem' => $remainder, |
475
|
|
|
|
|
|
|
'tot' => $total, |
476
|
|
|
|
|
|
|
'res' => 'FAIL'}; |
477
|
|
|
|
|
|
|
} |
478
|
|
|
|
|
|
|
} |
479
|
|
|
|
|
|
|
} |
480
|
|
|
|
|
|
|
elsif ($remainder == 0) { |
481
|
|
|
|
|
|
|
return {'ex' => $_rule->{ex}, |
482
|
|
|
|
|
|
|
'mod' => $_rule->{mod}, |
483
|
1
|
|
|
|
|
7
|
'rem' => $remainder, |
484
|
|
|
|
|
|
|
'tot' => $total, |
485
|
|
|
|
|
|
|
'res' => 'PASS'}; |
486
|
|
|
|
|
|
|
} |
487
|
|
|
|
|
|
|
else { |
488
|
|
|
|
|
|
|
return {'ex' => $_rule->{ex}, |
489
|
|
|
|
|
|
|
'mod' => $_rule->{mod}, |
490
|
1
|
|
|
|
|
6
|
'rem' => $remainder, |
491
|
|
|
|
|
|
|
'tot' => $total, |
492
|
|
|
|
|
|
|
'res' => 'FAIL'}; |
493
|
|
|
|
|
|
|
} |
494
|
|
|
|
|
|
|
} |
495
|
|
|
|
|
|
|
|
496
|
|
|
|
|
|
|
sub _init { |
497
|
90
|
|
|
90
|
|
191
|
my ($index, $data, $init) = @_; |
498
|
|
|
|
|
|
|
|
499
|
90
|
100
|
|
|
|
312
|
if ($data =~ /\,/) { |
500
|
3
|
|
|
|
|
12
|
map { $init->{$index++} = $_; } split /\,/,$data; |
|
24
|
|
|
|
|
38
|
|
501
|
|
|
|
|
|
|
} |
502
|
|
|
|
|
|
|
else { |
503
|
87
|
|
|
|
|
269
|
map { $init->{$index++} = $_; } split //,$data; |
|
578
|
|
|
|
|
1280
|
|
504
|
|
|
|
|
|
|
} |
505
|
|
|
|
|
|
|
|
506
|
90
|
|
|
|
|
159
|
return $init; |
507
|
|
|
|
|
|
|
} |
508
|
|
|
|
|
|
|
|
509
|
|
|
|
|
|
|
sub _check_result { |
510
|
43
|
|
|
43
|
|
86
|
my ($self) = @_; |
511
|
|
|
|
|
|
|
|
512
|
43
|
100
|
|
|
|
100
|
if ($self->{multi_rule}) { |
513
|
33
|
100
|
66
|
|
|
552
|
if (((defined $self->{last_ex}) |
|
|
100
|
100
|
|
|
|
|
|
|
100
|
66
|
|
|
|
|
|
|
100
|
66
|
|
|
|
|
|
|
|
66
|
|
|
|
|
|
|
|
100
|
|
|
|
|
|
|
|
66
|
|
|
|
|
|
|
|
100
|
|
|
|
|
|
|
|
66
|
|
|
|
|
|
|
|
66
|
|
|
|
|
514
|
|
|
|
|
|
|
&& ($self->{last_ex} =~ /^2|10|12$/) |
515
|
|
|
|
|
|
|
&& ($self->{last_check} == 1)) |
516
|
|
|
|
|
|
|
|| |
517
|
|
|
|
|
|
|
((defined $self->{last_ex}) |
518
|
|
|
|
|
|
|
&& ($self->{last_ex} =~ /^9|11|13$/) |
519
|
|
|
|
|
|
|
&& ($self->{last_check} == 1) |
520
|
|
|
|
|
|
|
&& ($self->{attempt} == 2))) { |
521
|
12
|
|
|
|
|
26
|
return 1; |
522
|
|
|
|
|
|
|
} |
523
|
|
|
|
|
|
|
elsif ((defined $self->{last_ex}) |
524
|
|
|
|
|
|
|
&& ($self->{last_ex} =~ /^5|6$/) |
525
|
|
|
|
|
|
|
&& ($self->{last_check} == 0)) { |
526
|
5
|
|
|
|
|
12
|
return 0; |
527
|
|
|
|
|
|
|
} |
528
|
|
|
|
|
|
|
elsif ((defined $self->{last_ex}) |
529
|
|
|
|
|
|
|
&& ($self->{last_ex} == 0) |
530
|
|
|
|
|
|
|
&& ($self->{last_check} == 1)) { |
531
|
3
|
|
|
|
|
7
|
return 1; |
532
|
|
|
|
|
|
|
} |
533
|
|
|
|
|
|
|
elsif ($self->{attempt} == 2) { |
534
|
4
|
|
|
|
|
10
|
return $self->{last_check}; |
535
|
|
|
|
|
|
|
} |
536
|
|
|
|
|
|
|
} |
537
|
|
|
|
|
|
|
else { |
538
|
10
|
|
|
|
|
29
|
return $self->{last_check}; |
539
|
|
|
|
|
|
|
} |
540
|
|
|
|
|
|
|
|
541
|
9
|
|
|
|
|
22
|
return; |
542
|
|
|
|
|
|
|
} |
543
|
|
|
|
|
|
|
|
544
|
|
|
|
|
|
|
sub _get_rules { |
545
|
35
|
|
|
35
|
|
86
|
my ($sc) = @_; |
546
|
|
|
|
|
|
|
|
547
|
35
|
50
|
33
|
|
|
224
|
return unless (defined($sc) && ($sc =~ /^\d+$/)); |
548
|
|
|
|
|
|
|
|
549
|
35
|
|
|
|
|
69
|
my $rules; |
550
|
35
|
|
|
|
|
56
|
foreach (@{BankAccount::Validator::UK::Rule::get_rules()}) { |
|
35
|
|
|
|
|
119
|
|
551
|
39480
|
100
|
100
|
|
|
84371
|
push @{$rules}, $_ if ($sc >= $_->{start} && $sc <= $_->{end}); |
|
60
|
|
|
|
|
191
|
|
552
|
|
|
|
|
|
|
} |
553
|
|
|
|
|
|
|
|
554
|
35
|
|
|
|
|
33393
|
return $rules; |
555
|
|
|
|
|
|
|
} |
556
|
|
|
|
|
|
|
|
557
|
|
|
|
|
|
|
sub _dbal_total { |
558
|
112
|
|
|
112
|
|
124
|
my ($_total) = @_; |
559
|
|
|
|
|
|
|
|
560
|
112
|
100
|
|
|
|
136
|
if ($_total > 9) { |
561
|
33
|
|
|
|
|
72
|
my ($left, $right) = split //, $_total; |
562
|
33
|
|
|
|
|
60
|
return ($left + $right); |
563
|
|
|
|
|
|
|
} |
564
|
|
|
|
|
|
|
else { |
565
|
79
|
|
|
|
|
114
|
return $_total; |
566
|
|
|
|
|
|
|
} |
567
|
|
|
|
|
|
|
} |
568
|
|
|
|
|
|
|
|
569
|
|
|
|
|
|
|
sub _prepare { |
570
|
39
|
|
|
39
|
|
86
|
my ($sc, $an) = @_; |
571
|
|
|
|
|
|
|
|
572
|
39
|
|
|
|
|
109
|
$sc =~ s/[\-\s]+//g; |
573
|
39
|
|
|
|
|
75
|
$an =~ s/\s+//g; |
574
|
|
|
|
|
|
|
|
575
|
39
|
100
|
|
|
|
158
|
die("ERROR: Invalid bank sort code [$sc].\n") unless ($sc =~ /^\d+$/); |
576
|
38
|
100
|
|
|
|
160
|
die("ERROR: Invalid bank account number [$an].\n") unless ($an =~ /^\d+$/); |
577
|
|
|
|
|
|
|
|
578
|
37
|
50
|
|
|
|
255
|
if (length($an) == 10) { |
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
579
|
0
|
0
|
|
|
|
0
|
if ($an =~ /^(\d+)\-(\d+)/) { |
580
|
0
|
|
|
|
|
0
|
$an = $2; |
581
|
|
|
|
|
|
|
} |
582
|
|
|
|
|
|
|
else { |
583
|
0
|
|
|
|
|
0
|
$an = substr($an, 0, 8); |
584
|
|
|
|
|
|
|
} |
585
|
|
|
|
|
|
|
} |
586
|
|
|
|
|
|
|
elsif (length($an) == 9) { |
587
|
0
|
|
|
|
|
0
|
my $_a = substr($an, 0, 1); |
588
|
0
|
|
|
|
|
0
|
$an = substr($an, 1, 8); |
589
|
0
|
|
|
|
|
0
|
$sc = substr($sc, 0, 5); |
590
|
0
|
|
|
|
|
0
|
$sc .= $_a; |
591
|
|
|
|
|
|
|
} |
592
|
|
|
|
|
|
|
elsif (length($an) == 7) { |
593
|
0
|
|
|
|
|
0
|
$an = '0'.$an; |
594
|
|
|
|
|
|
|
} |
595
|
|
|
|
|
|
|
elsif (length($an) == 6) { |
596
|
0
|
|
|
|
|
0
|
$an = '00'.$an; |
597
|
|
|
|
|
|
|
} |
598
|
|
|
|
|
|
|
|
599
|
37
|
|
|
|
|
104
|
return ($sc, $an); |
600
|
|
|
|
|
|
|
} |
601
|
|
|
|
|
|
|
|
602
|
|
|
|
|
|
|
=head1 AUTHOR |
603
|
|
|
|
|
|
|
|
604
|
|
|
|
|
|
|
Mohammad S Anwar, C<< >> |
605
|
|
|
|
|
|
|
|
606
|
|
|
|
|
|
|
=head1 REPOSITORY |
607
|
|
|
|
|
|
|
|
608
|
|
|
|
|
|
|
L |
609
|
|
|
|
|
|
|
|
610
|
|
|
|
|
|
|
=head1 BUGS |
611
|
|
|
|
|
|
|
|
612
|
|
|
|
|
|
|
Please report any bugs or feature requests to C
|
613
|
|
|
|
|
|
|
rt.cpan.org>, or through the web interface at L. |
614
|
|
|
|
|
|
|
I will be notified, and then you'll automatically be notified of progress on your |
615
|
|
|
|
|
|
|
bug as I make changes. |
616
|
|
|
|
|
|
|
|
617
|
|
|
|
|
|
|
=head1 SUPPORT |
618
|
|
|
|
|
|
|
|
619
|
|
|
|
|
|
|
You can find documentation for this module with the perldoc command. |
620
|
|
|
|
|
|
|
|
621
|
|
|
|
|
|
|
perldoc BankAccount::Validator::UK |
622
|
|
|
|
|
|
|
|
623
|
|
|
|
|
|
|
You can also look for information at: |
624
|
|
|
|
|
|
|
|
625
|
|
|
|
|
|
|
=over 4 |
626
|
|
|
|
|
|
|
|
627
|
|
|
|
|
|
|
=item * RT: CPAN's request tracker (report bugs here) |
628
|
|
|
|
|
|
|
|
629
|
|
|
|
|
|
|
L |
630
|
|
|
|
|
|
|
|
631
|
|
|
|
|
|
|
=item * CPAN Ratings |
632
|
|
|
|
|
|
|
|
633
|
|
|
|
|
|
|
L |
634
|
|
|
|
|
|
|
|
635
|
|
|
|
|
|
|
=item * MetaCPAN |
636
|
|
|
|
|
|
|
|
637
|
|
|
|
|
|
|
L |
638
|
|
|
|
|
|
|
|
639
|
|
|
|
|
|
|
=back |
640
|
|
|
|
|
|
|
|
641
|
|
|
|
|
|
|
=head1 LICENSE AND COPYRIGHT |
642
|
|
|
|
|
|
|
|
643
|
|
|
|
|
|
|
Copyright (C) 2012 - 2021 Mohammad S Anwar. |
644
|
|
|
|
|
|
|
|
645
|
|
|
|
|
|
|
This program is free software; you can redistribute it and / or modify it under |
646
|
|
|
|
|
|
|
the terms of the the Artistic License (2.0). You may obtain a copy of the full |
647
|
|
|
|
|
|
|
license at: |
648
|
|
|
|
|
|
|
|
649
|
|
|
|
|
|
|
L |
650
|
|
|
|
|
|
|
|
651
|
|
|
|
|
|
|
Any use, modification, and distribution of the Standard or Modified Versions is |
652
|
|
|
|
|
|
|
governed by this Artistic License.By using, modifying or distributing the Package, |
653
|
|
|
|
|
|
|
you accept this license. Do not use, modify, or distribute the Package, if you do |
654
|
|
|
|
|
|
|
not accept this license. |
655
|
|
|
|
|
|
|
|
656
|
|
|
|
|
|
|
If your Modified Version has been derived from a Modified Version made by someone |
657
|
|
|
|
|
|
|
other than you,you are nevertheless required to ensure that your Modified Version |
658
|
|
|
|
|
|
|
complies with the requirements of this license. |
659
|
|
|
|
|
|
|
|
660
|
|
|
|
|
|
|
This license does not grant you the right to use any trademark, service mark, |
661
|
|
|
|
|
|
|
tradename, or logo of the Copyright Holder. |
662
|
|
|
|
|
|
|
|
663
|
|
|
|
|
|
|
This license includes the non-exclusive, worldwide, free-of-charge patent license |
664
|
|
|
|
|
|
|
to make, have made, use, offer to sell, sell, import and otherwise transfer the |
665
|
|
|
|
|
|
|
Package with respect to any patent claims licensable by the Copyright Holder that |
666
|
|
|
|
|
|
|
are necessarily infringed by the Package. If you institute patent litigation |
667
|
|
|
|
|
|
|
(including a cross-claim or counterclaim) against any party alleging that the |
668
|
|
|
|
|
|
|
Package constitutes direct or contributory patent infringement,then this Artistic |
669
|
|
|
|
|
|
|
License to you shall terminate on the date that such litigation is filed. |
670
|
|
|
|
|
|
|
|
671
|
|
|
|
|
|
|
Disclaimer of Warranty: THE PACKAGE IS PROVIDED BY THE COPYRIGHT HOLDER AND |
672
|
|
|
|
|
|
|
CONTRIBUTORS "AS IS' AND WITHOUT ANY EXPRESS OR IMPLIED WARRANTIES. THE IMPLIED |
673
|
|
|
|
|
|
|
WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE, OR |
674
|
|
|
|
|
|
|
NON-INFRINGEMENT ARE DISCLAIMED TO THE EXTENT PERMITTED BY YOUR LOCAL LAW. UNLESS |
675
|
|
|
|
|
|
|
REQUIRED BY LAW, NO COPYRIGHT HOLDER OR CONTRIBUTOR WILL BE LIABLE FOR ANY DIRECT, |
676
|
|
|
|
|
|
|
INDIRECT, INCIDENTAL, OR CONSEQUENTIAL DAMAGES ARISING IN ANY WAY OUT OF THE USE |
677
|
|
|
|
|
|
|
OF THE PACKAGE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
678
|
|
|
|
|
|
|
|
679
|
|
|
|
|
|
|
=cut |
680
|
|
|
|
|
|
|
|
681
|
|
|
|
|
|
|
1; # End of BankAccount::Validator::UK |