use strict;
use warnings;
+use namespace::autoclean;
+use Moose ();
+use MooseX::ClassAttribute ();
+use Moose::Exporter;
+use Moose::Util::MetaRole;
+use Carp;
+
+Moose::Exporter->setup_import_methods;
+
=head1 NAME
MooseX::AlwaysCoerce - Automatically enable coercions for Moose attributes
=head1 VERSION
-Version 0.01
+Version 0.06
=cut
-our $VERSION = '0.01';
+our $VERSION = '0.06';
=head1 SYNOPSIS
use MooseX::AlwaysCoerce;
use MyTypeLib 'SomeType';
- has foo => (is => 'rw', isa => SomeType); # will be coerced
+ has foo => (is => 'rw', isa => SomeType); # coerce => 1 automatically added
+
+ # same, MooseX::ClassAttribute is automatically applied
+ class_has bar => (is => 'rw', isa => SomeType);
+
+=head1 DESCRIPTION
+
+Have you ever spent an hour or more trying to figure out "WTF, why did my
+coercion not run?" only to find out that you forgot C<< coerce => 1 >> ?
+
+Just load this module in your L<Moose> class and C<< coerce => 1 >> will be
+enabled for every attribute and class attribute automatically.
+
+Use C<< coerce => 0 >> to disable a coercion explicitly.
+
+=cut
+
+{
+ package MooseX::AlwaysCoerce::Role::Meta::Attribute;
+ use namespace::autoclean;
+ use Moose::Role;
+
+ has coerce => (
+ lazy => 1,
+ reader => "should_coerce",
+ default => sub {
+ return 1 if shift->type_constraint->has_coercion;
+ return 0;
+ }
+ );
+
+
+ package MooseX::AlwaysCoerce::Role::Meta::Class;
+ use namespace::autoclean;
+ use Moose::Role;
+ use Moose::Util::TypeConstraints;
+
+ around add_class_attribute => sub {
+ my $next = shift;
+ my $self = shift;
+ my ($what, %opts) = @_;
+
+ my $type = Moose::Util::TypeConstraints::find_or_parse_type_constraint($opts{isa});
+ $opts{coerce} = 1 if !exists $opts{coerce} and $type->has_coercion;
+
+ $self->$next($what, %opts);
+ };
+}
+
+my (undef, undef, $init_meta) = Moose::Exporter->build_import_methods(
+ install => [ qw(import unimport) ],
+ class_metaroles => {
+ attribute => ['MooseX::AlwaysCoerce::Role::Meta::Attribute'],
+ class => ['MooseX::AlwaysCoerce::Role::Meta::Class'],
+ },
+);
+
+sub init_meta {
+ my ($class, %options) = @_;
+ my $for_class = $options{for_class};
+
+ MooseX::ClassAttribute->import({ into => $for_class });
+
+ # call generated method to do the rest of the work.
+ goto $init_meta;
+}
=head1 AUTHOR
Rafael Kitover, C<< <rkitover at cpan.org> >>
+=head1 CONTRIBUTORS
+
+Schwern: Michael G. Schwern <mschwern@cpan.org>
+Ether: Karen Etheridge <ether@cpan.org>
+
=head1 BUGS
Please report any bugs or feature requests to C<bug-moosex-alwayscoerce at rt.cpan.org>, or through
My own stupidity, for inspiring me to write this module.
+Dave Rolsky, for telling me how to do it the L<Moose> way.
+
=head1 COPYRIGHT & LICENSE
Copyright (c) 2009 Rafael Kitover