]> git.vpit.fr Git - perl/modules/Test-Valgrind.git/commitdiff
Move and improve the subclass validation logic into a new helper module
authorVincent Pit <perl@profvince.com>
Wed, 11 Nov 2015 13:43:09 +0000 (11:43 -0200)
committerVincent Pit <perl@profvince.com>
Thu, 12 Nov 2015 15:12:58 +0000 (13:12 -0200)
MANIFEST
lib/Test/Valgrind/Action.pm
lib/Test/Valgrind/Command.pm
lib/Test/Valgrind/Tool.pm
lib/Test/Valgrind/Util.pm [new file with mode: 0644]

index 6d572f097c12c5d6c36ff031c7f101dbc1fb11c4..3421317201002f264160dd6471e2ea024a3e78af 100644 (file)
--- a/MANIFEST
+++ b/MANIFEST
@@ -27,6 +27,7 @@ lib/Test/Valgrind/Session.pm
 lib/Test/Valgrind/Suppressions.pm
 lib/Test/Valgrind/Tool.pm
 lib/Test/Valgrind/Tool/memcheck.pm
+lib/Test/Valgrind/Util.pm
 samples/map.pl
 samples/xml-output-protocol4.txt
 samples/xml-output.txt
index fbdffe639da60c1a0b1fba5fab8f7fab368a0c09..ca3fbd2abd3460ef341d3a9cf212d804e64c73fc 100644 (file)
@@ -23,6 +23,8 @@ Actions are called each time a tool encounter an error and decide what to do wit
 
 =cut
 
+use Test::Valgrind::Util;
+
 use base qw<Test::Valgrind::Component Test::Valgrind::Carp>;
 
 =head1 METHODS
@@ -43,11 +45,10 @@ sub new {
  my %args = @_;
 
  if ($class eq __PACKAGE__) {
-  my $action = delete $args{action} || 'Test';
-  $action =~ s/[^\w:]//g;
-  $action = __PACKAGE__ . "::$action" if $action !~ /::/;
-  $class->_croak("Couldn't load action $action: $@")
-                                               unless eval "require $action; 1";
+  my ($action, $msg) = Test::Valgrind::Util::validate_subclass(
+   $args{action} || 'Test',
+  );
+  $class->_croak($msg) unless defined $action;
   return $action->new(%args);
  }
 
index a305ed65b361143bbad8f530974891dfb1d26fa4..be786e9d281542d59f104acd202658dd8de75a55 100644 (file)
@@ -24,6 +24,8 @@ They should also provide a default setup for generating suppressions.
 
 =cut
 
+use Test::Valgrind::Util;
+
 use base qw<Test::Valgrind::Carp>;
 
 =head1 METHODS
@@ -48,10 +50,10 @@ sub new {
 
  my %args = @_;
 
- if ($class eq __PACKAGE__ and my $cmd = delete $args{command}) {
-  $cmd =~ s/[^\w:]//g;
-  $cmd = __PACKAGE__ . "::$cmd" if $cmd !~ /::/;
-  $class->_croak("Couldn't load command $cmd: $@") unless eval "require $cmd;1";
+ my $cmd = delete $args{command};
+ if ($class eq __PACKAGE__ and defined $cmd) {
+  ($cmd, my $msg) = Test::Valgrind::Util::validate_subclass($cmd);
+  $class->_croak($msg) unless defined $cmd;
   return $cmd->new(%args);
  }
 
index bb7adda0fa9e4d3013cf7b958dbb25bb0c39bd20..0961fb8c896d867589ff3b080000e0025f4141e0 100644 (file)
@@ -24,6 +24,8 @@ They are expected to function both in suppressions generation and in analysis mo
 
 =cut
 
+use Test::Valgrind::Util;
+
 use base qw<Test::Valgrind::Component Test::Valgrind::Carp>;
 
 =head1 METHODS
@@ -55,10 +57,10 @@ sub new {
  my %args = @_;
 
  if ($class eq __PACKAGE__) {
-  my $tool = delete $args{tool} || 'memcheck';
-  $tool =~ s/[^\w:]//g;
-  $tool = __PACKAGE__ . "::$tool" if $tool !~ /::/;
-  $class->_croak("Couldn't load tool $tool: $@") unless eval "require $tool; 1";
+  my ($tool, $msg) = Test::Valgrind::Util::validate_subclass(
+   delete $args{tool} || 'memcheck',
+  );
+  $class->_croak($msg) unless defined $tool;
   return $tool->new(%args);
  }
 
diff --git a/lib/Test/Valgrind/Util.pm b/lib/Test/Valgrind/Util.pm
new file mode 100644 (file)
index 0000000..441ce69
--- /dev/null
@@ -0,0 +1,88 @@
+package Test::Valgrind::Util;
+
+use strict;
+use warnings;
+
+=head1 NAME
+
+Test::Valgrind::Util - Utility routines for Test::Valgrind.
+
+=head1 VERSION
+
+Version 1.15
+
+=cut
+
+our $VERSION = '1.15';
+
+=head1 DESCRIPTION
+
+This module contains some helpers used by Test::Valgrind.
+It is not really designed to be used anywhere else.
+
+=head1 FUNCTIONS
+
+=head2 C<validate_subclass>
+
+    my ($validated_type, $error_msg) = validate_subclass($type);
+
+Try to interpret C<$type> as a subclass of the caller package, and load it if its C<@ISA> is empty.
+Returns the validated type, or C<undef> and the relevant error message.
+
+=cut
+
+sub validate_subclass {
+ my ($type) = @_;
+
+ my $base = (caller 0)[0];
+
+ $type =~ s/[^A-Za-z0-9_:]//g;
+ $type =  "${base}::$type" if $type !~ /::/;
+
+ my $stash = do { no strict 'refs'; \%{"${type}::"} };
+ my $ISA   = ($stash && $stash->{ISA}) ? *{$stash->{ISA}}{ARRAY} : undef;
+
+ unless ($ISA and @$ISA >= 1) {
+  local $@;
+  eval "require $type; 1" or return (undef, "Could not load subclass: $@");
+ }
+
+ return (undef, "$type is not a subclass of $base") unless $type->isa($base);
+
+ return $type;
+}
+
+=head1 EXPORT
+
+This module does not export anything.
+
+=head1 SEE ALSO
+
+L<Test::Valgrind>.
+
+=head1 AUTHOR
+
+Vincent Pit, C<< <perl at profvince.com> >>, L<http://www.profvince.com>.
+
+You can contact me by mail or on C<irc.perl.org> (vincent).
+
+=head1 BUGS
+
+Please report any bugs or feature requests to C<bug-test-valgrind at rt.cpan.org>, or through the web interface at L<http://rt.cpan.org/NoAuth/ReportBug.html?Queue=Test-Valgrind>.
+I will be notified, and then you'll automatically be notified of progress on your bug as I make changes.
+
+=head1 SUPPORT
+
+You can find documentation for this module with the perldoc command.
+
+    perldoc Test::Valgrind::Component
+
+=head1 COPYRIGHT & LICENSE
+
+Copyright 2015 Vincent Pit, all rights reserved.
+
+This program is free software; you can redistribute it and/or modify it under the same terms as Perl itself.
+
+=cut
+
+1; # End of Test::Valgrind::Util