X-Git-Url: http://git.shadowcat.co.uk/gitweb/gitweb.cgi?a=blobdiff_plain;f=lib%2FDBIx%2FClass%2F_Util.pm;h=1407ddcbdd935af71e7694a57e28ab7e3aaf88a0;hb=a446d7f8fdc34bde8a31936f7900b77a0c210415;hp=6b8f87c1b99df4a1cab8e62d9fb849c4891cd641;hpb=841efcb3f89a643bd283ccbbf5e73b98c4a6924a;p=dbsrgits%2FDBIx-Class.git diff --git a/lib/DBIx/Class/_Util.pm b/lib/DBIx/Class/_Util.pm index 6b8f87c..1407ddc 100644 --- a/lib/DBIx/Class/_Util.pm +++ b/lib/DBIx/Class/_Util.pm @@ -54,12 +54,18 @@ BEGIN { use DBIx::Class::Carp '^DBIx::Class|^DBICTest'; use Carp 'croak'; -use Scalar::Util qw(refaddr weaken blessed reftype); +use Scalar::Util qw(weaken blessed reftype); +use List::Util qw(first); +use overload (); use base 'Exporter'; -our @EXPORT_OK = qw(sigwarn_silencer modver_gt_or_eq fail_on_internal_wantarray refcount is_exception); +our @EXPORT_OK = qw( + sigwarn_silencer modver_gt_or_eq fail_on_internal_wantarray + refcount hrefaddr is_exception + is_plain_value is_literal_value +); -sub sigwarn_silencer { +sub sigwarn_silencer ($) { my $pattern = shift; croak "Expecting a regexp" if ref $pattern ne 'Regexp'; @@ -69,7 +75,9 @@ sub sigwarn_silencer { return sub { &$orig_sig_warn unless $_[0] =~ $pattern }; } -sub refcount { +sub hrefaddr ($) { sprintf '0x%x', &Scalar::Util::refaddr } + +sub refcount ($) { croak "Expecting a reference" if ! length ref $_[0]; require B; @@ -81,6 +89,12 @@ sub refcount { sub is_exception ($) { my $e = $_[0]; + # this is not strictly correct - an eval setting $@ to undef + # is *not* the same as an eval setting $@ to '' + # but for the sake of simplicity assume the following for + # the time being + return 0 unless defined $e; + my ($not_blank, $suberror); { local $@; @@ -93,7 +107,7 @@ sub is_exception ($) { if (defined $suberror) { if (length (my $class = blessed($e) )) { carp_unique( sprintf( - 'External exception object %s=%s(0x%x) implements partial (broken) ' + 'External exception object %s=%s(%s) implements partial (broken) ' . 'overloading preventing it from being used in simple ($x eq $y) ' . 'comparisons. Given Perl\'s "globally cooperative" exception ' . 'handling this type of brokenness is extremely dangerous on ' @@ -107,7 +121,7 @@ sub is_exception ($) { . "as generated by Perl itself:\n\n%s\n ", $class, reftype $e, - refaddr $e, + hrefaddr $e, $class, 'http://v.gd/DBIC_overload_tempfix/', $suberror, @@ -126,7 +140,7 @@ sub is_exception ($) { return $not_blank; } -sub modver_gt_or_eq { +sub modver_gt_or_eq ($$) { my ($mod, $ver) = @_; croak "Nonsensical module name supplied" @@ -138,10 +152,50 @@ sub modver_gt_or_eq { local $SIG{__WARN__} = sigwarn_silencer( qr/\Qisn't numeric in subroutine entry/ ) if SPURIOUS_VERSION_CHECK_WARNINGS; + croak "$mod does not seem to provide a version (perhaps it never loaded)" + unless $mod->VERSION; + local $@; eval { $mod->VERSION($ver) } ? 1 : 0; } +sub is_literal_value ($) { + ( + ref $_[0] eq 'SCALAR' + or + ( ref $_[0] eq 'REF' and ref ${$_[0]} eq 'ARRAY' ) + ) ? 1 : 0; +} + +# FIXME XSify - this can be done so much more efficiently +sub is_plain_value ($) { + no strict 'refs'; + ( + # plain scalar + (! length ref $_[0]) + or + ( + blessed $_[0] + and + # deliberately not using Devel::OverloadInfo - the checks we are + # intersted in are much more limited than the fullblown thing, and + # this is a relatively hot piece of code + ( + # either has stringification which DBI prefers out of the box + #first { *{$_ . '::(""'}{CODE} } @{ mro::get_linear_isa( ref $_[0] ) } + overload::Method($_[0], '""') + or + # has nummification and fallback is *not* disabled + ( + $_[1] = first { *{"${_}::(0+"}{CODE} } @{ mro::get_linear_isa( ref $_[0] ) } + and + ( ! defined ${"$_[1]::()"} or ${"$_[1]::()"} ) + ) + ) + ) + ) ? 1 : 0; +} + { my $list_ctx_ok_stack_marker; @@ -174,8 +228,8 @@ sub modver_gt_or_eq { my $obj = shift; DBIx::Class::Exception->throw( sprintf ( - "Improper use of %s(0x%x) instance in list context at %s line %d\n\n\tStacktrace starts", - ref($obj), refaddr($obj), (caller($cf))[1,2] + "Improper use of %s(%s) instance in list context at %s line %d\n\n\tStacktrace starts", + ref($obj), hrefaddr($obj), (caller($cf))[1,2] ), 'with_stacktrace'); }