Fix handling of union types (RT#58417)
Tomas Doran [Thu, 26 Aug 2010 09:42:38 +0000 (10:42 +0100)]
ChangeLog
lib/MooseX/Getopt/OptionTypeMap.pm
t/107_union_bug.t [new file with mode: 0644]

index ff05b58..cfd52cd 100644 (file)
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,5 +1,7 @@
 Revision history for Perl extension MooseX-Getopt
 
+  * Fix handling of Union types (RT#58417)
+
 0.31 Wed 7 Jul 2010
   * Fix issue causing tests to fail if MooseX::SimpleConfig isn't installed.
 
index 15c479f..4102c75 100644 (file)
@@ -16,6 +16,15 @@ my %option_type_map = (
 sub has_option_type {
     my (undef, $type_or_name) = @_;
 
+    if (blessed($type_or_name)
+        && $type_or_name->isa('Moose::Meta::TypeConstraint::Union')) {
+        foreach my $union_type (@{$type_or_name->type_constraints}) {
+            return 1
+                if __PACKAGE__->has_option_type($union_type);
+        }
+        return 0;
+    }
+
     return 1 if exists $option_type_map{blessed($type_or_name) ? $type_or_name->name : $type_or_name};
 
     my $current = blessed($type_or_name) ? $type_or_name : find_type_constraint($type_or_name);
@@ -34,6 +43,16 @@ sub has_option_type {
 sub get_option_type {
     my (undef, $type_or_name) = @_;
 
+    if (blessed($type_or_name)
+        && $type_or_name->isa('Moose::Meta::TypeConstraint::Union')) {
+        foreach my $union_type (@{$type_or_name->type_constraints}) {
+            my $option_type = __PACKAGE__->get_option_type($union_type);
+            return $option_type
+                if defined $option_type;
+        }
+        return;
+    }
+
     my $name = blessed($type_or_name) ? $type_or_name->name : $type_or_name;
 
     return $option_type_map{$name} if exists $option_type_map{$name};
diff --git a/t/107_union_bug.t b/t/107_union_bug.t
new file mode 100644 (file)
index 0000000..f3283c7
--- /dev/null
@@ -0,0 +1,65 @@
+#!/usr/bin/perl
+
+use strict;
+use warnings;
+
+use Test::Most tests => 5;
+
+{
+    package example;
+    
+    use Moose;
+    use Moose::Util::TypeConstraints;
+    with qw(
+        MooseX::Getopt
+    );
+    
+    subtype 'ResultSet' 
+        => as 'DBIx::Class::ResultSet';
+    
+    subtype 'ResultList' 
+        => as 'ArrayRef[Int]';
+
+    MooseX::Getopt::OptionTypeMap->add_option_type_to_map(
+            'ResultList'  => '=s',
+    );
+    
+    coerce 'ResultList' 
+        => from 'Str' 
+        => via {
+            return [ grep { m/^\d+$/ } split /\D/,$_ ]; # <- split string into arrayref
+        };
+    
+    has 'results' => (
+        is              => 'rw',
+        isa             => 'ResultList | ResultSet', # <- union constraint
+        coerce          => 1,
+    );
+    
+    has 'other' => (
+        is              => 'rw',
+        isa             => 'Str',
+    );
+}
+
+# Without MooseX::Getopt
+{
+    my $example = example->new({
+        results => '1234,5678,9012',
+        other   => 'test',
+    });
+    isa_ok($example, 'example');
+    explain($example->results);
+    cmp_deeply($example->results, [qw(1234 5678 9012)], 'result as expected');
+}
+
+# With MooseX::Getopt
+{
+    local @ARGV = ('--results','1234,5678,9012','--other','test');
+    my $example = example->new_with_options;
+    isa_ok($example, 'example');
+    
+    explain($example->results);
+    is($example->other,'test');
+    cmp_deeply($example->results, [qw(1234 5678 9012)], 'result as expected');
+}
\ No newline at end of file