package Test2::Compare::Float;
use strict;
use warnings;
use Carp qw/confess/;
use base 'Test2::Compare::Base';
our $VERSION = '0.000122';
our $DEFAULT_TOLERANCE = 1e-08;
use Test2::Util::HashBase qw/input tolerance precision/;
# Overloads '!' for us.
use Test2::Compare::Negatable;
sub init {
my $self = shift;
my $input = $self->{+INPUT};
if ( exists $self->{+TOLERANCE} and exists $self->{+PRECISION} ) {
confess "can't set both tolerance and precision";
} elsif (!exists $self->{+PRECISION} and !exists $self->{+TOLERANCE}) {
$self->{+TOLERANCE} = $DEFAULT_TOLERANCE
}
confess "input must be defined for 'Float' check"
unless defined $input;
# Check for ''
confess "input must be a number for 'Float' check"
unless length($input) && $input =~ m/\S/;
confess "precision must be an integer for 'Float' check"
if exists $self->{+PRECISION} && $self->{+PRECISION} !~ m/^\d+$/;
$self->SUPER::init(@_);
}
sub name {
my $self = shift;
my $in = $self->{+INPUT};
my $precision = $self->{+PRECISION};
if ( defined $precision) {
return sprintf "%.*f", $precision, $in;
}
my $tolerance = $self->{+TOLERANCE};
return "$in +/- $tolerance";
}
sub operator {
my $self = shift;
return '' unless @_;
my ($got) = @_;
return '' unless defined($got);
return '' unless length($got) && $got =~ m/\S/;
if ( $self->{+PRECISION} )
{
return 'ne' if $self->{+NEGATE};
return 'eq';
}
return '!=' if $self->{+NEGATE};
return '==';
}
sub verify {
my $self = shift;
my %params = @_;
my ($got, $exists) = @params{qw/got exists/};
return 0 unless $exists;
return 0 unless defined $got;
return 0 if ref $got;
return 0 unless length($got) && $got =~ m/\S/;
my $input = $self->{+INPUT};
my $negate = $self->{+NEGATE};
my $tolerance = $self->{+TOLERANCE};
my $precision = $self->{+PRECISION};
my @warnings;
my $out;
{
local $SIG{__WARN__} = sub { push @warnings => @_ };
my $equal = ($input == $got);
if (!$equal) {
if (defined $tolerance) {
$equal = 1 if
$got > $input - $tolerance &&
$got < $input + $tolerance;
} else {
$equal =
sprintf("%.*f", $precision, $got) eq
sprintf("%.*f", $precision, $input);
}
}
$out = $negate ? !$equal : $equal;
}
for my $warn (@warnings) {
if ($warn =~ m/numeric/) {
$out = 0;
next; # This warning won't help anyone.
}
warn $warn;
}
return $out;
}
1;
__END__
=pod
=encoding UTF-8
=head1 NAME
Test2::Compare::Float - Compare two values as numbers with tolerance.
=head1 DESCRIPTION
This is used to compare two numbers. You can also check that two numbers are not
the same.
This is similar to Test2::Compare::Number, with extra checks to work around floating
point representation issues.
The optional 'tolerance' parameter controls how close the two numbers must be to
be considered equal. Tolerance defaults to 1e-08.
B<Note>: This will fail if the received value is undefined. It must be a number.
B<Note>: This will fail if the comparison generates a non-numeric value warning
(which will not be shown). This is because it must get a number. The warning is
not shown as it will report to a useless line and filename. However, the test
diagnostics show both values.
=head1 SOURCE
The source code repository for Test2-Suite can be found at
F<https://github.com/Test-More/Test2-Suite/>.
=head1 MAINTAINERS
=over 4
=item Chad Granum E<lt>exodist@cpan.orgE<gt>
=back
=head1 AUTHORS
=over 4
=item Andrew Grangaard E<lt>spazm@cpan.orgE<gt>
=back
=head1 COPYRIGHT
Copyright 2018 Chad Granum E<lt>exodist@cpan.orgE<gt>.
This program is free software; you can redistribute it and/or
modify it under the same terms as Perl itself.
See F<http://dev.perl.org/licenses/>
=cut
Copyright 2K16 - 2K18 Indonesian Hacker Rulez