action is available in the content, not as a method
[Business-OnlinePayment-PayflowPro.git] / PayflowPro.pm
index 45d0fb8..86a43a0 100644 (file)
@@ -6,7 +6,8 @@ use Carp qw(croak);
 use AutoLoader;
 use Business::OnlinePayment;
 
-use PFProAPI; #Payflow PRO SDK from Verisign
+#PayflowPRO SDK from Verisign
+use PFProAPI qw( pfpro );
 
 require Exporter;
 
@@ -18,9 +19,12 @@ $VERSION = '0.02';
 sub set_defaults {
     my $self = shift;
 
-    #$self->server('staging.linkpt.net');
-    $self->server('secure.linkpt.net');
-    $self->port('1139');
+    $self->server('payflow.verisign.com');
+    $self->port('443');
+
+    $self->build_subs(qw(
+      vendor partner order_number cert_path avs_code cvv2_code
+    ));
 
 }
 
@@ -30,13 +34,25 @@ sub map_fields {
     my %content = $self->content();
 
     #ACTION MAP
-    my %actions = ('normal authorization' => 'ApproveSale',
-                   'authorization only'   => 'CapturePayment',
-                   'credit'               => 'ReturnOrder',
-                   'post authorization'   => 'BillOrders',
+    my %actions = ('normal authorization' => 'S', #Sale
+                   'authorization only'   => 'A', #Authorization
+                   'credit'               => 'C', #Credit (refund)
+                   'post authorization'   => 'D', #Delayed Capture
+                   'void'                 => 'V',
                   );
     $content{'action'} = $actions{lc($content{'action'})} || $content{'action'};
 
+    # TYPE MAP
+    my %types = ('visa'               => 'C',
+                 'mastercard'         => 'C',
+                 'american express'   => 'C',
+                 'discover'           => 'C',
+                 'cc'                 => 'C'
+                 #'check'              => 'ECHECK',
+                );
+    $content{'type'} = $types{lc($content{'type'})} || $content{'type'};
+    $self->transaction_type($content{'type'});
+
     # stuff it back into %content
     $self->content(%content);
 }
@@ -86,83 +102,116 @@ sub get_fields {
 sub submit {
     my($self) = @_;
 
-
     $self->map_fields();
 
     my %content = $self->content;
 
-    my($month, $year);
-    unless ( $content{action} eq 'BillOrders' ) {
+    my($month, $year, $zip);
 
-        if (  $self->transaction_type() =~
-                /^(cc|visa|mastercard|american express|discover)$/i
-           ) {
+    #unless ( $content{action} eq 'BillOrders' ) {
+
+        if (  $self->transaction_type() eq 'C' ) {
         } else {
-            Carp::croak("PayflowPro can't handle transaction type: ".
+            Carp::croak("PayflowPro can't (yet?) handle transaction type: ".
                         $self->transaction_type());
         }
 
-      $content{'expiration'} =~ /^(\d+)\D+\d*(\d{2})$/
-        or croak "unparsable expiration $content{expiration}";
+      if ( exists($content{'expiration'}) && defined($content{'expiration'})
+           && length($content{'expiration'})                                 ) {
+        $content{'expiration'} =~ /^(\d+)\D+\d*(\d{2})$/
+          or croak "unparsable expiration $content{expiration}";
 
-      ( $month, $year ) = ( $1, $2 );
-      $month = '0'. $month if $month =~ /^\d$/;
-      $year += 2000 if $year < 2000; #not y4k safe, oh shit
-    }
+        ( $month, $year ) = ( $1, $2 );
+        $month = '0'. $month if $month =~ /^\d$/;
+      }
 
-    $content{'address'} =~ /^(\S+)\s/;
-    my $addrnum = $1;
+      $zip = $content{'zip'} =~ s/\D//;
+    #}
 
-    $self->server('staging.linkpt.net') if $self->test_transaction;
+    #$content{'address'} =~ /^(\S+)\s/;
+    #my $addrnum = $1;
+
+    $self->server('test-payflow.verisign.com') if $self->test_transaction;
 
     $self->revmap_fields(
-      hostname     => \( $self->server ),
-      port         => \( $self->port ),
-      storename    => \( $self->storename ),
-      keyfile      => \( $self->keyfile ),
-      addrnum      => \$addrnum,
-
-      cardNumber   => 'card_number',
-      cardExpMonth => \$month,
-      cardExpYear  => \$year,
+      ACCT       => 'card_number',
+      EXPDATE     => \( $month.$year ),
+      AMT         => 'amount',
+      USER        => 'login',
+      #VENDOR      => \( $self->vendor ),
+      VENDOR      => 'login',
+      PARTNER     => \( $self->partner ),
+      PWD         => 'password',
+      TRXTYPE     => 'action',
+      TENDER      => 'type',
+
+      STREET      => 'address',
+      ZIP         => \$zip,
+
+      CITY        => 'city',
+      COMMENT1    => 'description',
+      COMMENT2    => 'invoice_number',
+      COMPANYNAME => 'company',
+      COUNTRY     => 'country',
+      FIRSTNAME   => 'first_name',
+      LASTNAME    => 'last_name',
+      NAME        => 'name',
+      EMAIL       => 'email',
+      STATE       => 'state',
+
+      CVV2        => 'cvv2',
+      ORIGID      => 'order_number'
+
     );
 
-    my $lperl = new LPERL
-      $self->lbin,
-      'FILE',
-      $self->can('tmp')
-        ? $self->tmp
-        : '/tmp';
-    my $action = $content{action};
-
-    $self->required_fields(qw/
-      hostname port storename keyfile amount cardNumber cardExpMonth cardExpYear
-    /);
-
-    my %post_data = $self->get_fields(qw/
-      hostname port storename keyfile
-      result
-      amount cardNumber cardExpMonth cardExpYear
-      name email phone address city state zip country
-    /);
-
-    #print "$_ => $post_data{$_}\n" foreach keys %post_data;
-
-    my %response;
-    {
-      local($^W)=0;
-      %response = $lperl->$action(\%post_data);
+    my @required = qw( TRXTYPE TENDER PARTNER VENDOR USER PWD );
+    if (  $self->transaction_type() eq 'C' ) { #credit card
+      if ( $content{'action'} =~ /^[CDV]$/ && exists($content{'ORIGID'})
+           && defined($content{'ORIGID'}) && length($content{'ORIGID'}) ) {
+        push @required, qw(ORIGID);
+      } else {
+        push @required, qw(AMT ACCT EXPDATE);
+      }
     }
+    $self->required_fields(@required);
 
-    if ( $response{'statusCode'} == 0 ) {
-      $self->is_success(0);
-      $self->result_code('');
-      $self->error_message($response{'statusMessage'});
-    } else {
+    my %params = $self->get_fields(qw(
+      ACCT EXPDATE AMT USER VENDOR PARTNER PWD TRXTYPE TENDER
+      STREET ZIP
+      CITY COMMENT1 COMMENT2 COMPANYNAME COUNTRY FIRSTNAME LASTNAME NAME EMAIL
+        STATE
+      CVV2 ORIGID
+    ));
+
+    #print "$_ => $params{$_}\n" foreach keys %params;
+
+    $ENV{'PFPRO_CERT_PATH'} = $self->cert_path;
+    my( $response, $resultstr ) = pfpro( \%params, $self->server, $self->port );
+
+    #if ( $response->{'RESULT'} == 0 ) {
+    if ( $response->{'RESULT'} eq '0' ) { #want an explicit zero, not just
+                                          #numerically equal
       $self->is_success(1);
-      $self->result_code($response{'AVCCode'});
-      $self->authorization($response{'trackingID'});
-#      $self->order_number($response{'neworderID'});
+      $self->result_code(   $response->{'RESULT'}   );
+      $self->error_message( $response->{'RESPMSG'}  );
+      $self->authorization( $response->{'AUTHCODE'} );
+      $self->order_number(  $response->{'PNREF'}    );
+      my $avs_code = '';
+      if ( $response->{AVSADDR} eq 'Y' && $response->{AVSZIP} eq 'Y' ) {
+        $avs_code = 'Y';
+      } elsif ( $response->{AVSADDR} eq 'Y' ) {
+        $avs_code = 'A';
+      } elsif ( $response->{AVSZIP} eq 'Y' ) {
+        $avs_code = 'Z';
+      } elsif ( $response->{AVSADDR} eq 'N' || $response->{AVSZIP} eq 'N' ) {
+        $avs_code = 'N';
+      }
+      $self->avs_code(      $avs_code               );
+      $self->cvv2_code(     $response->{'CVV2MATCH'});
+    } else {
+      $self->is_success(0);
+      $self->result_code(   $response->{'RESULT'}  );
+      $self->error_message( $response->{'RESPMSG'} );
     }
 
 }
@@ -179,10 +228,9 @@ Business::OnlinePayment::PayflowPro - Verisign PayflowPro backend for Business::
   use Business::OnlinePayment;
 
   my $tx = new Business::OnlinePayment( 'PayflowPro',
-    'storename' => 'your_store_number',
-    'keyfile'   => '/path/to/keyfile.pem',
-    'lbin'      => '/path/to/binary/lbin',
-    'tmp'       => '/secure/tmp',          # a secure tmp directory
+    'vendor'    => 'your_vendor',
+    'partner'   => 'your_partner',
+    'cert_path' => '/path/to/your/certificate/file/', #just the dir
   );
 
   $tx->content(
@@ -199,19 +247,41 @@ Business::OnlinePayment::PayflowPro - Verisign PayflowPro backend for Business::
       zip            => '84058',
       email          => 'ivan-payflowpro@420.am',
       card_number    => '4007000000027',
-      expiration     => '09/99',
+      expiration     => '09/04',
+
+      #advanced params
+      cvv2           => '420',
+      order_number   => 'string', # returned by $tx->order_number() from an
+                                  # "authorization only" or
+                                  # "normal authorization" action, used by a
+                                  # "credit", "void", or "post authorization"
   );
   $tx->submit();
 
   if($tx->is_success()) {
       print "Card processed successfully: ".$tx->authorization."\n";
+      print "order number: ". $tx->order_number. "\n";
+      print "AVS code: ". $tx->avs_code. "\n"; # Y - Address and ZIP match
+                                               # A - Address matches but not ZIP
+                                               # Z - ZIP matches bu tnot address
+                                               # N - no match
+                                               # E - AVS error or unsupported
+                                               # (null) - AVS error
+      print "CVV2 code: ". $tx->cvv2_code. "\n";
+
   } else {
-      print "Card was rejected: ".$tx->error_message."\n";
+      print "Card was rejected: ".$tx->error_message;
+      print " (CVV2 mismatch)" if $tx->result_code == 114;
+      print "\n";
   }
 
 =head1 SUPPORTED TRANSACTION TYPES
 
-=head2 Visa, MasterCard, American Express, JCB, Discover/Novus, Carte blanche/Diners Club
+=head2 Visa, MasterCard, American Express, JCB, Discover/Novus, Carte blanche/Diners Club, CC
+
+=head1 SUPPORTED ACTIONS
+
+=head2 Normal Authorization, Authorization Only, Post Authorization, Credit, Void
 
 =head1 DESCRIPTION