Group
Extension

Type-Tiny/t/21-types/TypeTiny.t

=pod

=encoding utf-8

=head1 PURPOSE

Basic tests for B<TypeTiny> from L<Types::TypeTiny>.

=head1 AUTHOR

Toby Inkster E<lt>tobyink@cpan.orgE<gt>.

=head1 COPYRIGHT AND LICENCE

This software is copyright (c) 2019-2023 by Toby Inkster.

This is free software; you can redistribute it and/or modify it under
the same terms as the Perl 5 programming language system itself.

=cut

use strict;
use warnings;
use Test::More;
use Test::Fatal;
use Test::TypeTiny;
use Types::TypeTiny qw( TypeTiny );

isa_ok(TypeTiny, 'Type::Tiny', 'TypeTiny');
is(TypeTiny->name, 'TypeTiny', 'TypeTiny has correct name');
is(TypeTiny->display_name, 'TypeTiny', 'TypeTiny has correct display_name');
is(TypeTiny->library, 'Types::TypeTiny', 'TypeTiny knows it is in the Types::TypeTiny library');
ok(Types::TypeTiny->has_type('TypeTiny'), 'Types::TypeTiny knows it has type TypeTiny');
ok(!TypeTiny->deprecated, 'TypeTiny is not deprecated');
ok(!TypeTiny->is_anon, 'TypeTiny is not anonymous');
ok(TypeTiny->can_be_inlined, 'TypeTiny can be inlined');
is(exception { TypeTiny->inline_check(q/$xyz/) }, undef, "Inlining TypeTiny doesn't throw an exception");
ok(TypeTiny->has_coercion, "TypeTiny has a coercion");
ok(!TypeTiny->is_parameterizable, "TypeTiny isn't parameterizable");
isnt(TypeTiny->type_default, undef, "TypeTiny has a type_default");
is(TypeTiny->type_default->(), do { require Types::Standard; Types::Standard::Any() }, "TypeTiny type_default is Any");

#
# The @tests array is a list of triples:
#
# 1. Expected result - pass, fail, or xxxx (undefined).
# 2. A description of the value being tested.
# 3. The value being tested.
#

my @tests = (
	fail => 'undef'                    => undef,
	fail => 'false'                    => !!0,
	fail => 'true'                     => !!1,
	fail => 'zero'                     =>  0,
	fail => 'one'                      =>  1,
	fail => 'negative one'             => -1,
	fail => 'non integer'              =>  3.1416,
	fail => 'empty string'             => '',
	fail => 'whitespace'               => ' ',
	fail => 'line break'               => "\n",
	fail => 'random string'            => 'abc123',
	fail => 'loaded package name'      => 'Type::Tiny',
	fail => 'unloaded package name'    => 'This::Has::Probably::Not::Been::Loaded',
	fail => 'a reference to undef'     => do { my $x = undef; \$x },
	fail => 'a reference to false'     => do { my $x = !!0; \$x },
	fail => 'a reference to true'      => do { my $x = !!1; \$x },
	fail => 'a reference to zero'      => do { my $x = 0; \$x },
	fail => 'a reference to one'       => do { my $x = 1; \$x },
	fail => 'a reference to empty string' => do { my $x = ''; \$x },
	fail => 'a reference to random string' => do { my $x = 'abc123'; \$x },
	fail => 'blessed scalarref'        => bless(do { my $x = undef; \$x }, 'SomePkg'),
	fail => 'empty arrayref'           => [],
	fail => 'arrayref with one zero'   => [0],
	fail => 'arrayref of integers'     => [1..10],
	fail => 'arrayref of numbers'      => [1..10, 3.1416],
	fail => 'blessed arrayref'         => bless([], 'SomePkg'),
	fail => 'empty hashref'            => {},
	fail => 'hashref'                  => { foo => 1 },
	fail => 'blessed hashref'          => bless({}, 'SomePkg'),
	fail => 'coderef'                  => sub { 1 },
	fail => 'blessed coderef'          => bless(sub { 1 }, 'SomePkg'),
	fail => 'glob'                     => do { no warnings 'once'; *SOMETHING },
	fail => 'globref'                  => do { no warnings 'once'; my $x = *SOMETHING; \$x },
	fail => 'blessed globref'          => bless(do { no warnings 'once'; my $x = *SOMETHING; \$x }, 'SomePkg'),
	fail => 'regexp'                   => qr/./,
	fail => 'blessed regexp'           => bless(qr/./, 'SomePkg'),
	fail => 'filehandle'               => do { open my $x, '<', $0 or die; $x },
	fail => 'filehandle object'        => do { require IO::File; 'IO::File'->new($0, 'r') },
	fail => 'ref to scalarref'         => do { my $x = undef; my $y = \$x; \$y },
	fail => 'ref to arrayref'          => do { my $x = []; \$x },
	fail => 'ref to hashref'           => do { my $x = {}; \$x },
	fail => 'ref to coderef'           => do { my $x = sub { 1 }; \$x },
	fail => 'ref to blessed hashref'   => do { my $x = bless({}, 'SomePkg'); \$x },
	fail => 'object stringifying to ""' => do { package Local::OL::StringEmpty; use overload q[""] => sub { "" }; bless [] },
	fail => 'object stringifying to "1"' => do { package Local::OL::StringOne; use overload q[""] => sub { "1" }; bless [] },
	fail => 'object numifying to 0'    => do { package Local::OL::NumZero; use overload q[0+] => sub { 0 }; bless [] },
	fail => 'object numifying to 1'    => do { package Local::OL::NumOne; use overload q[0+] => sub { 1 }; bless [] },
	fail => 'object overloading arrayref' => do { package Local::OL::Array; use overload q[@{}] => sub { $_[0]{array} }; bless {array=>[]} },
	fail => 'object overloading hashref' => do { package Local::OL::Hash; use overload q[%{}] => sub { $_[0][0] }; bless [{}] },
	fail => 'object overloading coderef' => do { package Local::OL::Code; use overload q[&{}] => sub { $_[0][0] }; bless [sub { 1 }] },
	fail => 'object booling to false'  => do { package Local::OL::BoolFalse; use overload q[bool] => sub { 0 }; bless [] },
	fail => 'object booling to true'   => do { package Local::OL::BoolTrue;  use overload q[bool] => sub { 1 }; bless [] },
#TESTS
);

