commit cef8c5bde10f793db0890dcfd18cbf20b0e69c90
Author: Colin Kuskie <colink@perldreamer.com>
Date: Tue May 24 11:52:48 2011 -0700
Build a Shop::Transaction.
commit c386079ac29fc70c1cc63d0c2844511ce9db553e
Author: Colin Kuskie <colink@perldreamer.com>
Date: Tue May 24 11:34:13 2011 -0700
Pull a session out of the object.
commit 59d780eb0ffcc82572491f85f08eb4ac04cab109
Author: Colin Kuskie <colink@perldreamer.com>
Date: Tue May 24 11:15:05 2011 -0700
Rework PayPalStd driver to create transaction before sending the user off to PayPal.
commit cd2683aa8595875f7c501f29c622abaf350e90f6
Author: Colin Kuskie <colink@perldreamer.com>
Date: Tue May 24 11:14:19 2011 -0700
Fix some comments to be more cleanly readable.
commit cf1fecfb916906c4d8ec8d33bd85c59b0aea3b7c
Author: Colin Kuskie <colink@perldreamer.com>
Date: Tue May 24 11:13:52 2011 -0700
Make the transaction optional when displaying a payment error.
commit f3c949e03a18ac513938f2ed483002c5304663d5
Author: Colin Kuskie <colink@perldreamer.com>
Date: Mon May 23 19:19:56 2011 -0700
Remove dead code.
commit 5ddcb49f094fd054f79da38c4a95dd86c55a157f
Author: Colin Kuskie <colink@perldreamer.com>
Date: Thu May 19 11:31:15 2011 -0700
If a transaction is updated with a cart object, remove all transationItems from the transaction before adding new ones.
commit 3315cb30a5c1fd4d583ce352cbc9022e52544404
Author: Colin Kuskie <colink@perldreamer.com>
Date: Thu May 19 11:30:58 2011 -0700
Remove duplicate entries from ExpressCheckout i18n
commit b46d5af528d7223d12ecbed383e798cfc376ad2a
Author: Colin Kuskie <colink@perldreamer.com>
Date: Mon May 23 16:07:33 2011 -0700
Fix the version number of the PayPal API we send. Add the serialized request to the log file. Fix i18n labels so it's clear which username and password to use.
commit 1604d375822eb013c633b72993fa524703a72127
Author: Colin Kuskie <colink@perldreamer.com>
Date: Wed May 18 15:50:42 2011 -0700
Fix copy/paste errors from other i18n modules into ExpressCheckout.
commit f2c958fc7128348a18a005bfbadf83457861d6e2
Author: Colin Kuskie <colink@perldreamer.com>
Date: Wed May 18 11:26:15 2011 -0700
Update out of date POD about checking out.
commit 61ca80b15701733a1a7c7eae5d825b161e0c71c1
Author: Colin Kuskie <colink@perldreamer.com>
Date: Mon May 23 16:05:09 2011 -0700
Fix documentation in appendCartVariables, and return the hash instead of $self.
commit d3b7341c44c924f395f8594c8ae77d8187170c9f
Author: Colin Kuskie <colink@perldreamer.com>
Date: Mon May 23 16:03:20 2011 -0700
Remove variables that were not being used.
commit 2913f96182a7630bce01998bb022d3ebf4842171
Author: Colin Kuskie <colink@perldreamer.com>
Date: Mon May 16 21:05:54 2011 -0700
Pull isRecurring directly from the Cart if creating/updating a transaction from one.
390 lines
12 KiB
Perl
390 lines
12 KiB
Perl
package WebGUI::Shop::PayDriver::PayPal::PayPalStd;
|
|
|
|
=head1 LEGAL
|
|
|
|
-------------------------------------------------------------------
|
|
WebGUI is Copyright 2001-2009 Plain Black Corporation.
|
|
-------------------------------------------------------------------
|
|
Please read the legal notices (docs/legal.txt) and the license
|
|
(docs/license.txt) that came with this distribution before using
|
|
this software.
|
|
-------------------------------------------------------------------
|
|
http://www.plainblack.com info@plainblack.com
|
|
-------------------------------------------------------------------
|
|
|
|
=cut
|
|
|
|
use strict;
|
|
|
|
use base qw/WebGUI::Shop::PayDriver::PayPal/;
|
|
|
|
use URI;
|
|
use URI::Escape;
|
|
use LWP::UserAgent;
|
|
use Readonly;
|
|
use WebGUI::Shop::Transaction;
|
|
|
|
Readonly my $I18N => 'PayDriver_PayPalStd';
|
|
|
|
=head1 NAME
|
|
|
|
PayPal Website payments standard
|
|
|
|
=head1 DESCRIPTION
|
|
|
|
A PayPal Website payments standard handler for WebGUI. Provides an interface to PayPal with cart contents
|
|
and transaction information on return.
|
|
|
|
=head2 IMPORTANT NOTE
|
|
|
|
In order to use this module, Auto Return and PDT must be enabled in your
|
|
paypal seller account. If they are not, everything will break!
|
|
|
|
=head1 SYNOPSIS
|
|
|
|
Add "WebGUI::Shop::PayDriver::PayPal::PayPalStd" to the paymentDrivers list in your WebGUI site config file.
|
|
Re-start the WebGUI modperl and modproxy web servers.
|
|
|
|
=cut
|
|
|
|
#-------------------------------------------------------------------
|
|
# local subs
|
|
|
|
#-------------------------------------------------------------------
|
|
|
|
=head2 handlesRecurring
|
|
|
|
Tells the commerce system that this payment plugin can handle recurring payments.
|
|
1 = yes, 0 = no. This module == no.
|
|
|
|
=cut
|
|
|
|
sub handlesRecurring { 0 }
|
|
|
|
#-------------------------------------------------------------------
|
|
|
|
# Recurring TX stuff removed, for now.
|
|
|
|
#-------------------------------------------------------------------
|
|
sub definition {
|
|
my $class = shift;
|
|
my $session = shift;
|
|
WebGUI::Error::InvalidParam->throw( error => q{Must provide a session variable} )
|
|
unless ref $session eq 'WebGUI::Session';
|
|
my $definition = shift;
|
|
|
|
my $i18n = WebGUI::International->new( $session, $I18N );
|
|
|
|
tie my %fields, 'Tie::IxHash';
|
|
%fields = (
|
|
vendorId => {
|
|
fieldType => 'text',
|
|
label => $i18n->get('vendorId'),
|
|
hoverHelp => $i18n->get('vendorId help'),
|
|
},
|
|
signature => {
|
|
fieldType => 'textarea',
|
|
label => $i18n->get('signature'),
|
|
hoverHelp => $i18n->get('signature help'),
|
|
},
|
|
identityToken => {
|
|
fieldType => 'text',
|
|
label => $i18n->get('identity token'),
|
|
hoverHelp => $i18n->get('identity token help'),
|
|
},
|
|
currency => {
|
|
fieldType => 'selectBox',
|
|
label => $i18n->get('currency'),
|
|
hoverHelp => $i18n->get('currency help'),
|
|
defaultValue => 'USD',
|
|
options => $class->getPaymentCurrencies(),
|
|
},
|
|
useSandbox => {
|
|
fieldType => 'yesNo',
|
|
label => $i18n->get('use sandbox'),
|
|
hoverHelp => $i18n->get('use sandbox help'),
|
|
defaultValue => 1,
|
|
},
|
|
sandboxUrl => {
|
|
fieldType => 'text',
|
|
label => $i18n->get('sandbox url'),
|
|
hoverHelp => $i18n->get('sandbox url help'),
|
|
defaultValue => 'https://www.sandbox.paypal.com/cgi-bin/webscr',
|
|
},
|
|
liveUrl => {
|
|
fieldType => 'text',
|
|
label => $i18n->get('live url'),
|
|
hoverHelp => $i18n->get('live url help'),
|
|
defaultValue => 'https://www.paypal.com/cgi-bin/webscr',
|
|
},
|
|
buttonImage => {
|
|
fieldType => 'text',
|
|
label => $i18n->get('button image'),
|
|
hoverHelp => $i18n->get('button image help'),
|
|
defaultValue => '',
|
|
},
|
|
summaryTemplateId => {
|
|
fieldType => 'template',
|
|
label => $i18n->get('summary template'),
|
|
hoverHelp => $i18n->get('summary template help'),
|
|
namespace => 'Shop/Credentials',
|
|
defaultValue => '',
|
|
},
|
|
);
|
|
|
|
push @{$definition},
|
|
{
|
|
name => $i18n->get('PayPal'),
|
|
properties => \%fields,
|
|
};
|
|
|
|
return $class->SUPER::definition( $session, $definition );
|
|
}
|
|
|
|
#-------------------------------------------------------------------
|
|
|
|
=head2 getButton
|
|
|
|
Extends the base class to add a user configurable button image.
|
|
|
|
=cut
|
|
|
|
sub getButton {
|
|
my $self = shift;
|
|
my $session = $self->session;
|
|
|
|
my $header = WebGUI::Form::formHeader(
|
|
$session, {
|
|
action => $self->payPalUrl,
|
|
method => 'POST',
|
|
}
|
|
);
|
|
|
|
# All the API stuff is done in paymentVariables; we'll just turn it into
|
|
# hidden form fields here
|
|
my $v = $self->paymentVariables;
|
|
my $transaction = $self->processTransaction();
|
|
$v->{custom} = $transaction->getId;
|
|
my $fields = join "\n", map {
|
|
WebGUI::Form::hidden( $session, { name => $_, value => $v->{$_} } )
|
|
} (keys %$v);
|
|
|
|
# Customized buttons are allowed; If they didn't give us one, we'll just
|
|
# do a submit button with i18n'd paypal text. If they did, we'll use an
|
|
# image submit.
|
|
my $button;
|
|
my $i18n = WebGUI::International->new( $session, 'PayDriver_PayPalStd' );
|
|
my $text = $i18n->get('PayPal');
|
|
if ( $self->get('buttonImage') ) {
|
|
my $raw = $self->get('buttonImage');
|
|
WebGUI::Macro::process( $session, \$raw );
|
|
$button = qq{
|
|
<input type='image'
|
|
src='$raw'
|
|
border='0'
|
|
name='submit'
|
|
alt='$text'>
|
|
};
|
|
}
|
|
else {
|
|
$button = WebGUI::Form::submit( $session, { value => $text } );
|
|
}
|
|
|
|
my $footer = WebGUI::Form::formFooter($session);
|
|
return join "\n", $header, $fields, $button, $footer;
|
|
}
|
|
|
|
#-------------------------------------------------------------------
|
|
|
|
=head2 getPayPalParams
|
|
|
|
Using the tx form variable, dial up PayPal and ask them for details about the transaction.
|
|
Return a hashreference of name/value pairs, along with PAYPAL_TX, the transactionId and
|
|
PAYPAL_REQUEST_STATUS, the HTTP code from the response from PayPal.
|
|
|
|
=cut
|
|
|
|
sub getPayPalParams {
|
|
my $self = shift;
|
|
my $session = $self->session;
|
|
# instead of relying on what was passed to us.
|
|
return $self->{_params} if $self->{_params};
|
|
my $tx = $session->form->process('tx');
|
|
|
|
my %form = (
|
|
cmd => '_notify-synch',
|
|
tx => $tx,
|
|
at => $self->get('identityToken'),
|
|
);
|
|
my $response = LWP::UserAgent->new->post($self->payPalUrl, \%form);
|
|
my ($status, @lines) = split("\n", $response->content);
|
|
my %params = map { split /=/ }
|
|
map { uri_unescape($_) } @lines;
|
|
$params{PAYPAL_REQUEST_STATUS} = $status;
|
|
$params{PAYPAL_TX} = $tx;
|
|
$self->{_params} = \%params;
|
|
return $self->{_params};
|
|
}
|
|
|
|
#-------------------------------------------------------------------
|
|
|
|
=head2 paymentVariables
|
|
|
|
Returns a hashref of the payment variables to be used as hidden form fields
|
|
when clicking the getButton button.
|
|
|
|
=cut
|
|
|
|
sub paymentVariables {
|
|
my $self = shift;
|
|
my $url = $self->session->url;
|
|
my $base = $url->getSiteURL . $url->page;
|
|
my $cart = $self->getCart;
|
|
|
|
my $return = URI->new($base);
|
|
$return->query_form( {
|
|
shop => 'pay',
|
|
method => 'do',
|
|
do => 'completeTransaction',
|
|
paymentGatewayId => $self->getId,
|
|
}
|
|
);
|
|
|
|
my $cancel = URI->new($base);
|
|
$cancel->query_form({ shop => 'cart' });
|
|
|
|
my %params = (
|
|
cmd => '_cart',
|
|
upload => 1,
|
|
business => $self->get('vendorId'),
|
|
currency_code => $self->get('currency'),
|
|
no_shipping => 1,
|
|
|
|
return => $return->as_string,
|
|
cancel_return => $cancel->as_string,
|
|
|
|
handling_cart => $cart->calculateShipping, ##According to https://www.x.com/message/180018#180018
|
|
tax_cart => $cart->calculateTaxes,
|
|
discount_amount_cart => -($cart->calculateShopCreditDeduction),
|
|
|
|
# When we verify that we have a valid transaction ID later on in
|
|
# processPayment, we'll make sure it's the cart we think it is.
|
|
custom => $cart->getId,
|
|
);
|
|
|
|
my $counter = 0;
|
|
foreach my $item (@{ $cart->getItems}) {
|
|
my $n = ++$counter;
|
|
$params{"amount_$n"} = $item->getSku->getPrice;
|
|
$params{"quantity_$n"} = $item->get('quantity');
|
|
$params{"item_name_$n"} = $item->get('configuredTitle');
|
|
$params{"item_number_$n"} = $item->get('itemId');
|
|
}
|
|
|
|
return \%params;
|
|
}
|
|
|
|
#-------------------------------------------------------------------
|
|
|
|
=head2 payPalUrl
|
|
|
|
Returns the url of the paypal gateway, taking into account useSandbox.
|
|
|
|
=cut
|
|
|
|
sub payPalUrl {
|
|
my $self = shift;
|
|
my $field = $self->get('useSandbox') ? 'sandboxUrl' : 'liveUrl';
|
|
return $self->get($field);
|
|
}
|
|
|
|
#-------------------------------------------------------------------
|
|
|
|
=head2 processPayment ( transaction )
|
|
|
|
Implements the interface defined in WebGUI::Shop::PayDriver. Notably, in case
|
|
of an error, the error is rendered as an html table of the params that paypal
|
|
passed to us.
|
|
|
|
=cut
|
|
|
|
sub processPayment {
|
|
my ( $self ) = @_;
|
|
|
|
my $success = $self->{_transactionSuccessful} || 0;
|
|
my $id = $self->{_tx} || undef;
|
|
my $status = $self->{_statusCode} || undef;
|
|
my $message = $self->{_statusMessage} || 'Waiting for checkout';
|
|
|
|
return ( $success, $id, $status, $message );
|
|
}
|
|
|
|
#-------------------------------------------------------------------
|
|
|
|
=head2 _setPaymentStatus ( transactionSuccessful, ogoneId, statusCode, statusMessage )
|
|
|
|
Update the internal status of a payment, so that the next call to processPayment
|
|
returns the correct data.
|
|
|
|
=head3 transactionSuccessful
|
|
|
|
A boolean indicating whether or not the payment was successful.
|
|
|
|
=head3 tx
|
|
|
|
The PayPal issued transaction ID.
|
|
|
|
=head3 statusCode
|
|
|
|
The PayPal issued status code.
|
|
|
|
=head3 statusMessage
|
|
|
|
An updates status message
|
|
|
|
=cut
|
|
|
|
sub _setPaymentStatus {
|
|
my ( $self ) = @_;
|
|
|
|
$self->{_transactionSuccessful} = shift || 0;
|
|
$self->{_tx} = shift || undef;
|
|
$self->{_statusCode} = shift || undef;
|
|
$self->{_statusMessage} = shift || undef;
|
|
}
|
|
|
|
#-------------------------------------------------------------------
|
|
|
|
=head2 www_completeTransaction
|
|
|
|
Where paypal comes back to when a transaction has been completed.
|
|
|
|
=cut
|
|
|
|
sub www_completeTransaction {
|
|
my $self = shift;
|
|
my $session = $self->session;
|
|
|
|
my $params = $self->getPayPalParams;
|
|
if ($params->{PAYPAL_REQUEST_STATUS} =~ /FAIL/) {
|
|
my $message = "<table><tr><th>Field</th><th>Value</th></tr>\n";
|
|
foreach my $key ( keys %{ $params } ) {
|
|
$message .= sprintf "<tr><td>%s</td><td>%s</td></tr>\n", $key, $params->{key};
|
|
}
|
|
$message .= "</table>\n";
|
|
return ( 0, $params->{PAYPAL_TX}, $params->{PAYPAL_REQUEST_STATUS}, $message );
|
|
}
|
|
my $transaction = eval { WebGUI::Shop::Transaction->new($session, $params->{custom}); };
|
|
if (my $e = Exception::Class->caught) {
|
|
return $self->displayPaymentError();
|
|
}
|
|
$self->_setPaymentStatus(1, $params->{PAYPAL_TX}, $params->{payment_status}, 'Complete');
|
|
$self->processTransaction($transaction);
|
|
|
|
return $transaction->get('isSuccessful')
|
|
? $transaction->thankYou
|
|
: $self->displayPaymentError($transaction);
|
|
}
|
|
|
|
|
|
1;
|