summaryrefslogtreecommitdiffstats
path: root/lib
diff options
context:
space:
mode:
authorJesse Luehrs <doy@tozt.net>2012-10-09 20:52:44 -0500
committerJesse Luehrs <doy@tozt.net>2012-10-09 21:11:49 -0500
commitbb2bf61723962222952d2f68da497f8391dc6a5e (patch)
tree8b7f1eaba5805b31eb5b13523521896d9063acef /lib
parent187562dd74bec5692831c08e8d8edf54f91ce031 (diff)
downloadmoosex-validation-doctypes-bb2bf61723962222952d2f68da497f8391dc6a5e.tar.gz
moosex-validation-doctypes-bb2bf61723962222952d2f68da497f8391dc6a5e.zip
initial implementation
Diffstat (limited to 'lib')
-rw-r--r--lib/MooseX/Meta/TypeConstraint/Doctype.pm133
-rw-r--r--lib/MooseX/Validation/Doctypes.pm35
-rw-r--r--lib/MooseX/Validation/Doctypes/Errors.pm17
3 files changed, 185 insertions, 0 deletions
diff --git a/lib/MooseX/Meta/TypeConstraint/Doctype.pm b/lib/MooseX/Meta/TypeConstraint/Doctype.pm
new file mode 100644
index 0000000..0d333cc
--- /dev/null
+++ b/lib/MooseX/Meta/TypeConstraint/Doctype.pm
@@ -0,0 +1,133 @@
+package MooseX::Meta::TypeConstraint::Doctype;
+use Moose;
+
+use Devel::PartialDump 'dump';
+use Moose::Util::TypeConstraints qw(class_type find_type_constraint
+ match_on_type);
+use Scalar::Util 'weaken';
+
+use MooseX::Validation::Doctypes::Errors;
+
+extends 'Moose::Meta::TypeConstraint';
+
+class_type('Moose::Meta::TypeConstraint');
+
+has doctype => (
+ is => 'ro',
+ isa => 'Ref',
+ required => 1,
+);
+
+has '+parent' => (
+ default => sub { find_type_constraint('Ref') },
+);
+
+has '+constraint' => (
+ lazy => 1,
+ default => sub {
+ weaken(my $self = shift);
+ return sub { !$self->validate_doctype($_) };
+ },
+);
+
+has '+message' => (
+ default => sub {
+ weaken(my $self = shift);
+ return sub { $self->validate_doctype($_) };
+ },
+);
+
+sub validate_doctype {
+ my $self = shift;
+ my ($data, $doctype, $prefix) = @_;
+
+ $doctype = $self->doctype
+ unless defined $doctype;
+ $prefix = ''
+ unless defined $prefix;
+
+ my ($errors, $extra_data);
+
+ match_on_type $doctype => (
+ 'HashRef' => sub {
+ if (!find_type_constraint('HashRef')->check($data)) {
+ $errors = $data;
+ }
+ else {
+ for my $key (keys %$doctype) {
+ my $sub_errors = $self->validate_doctype(
+ $data->{$key},
+ $doctype->{$key},
+ join('.', (length($prefix) ? $prefix : ()), $key)
+ );
+ if ($sub_errors) {
+ if ($sub_errors->has_errors) {
+ $errors ||= {};
+ $errors->{$key} = $sub_errors->errors;
+ }
+ if ($sub_errors->has_extra_data) {
+ $extra_data ||= {};
+ $extra_data->{$key} = $sub_errors->extra_data;
+ }
+ }
+ }
+ for my $key (keys %$data) {
+ if (!exists $doctype->{$key}) {
+ $extra_data ||= {};
+ $extra_data->{$key} = $data->{$key};
+ }
+ }
+ }
+ },
+ 'ArrayRef' => sub {
+ if (!find_type_constraint('ArrayRef')->check($data)) {
+ $errors = $data;
+ }
+ else {
+ for my $i (0..$#$doctype) {
+ my $sub_errors = $self->validate_doctype(
+ $data->[$i],
+ $doctype->[$i],
+ join('.', (length($prefix) ? $prefix : ()), "[$i]")
+ );
+ if ($sub_errors) {
+ if ($sub_errors->has_errors) {
+ $errors ||= [];
+ $errors->[$i] = $sub_errors->errors;
+ }
+ if ($sub_errors->has_extra_data) {
+ $extra_data ||= [];
+ $extra_data->[$i] = $sub_errors->extra_data;
+ }
+ }
+ }
+ for my $i (0..$#$data) {
+ next if $i < @$doctype;
+ $extra_data ||= [];
+ $extra_data->[$i] = $data->[$i];
+ }
+ }
+ },
+ 'Str|Moose::Meta::TypeConstraint' => sub {
+ my $tc = Moose::Util::TypeConstraints::find_or_parse_type_constraint($doctype);
+ die "Unknown type $doctype" unless $tc;
+ if (!$tc->check($data)) {
+ $errors = "invalid value " . dump($data) . " for '$prefix'";
+ }
+ },
+ => sub {
+ die "Unknown doctype at position '$prefix': " . dump($doctype);
+ },
+ );
+
+ return unless $errors || $extra_data;
+
+ return MooseX::Validation::Doctypes::Errors->new(
+ ($errors ? (errors => $errors) : ()),
+ ($extra_data ? (extra_data => $extra_data) : ()),
+ );
+}
+
+no Moose;
+
+1;
diff --git a/lib/MooseX/Validation/Doctypes.pm b/lib/MooseX/Validation/Doctypes.pm
index e69de29..668c4f0 100644
--- a/lib/MooseX/Validation/Doctypes.pm
+++ b/lib/MooseX/Validation/Doctypes.pm
@@ -0,0 +1,35 @@
+package MooseX::Validation::Doctypes;
+use strict;
+use warnings;
+
+use MooseX::Meta::TypeConstraint::Doctype;
+
+use Sub::Exporter -setup => {
+ exports => ['doctype'],
+ groups => {
+ default => ['doctype'],
+ },
+};
+
+sub doctype {
+ my $name;
+ $name = shift if @_ > 1;
+
+ my ($doctype) = @_;
+
+ # XXX validate name
+
+ my $args = {
+ ($name ? (name => $name) : ()),
+ doctype => $doctype,
+ package_defined_in => scalar(caller),
+ };
+
+ my $tc = MooseX::Meta::TypeConstraint::Doctype->new($args);
+ Moose::Util::TypeConstraints::register_type_constraint($tc)
+ if $name;
+
+ return $tc;
+}
+
+1;
diff --git a/lib/MooseX/Validation/Doctypes/Errors.pm b/lib/MooseX/Validation/Doctypes/Errors.pm
new file mode 100644
index 0000000..d510670
--- /dev/null
+++ b/lib/MooseX/Validation/Doctypes/Errors.pm
@@ -0,0 +1,17 @@
+package MooseX::Validation::Doctypes::Errors;
+use Moose;
+
+has errors => (
+ is => 'ro',
+ predicate => 'has_errors',
+);
+
+has extra_data => (
+ is => 'ro',
+ predicate => 'has_extra_data',
+);
+
+__PACKAGE__->meta->make_immutable;
+no Moose;
+
+1;