File Coverage

blib/lib/Business/RO/TaxDeduction.pm
Criterion Covered Total %
statement 46 46 100.0
branch 10 10 100.0
condition 5 6 83.3
subroutine 13 13 100.0
pod 1 1 100.0
total 75 76 98.6


line stmt bran cond sub pod time code
1             package Business::RO::TaxDeduction;
2             $Business::RO::TaxDeduction::VERSION = '0.010';
3             # ABSTRACT: Romanian salary tax deduction calculator
4              
5 2     2   161431 use 5.010001;
  2         9  
6 2     2   613 use utf8;
  2         23  
  2         9  
7 2     2   587 use Moo;
  2         18277  
  2         9  
8 2     2   2606 use MooX::HandlesVia;
  2         13518  
  2         11  
9 2     2   1600 use Math::BigFloat;
  2         76176  
  2         10  
10 2     2   33067 use Scalar::Util qw(blessed);
  2         5  
  2         112  
11 2         19 use Business::RO::TaxDeduction::Types qw(
12             Int
13             MathBigFloat
14 2     2   650 );
  2         6  
15 2     2   2030 use Business::RO::TaxDeduction::Amount;
  2         6  
  2         62  
16 2     2   596 use Business::RO::TaxDeduction::Ranges;
  2         4  
  2         832  
17              
18             has 'vbl' => (
19             is => 'ro',
20             isa => Int,
21             required => 1,
22             );
23              
24             has 'year' => (
25             is => 'ro',
26             isa => Int,
27             default => sub { 2016 },
28             );
29              
30             has 'persons' => (
31             is => 'ro',
32             isa => Int,
33             default => sub { 0 },
34             );
35              
36             has 'deduction' => (
37             is => 'ro',
38             lazy => 1,
39             default => sub {
40             my $self = shift;
41             return Business::RO::TaxDeduction::Amount->new(
42             year => $self->year,
43             persons => $self->persons,
44             );
45             },
46             );
47              
48             has 'ranges' => (
49             is => 'ro',
50             lazy => 1,
51             default => sub {
52             my $self = shift;
53             return Business::RO::TaxDeduction::Ranges->new(
54             year => $self->year,
55             );
56             },
57             handles => [ qw( vbl_min vbl_max f_min f_max ) ],
58             );
59              
60             has 'ten' => (
61             is => 'ro',
62             isa => MathBigFloat,
63             default => sub {
64             return Math::BigFloat->new(10);
65             },
66             );
67              
68             sub tax_deduction {
69 991     991 1 457055 my $self = shift;
70 991         2839 my $vbl = $self->_round_to_int( $self->vbl );
71 991         19143 my $amount = $self->deduction->amount;
72 991 100 66     124759 if ( $vbl <= $self->vbl_min ) {
    100          
73 11         473 return $amount;
74             }
75             elsif ( ( $vbl > $self->vbl_min ) && ( $vbl <= $self->vbl_max ) ) {
76 970         33970 $amount = $self->_tax_deduction_formula($vbl, $amount);
77 970 100       255520 return ( blessed $amount ) ? $amount->bstr : $amount;
78             }
79             else {
80 10         366 return 0; # 0 for VBL > vbl_max
81             }
82             }
83              
84             sub _tax_deduction_formula {
85 970     970   1799 my ( $self, $vbl, $base_deduction ) = @_;
86 970         13112 my $amount = $base_deduction * ( 1 - ( $vbl - $self->f_min ) / $self->f_max );
87 970         33240 return $self->_round_to_tens($amount);
88             }
89              
90             sub _round_to_int {
91 1003     1003   8112 my ( $self, $amount ) = @_;
92 1003         3559 return int( $amount + 0.5 * ( $amount <=> 0 ) );
93             }
94              
95             sub _round_to_tens {
96 978     978   5838 my ( $self, $para_amount ) = @_;
97 978         2928 my $amount = Math::BigFloat->new($para_amount);
98              
99 978 100       168356 return 0 if $amount == 0;
100              
101 967         164272 my $afloor = $amount->copy()->bfloor();
102 967         78417 my $amodulo = $afloor->copy()->bmod( $self->ten );
103              
104 967 100 100     264964 return $amount if $amount->is_int && $amodulo == 0;
105 752         6442 return $afloor->bsub($amodulo)->badd( $self->ten );
106             }
107              
108             1;
109              
110             __END__