1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263
|
package Test2::Tools::Type;
use strict;
use warnings;
use base qw(Exporter);
our $VERSION = '1.0.1';
use Carp qw(croak);
use Test2::API qw(context);
use Test2::Compare::Type ();
use Scalar::Type qw(bool_supported);
our @EXPORT = qw(
is_integer is_number
is_bool bool_supported
type
);
sub import {
if(@_) {
if(grep { $_ eq ':extras' } @_) {
require Test2::Tools::Type::Extras;
Test2::Tools::Type::Extras->import();
push @EXPORT, @Test2::Tools::Type::Extras::EXPORT;
@_ = grep { $_ ne ':extras' } @_;
}
if(grep { $_ eq 'show_types' } @_) {
print "Supported types:\n";
print " ".substr($_, 3)."\n" foreach(sort grep { /^is_/ } @EXPORT);
return;
}
}
goto &Exporter::import;
}
sub is_integer { _checker(\&Scalar::Type::is_integer, @_); }
sub is_number { _checker(\&Scalar::Type::is_number, @_); }
sub is_bool {
croak("You need perl 5.36 or higher to use is_bool")
unless(bool_supported());
_checker(\&Scalar::Type::is_bool, @_);
}
sub _checker {
my($checker, $candidate, $name) = @_;
my $result = $checker->($candidate);
# if we're coming from Test2::Compare::Type just do the check, don't
# get/twiddle/release a context
return $result if($Test2::Compare::Type::verifying);
my $ctx = context();
return $ctx->pass_and_release($name) if($result);
return $ctx->fail_and_release($name);
}
sub type {
my @caller = caller;
return Test2::Compare::Type->new(
file => $caller[1],
lines => [$caller[2]],
type => \@_,
);
}
1;
=head1 NAME
Test2::Tools::Type - Tools for checking data types
=head1 SYNOPSIS
use Test2::V0;
use Test2::Tools::Type;
is_integer(1, "is 1 integer?"); # pass, yes it is
is_integer('1', "is '1' an integer?"); # fail, no it's a string
SKIP: {
skip "Your perl is too old" unless(bool_supported());
is_bool(1 == 2, "is false a Boolean?"); # pass, yes it is
is_bool(3.1415, "is pi a Boolean?"); # fail, no it isn't
}
like
{ should_be_int => 1, other_stuff => "we don't care about this" },
hash {
field should_be_int => type('integer');
},
"is the should_be_int field an integer?";
or if you want even more check functions:
use Test2::V0;
use Test2::Tools::Type qw(:extras);
is_hashref($foo);
=head1 OVERVIEW
Sometimes you don't want to be too precise in your tests, you just want to
check that your code returns the right type of result but you don't care whether
it's returning 192 or 193 - just checking that it returns an integer is good
enough.
=head1 FUNCTIONS
All these are exported by default.
=head2 bool_supported
Returns true if your perl is recent enough to have the Boolean type, false
otherwise. It will be true if your perl is version 5.35.7 or higher.
=head2 is_bool
Emits a test pass if its argument is a Boolean - ie is the result of a comparison -
and a fail otherwise.
It is a fatal error to call this on a perl that is too old. If your tests need
to run on perl 5.35.6 or earlier then you will need to check C<bool_supported>
before using it. See the L</SYNOPSIS> above.
=head2 is_integer
Emits a test pass if its argument is an integer and a fail otherwise. Note that it
can tell the difference between C<1> (an integer) and C<'1'> (a string).
=head2 is_number
Emits a test pass if its argument is a number and a fail otherwise. Note that it
can tell the difference between C<1> (a number), C<1.2> (also a number) and
C<'1'> (a string).
=head2 type
Returns a check that you can use in a test such as:
like
{ int => 1 },
hash { field int => type('integer'); },
"the 'int' field is an integer";
You can negate the test with a C<!> thus. This test will fail:
like
{ int => 1 },
hash { field int => !type('integer'); },
"the 'int' field is an integer";
You can supply more than one argument, so if you want to check that
something is a I<positive> integer, for example, you can do:
is(94, type(qw(positive integer)));
You can check something's type and value:
# this uses 'number' from Test2::Tools::Compare
is($foo, type('integer', number(94)));
And indeed you can use any other Test2 checker:
# 'in_set' also comes from Test2::Tools::Compare
is($foo, type('integer', in_set(1, 5, 8)));
Valid arguments are any other Test2 checker (specifically, anything that
inherits from L<Test2::Compare::Base>), and any of the C<is_*> methods' names,
with the leading C<is_> removed. You can see a list of supported types thus:
$ perl -MTest2::Tools::Type=show_types
or to include the extra functions:
$ perl -MTest2::Tools::Type=show_types,:extras
=head1 EXTRA FUNCTIONS
By default the only check functions you get are those that are thin wrappers
around L<Scalar::Type>. If you pass the C<:extras> argument at C<use>-time then
all the following are available as well:
=head2 regex_supported
Returns true if your perl can reliably report the difference between a regex
and a reference to a scalar, or false otherwise. It will be true if your perl
is version 5.12 or higher.
=head2 is_positive, is_negative
Emit a test pass/fail depending on the argument's sign. Note that C<0> is
considered neither positive nor negative.
=head2 is_zero
Emit a pass/fail depending on whether the argument is zero.
=head2 is_ref
Emit a pass/fail depending on whether the argument is a reference. This
includes blessed objects.
=head2 is_object
Emit a pass/fail depending on whether the argument is a blessed object.
=head2 is_regex
Emit a test pass if its argument is a regex, and a fail otherwise.
It is a fatal error to call this on a perl that is too old. If your tests need
to run on perl 5.10.1 or earlier then you will need to check C<regex_supported>
before using it.
=head2 is_hashref, is_arrayref, is_scalarref, is_coderef, is_globref, is_regex, is_refref
Emit a pass/fail if the argumet is a reference to something of the appropriate type.
=head1 CAVEATS
The definitions of Boolean, integer and number are exactly the same as those in
L<Scalar::Type>, which this is a thin wrapper around.
Blessed objects will match both C<is_object> and the appropriate C<is_*ref>. If you
need to check that something is a ref, but is I<not> blessed, do something like:
is($foo, type(hashref => !type('object')));
=head1 SEE ALSO
L<Scalar::Type>
L<Test2>
=head1 BUGS
If you find any bugs please report them on Github, preferably with a test case.
=head1 FEEDBACK
I welcome feedback about my code, especially constructive criticism.
=head1 AUTHOR, COPYRIGHT and LICENCE
Copyright 2024 David Cantrell E<lt>F<david@cantrell.org.uk>E<gt>
This software is free-as-in-speech software, and may be used,
distributed, and modified under the terms of either the GNU
General Public Licence version 2 or the Artistic Licence. It's
up to you which one you use. The full text of the licences can
be found in the files GPL2.txt and ARTISTIC.txt, respectively.
=head1 CONSPIRACY
This module is also free-as-in-mason software.
=cut
|