while (@tests) {
	my ($expect, $label, $value) = splice(@tests, 0 , 3);
	if ($expect eq 'xxxx') {
		note("UNDEFINED OUTCOME: $label");
	}
	elsif ($expect eq 'pass') {
		should_pass($value, TypeTiny, ucfirst("$label should pass TypeTiny"));
	}
	elsif ($expect eq 'fail') {
		should_fail($value, TypeTiny, ucfirst("$label should fail TypeTiny"));
	}
	else {
		fail("expected '$expect'?!");
	}
}

should_pass(TypeTiny, TypeTiny);  # dogfooding

subtest "Can coerce from coderef to TypeTiny" => sub {
	my $Arrayref = TypeTiny->coerce(
		sub { ref($_) eq 'ARRAY' }
	);
	should_pass( $Arrayref, TypeTiny );
	should_pass( [], $Arrayref );
	should_fail( {}, $Arrayref );
};

subtest "Can coerce from Type::Nano to TypeTiny" => sub {
	my $Arrayref = TypeTiny->coerce(
		Local::Dummy1::ArrayRef()
	);
	should_pass( $Arrayref, TypeTiny );
	should_pass( [], $Arrayref );
	should_fail( {}, $Arrayref );
} if eval q{ package Local::Dummy1; use Type::Nano qw(ArrayRef); 1 };

subtest "Can coerce from MooseX::Types to TypeTiny" => sub {
	my $Arrayref = TypeTiny->coerce(
		Local::Dummy2::ArrayRef()
	);
	should_pass( $Arrayref, TypeTiny );
	should_pass( [], $Arrayref );
	should_fail( {}, $Arrayref );
	ok($Arrayref->is_parameterizable);
	ok($Arrayref->can_be_inlined);
} if eval q{ package Local::Dummy2; use MooseX::Types::Moose qw(ArrayRef); 1 };

subtest "Can coerce from MouseX::Types to TypeTiny" => sub {
	my $Arrayref = TypeTiny->coerce(
		Local::Dummy3::ArrayRef()
	);
	should_pass( $Arrayref, TypeTiny );
	should_pass( [], $Arrayref );
	should_fail( {}, $Arrayref );
	ok($Arrayref->is_parameterizable);
} if eval q{ package Local::Dummy3; use MouseX::Types::Mouse qw(ArrayRef); 1 };

subtest "Can coerce from Specio to TypeTiny" => sub {
	my $Arrayref = TypeTiny->coerce(
		Local::Dummy4::t('ArrayRef')
	);
	should_pass( $Arrayref, TypeTiny );
	should_pass( [], $Arrayref );
	should_fail( {}, $Arrayref );
	ok($Arrayref->can_be_inlined);
} if eval q{ package Local::Dummy4; use Specio::Library::Builtins; 1 };

done_testing;



Powered by Groonga
Maintained by Kenichi Ishigaki <ishigaki@cpan.org>. If you find anything, submit it on GitHub.