- Documentation: Add link to Neil Bowers' review of CC check modules
authorIvan Kohler <ivan@freeside.biz>
Sat, 22 Sep 2012 07:30:31 +0000 (00:30 -0700)
committerIvan Kohler <ivan@freeside.biz>
Sat, 22 Sep 2012 07:30:31 +0000 (00:30 -0700)
        - Rearrange cardtype() for performance

Changes
CreditCard.pm

diff --git a/Changes b/Changes
index 2a413fb..0d352a7 100644 (file)
--- a/Changes
+++ b/Changes
@@ -5,6 +5,7 @@ Revision history for Perl extension Business::CreditCard.
         - Add LICENSE=>perl to Makefile.PL; add license to META.yml,
           closes: cpan #27735
         - Documentation: Add link to Neil Bowers' review of CC check modules
+        - Rearrange cardtype() for performance
 
 0.31  Mon Oct 19 18:51:35 PDT 2009
         - Add LICENSE section to POD documentation
index d657e42..d0dd3b8 100644 (file)
@@ -165,6 +165,13 @@ providing credit card number verification (LUHN checking).
 
 @EXPORT = qw(cardtype validate generate_last_digit);
 
+## ref http://neilb.org/reviews/luhn.html#Comparison it looks like
+## Business::CCCheck is 2x faster than we are.  looking at their implementation
+## not entirely a fair comparison, we also do the equivalent of their CC_clean,
+## they don't recognize certain cards at all (i.e. Switch) which require
+## an expensive check before VISA, Diners doesn't exist anymore, Discover is
+## a lot more than just 6011*, they don't handle processing agreements, etc.
+
 sub cardtype {
     my ($number) = @_;
 
@@ -183,17 +190,14 @@ sub cardtype {
             && 0+$number;
     }
 
-    return "Switch"
-      if $number =~ /^49(03(0[2-9]|3[5-9])|11(0[1-2]|7[4-9]|8[1-2])|36[0-9]{2})[\dx]{10}([\dx]{2,3})?$/o
-      || $number =~ /^564182[\dx]{10}([\dx]{2,3})?$/o
-      || $number =~ /^6(3(33[0-4][0-9])|759[0-9]{2})[\dx]{10}([\dx]{2,3})?$/o;
-
-    return "VISA card" if $number =~ /^4[\dx]{12}([\dx]{3})?$/o;
+    return "VISA card" if $number =~ /^4[0-8][\dx]{11}([\dx]{3})?$/o;
 
     return "MasterCard"
       if   $number =~ /^5[1-5][\dx]{14}$/o
       ;# || ( $number =~ /^36[\dx]{12}/ && $Country =~ /^(US|CA)$/oi );
 
+    return "American Express card" if $number =~ /^3[47][\dx]{13}$/o;
+
     return "Discover card"
       if   $number =~ /^30[0-5][\dx]{11}([\dx]{2})?$/o  #diner's: 300-305
       ||   $number =~ /^3095[\dx]{10}([\dx]{2})?$/o     #diner's: 3095
@@ -204,7 +208,12 @@ sub cardtype {
       || ( $number =~ /^62[24-68][\dx]{13}$/o && uc($Country) ne 'CN' ) #CUP
       || ( $number =~ /^35(2[89]|[3-8][\dx])[\dx]{10}$/o && uc($Country) eq 'US' );
 
-    return "American Express card" if $number =~ /^3[47][\dx]{13}$/o;
+    return "Switch"
+      if $number =~ /^49(03(0[2-9]|3[5-9])|11(0[1-2]|7[4-9]|8[1-2])|36[0-9]{2})[\dx]{10}([\dx]{2,3})?$/o
+      || $number =~ /^564182[\dx]{10}([\dx]{2,3})?$/o
+      || $number =~ /^6(3(33[0-4][0-9])|759[0-9]{2})[\dx]{10}([\dx]{2,3})?$/o;
+    #redunant with above, catch 49* that's not Switch
+    return "VISA card" if $number =~ /^4[\dx]{12}([\dx]{3})?$/o;
 
     #return "Diner's Club/Carte Blanche"
     #  if $number =~ /^3(0[0-59]|[68][\dx])[\dx]{11}$/o;
@@ -248,6 +257,21 @@ sub generate_last_digit {
     return (10 - $sum % 10) % 10;
 }
 
+
+## this (GPLed) code from Business::CCCheck is apparantly 4x faster than ours
+## ref http://neilb.org/reviews/luhn.html#Comparison
+## maybe see if we can spped ours up a bit
+#  my @ccn = split('',$ccn);
+#  my $even = 0;
+#  $ccn = 0;
+#  for($i=$#ccn;$i >=0;--$i) {
+#    $ccn[$i] *= 2 if $even;
+#    $ccn -= 9 if $ccn[$i] > 9;
+#    $ccn += $ccn[$i];
+#    $even = ! $even;
+#  }
+#  $type = '' if $ccn % 10;
+#  return $type;
 sub validate {
     my ($number) = @_;