blob: d4731cca6815ac52978952814a6ef062ed5a3ae6 (
plain) (
blame)
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
|
#!/usr/bin/perl
package MooseX::Role::Matcher;
use MooseX::Role::Parameterized;
use List::Util qw/first/;
use List::MoreUtils qw/any/;
parameter default_match => (
is => 'ro',
isa => 'Str',
);
role {
my $p = shift;
my $default = $p->default_match;
method _apply_to_matches => sub {
my $class = shift;
my $on_match = shift;
my @list = @{ shift() };
my @matchers = @_;
unshift @matchers, $default if (@_ % 2 == 1);
$on_match->(sub { $_->match(@matchers) }, @list);
};
method first_match => sub {
my $class = shift;
$class->_apply_to_matches(\&first, @_);
};
method grep_matches => sub {
my $class = shift;
my $grep = sub { my $code = shift; grep { $code->() } @_ };
$class->_apply_to_matches($grep, @_);
};
method any_match => sub {
my $class = shift;
$class->_apply_to_matches(\&any, @_);
};
method _match => sub {
my $self = shift;
my $value = shift;
my $seek = shift;
return !defined $value if !defined $seek;
return 0 if !defined $value;
return $value =~ $seek if ref($seek) eq 'Regexp';
return $seek->($value) if ref($seek) eq 'CODE';
if (ref($seek) eq 'ARRAY') {
for (@$seek) {
return 1 if $self->_match($value => $_);
}
}
return $value eq $seek;
};
method match => sub {
my $self = shift;
my %args = @_;
# All the conditions must be true for true to be returned. Return
# immediately if a false condition is found.
for my $matcher (keys %args) {
my ($invert, $name) = $matcher =~ /^(!)?(.*)$/;
my $value = $self->can($name) ? $self->$name : undef;
my $seek = $args{$matcher};
my $matched = $self->_match($value => $seek) ? 1 : 0;
if ($invert) {
return 0 if $matched;
}
else {
return 0 unless $matched;
}
}
return 1;
};
};
no MooseX::Role::Parameterized;
1;
|