2009-12-02 16:44:18 +00:00
|
|
|
package Business::PayPal::API::RecurringPayments;
|
|
|
|
|
|
|
|
use 5.008001;
|
|
|
|
use strict;
|
|
|
|
use warnings;
|
|
|
|
|
|
|
|
use SOAP::Lite 0.67;
|
|
|
|
use Business::PayPal::API ();
|
|
|
|
|
2014-03-23 02:31:50 +00:00
|
|
|
our @ISA = qw(Business::PayPal::API);
|
2009-12-02 16:44:18 +00:00
|
|
|
our @EXPORT_OK = qw( SetCustomerBillingAgreement
|
2014-03-23 02:31:50 +00:00
|
|
|
GetBillingAgreementCustomerDetails
|
|
|
|
CreateRecurringPaymentsProfile
|
|
|
|
DoReferenceTransaction);
|
2009-12-02 16:44:18 +00:00
|
|
|
|
|
|
|
our $API_VERSION = '50.0';
|
|
|
|
|
|
|
|
sub SetCustomerBillingAgreement {
|
|
|
|
my $self = shift;
|
|
|
|
my %args = @_;
|
|
|
|
|
|
|
|
## billing agreement details type
|
2014-03-23 02:31:50 +00:00
|
|
|
my %badtypes = (
|
|
|
|
BillingType => '', # 'ns:BillingCodeType',
|
|
|
|
BillingAgreementDescription => 'xs:string',
|
|
|
|
PaymentType => '', # 'ns:MerchantPullPaymentCodeType',
|
|
|
|
BillingAgreementCustom => 'xs:string',
|
|
|
|
);
|
|
|
|
|
|
|
|
my %types = (
|
|
|
|
BillingAgreementDetails => 'ns:BillingAgreementDetailsType',
|
|
|
|
ReturnURL => 'xs:string',
|
|
|
|
CancelURL => 'xs:string',
|
|
|
|
LocaleCode => 'xs:string',
|
|
|
|
PageStyle => 'xs:string',
|
|
|
|
'cpp-header-image' => 'xs:string',
|
|
|
|
'cpp-header-border-color' => 'xs:string',
|
|
|
|
'cpp-header-back-color' => 'xs:string',
|
|
|
|
'cpp-payflow-color' => 'xs:string',
|
|
|
|
PaymentAction => '',
|
|
|
|
BuyerEmail => 'ns:EmailAddressType',
|
|
|
|
);
|
2009-12-02 16:44:18 +00:00
|
|
|
|
|
|
|
## set defaults
|
2014-03-23 02:31:50 +00:00
|
|
|
$args{BillingType} ||= 'RecurringPayments';
|
|
|
|
$args{PaymentType} ||= 'InstantOnly';
|
|
|
|
$args{currencyID} ||= 'USD';
|
2009-12-02 16:44:18 +00:00
|
|
|
|
|
|
|
my @btypes = ();
|
|
|
|
for my $field ( keys %badtypes ) {
|
|
|
|
next unless $args{$field};
|
2014-03-23 02:31:50 +00:00
|
|
|
push @btypes,
|
|
|
|
SOAP::Data->name( $field => $args{$field} )
|
|
|
|
->type( $badtypes{$field} );
|
2009-12-02 16:44:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
my @scba = ();
|
|
|
|
for my $field ( keys %types ) {
|
|
|
|
next unless $args{$field};
|
2014-03-23 02:31:50 +00:00
|
|
|
push @scba,
|
|
|
|
SOAP::Data->name( $field => $args{$field} )
|
|
|
|
->type( $types{$field} );
|
2009-12-02 16:44:18 +00:00
|
|
|
}
|
2014-03-23 02:31:50 +00:00
|
|
|
push @scba,
|
|
|
|
SOAP::Data->name(
|
|
|
|
BillingAgreementDetails => \SOAP::Data->value( @btypes ) );
|
|
|
|
|
|
|
|
my $request = SOAP::Data->name(
|
|
|
|
SetCustomerBillingAgreementRequest => \SOAP::Data->value(
|
|
|
|
$self->version_req, #$API_VERSION,
|
|
|
|
SOAP::Data->name(
|
|
|
|
SetCustomerBillingAgreementRequestDetails =>
|
|
|
|
\SOAP::Data->value( @scba )
|
|
|
|
)->attr( { xmlns => $self->C_xmlns_ebay } ),
|
|
|
|
)
|
|
|
|
)->type( 'ns:SetCustomerBillingAgreementRequestDetailsType' );
|
2009-12-02 16:44:18 +00:00
|
|
|
|
|
|
|
my $som = $self->doCall( SetCustomerBillingAgreementReq => $request )
|
2014-03-23 02:31:50 +00:00
|
|
|
or return;
|
2009-12-02 16:44:18 +00:00
|
|
|
|
|
|
|
my $path = '/Envelope/Body/SetCustomerBillingAgreementResponse';
|
|
|
|
|
|
|
|
my %response = ();
|
2014-03-23 02:31:50 +00:00
|
|
|
unless ( $self->getBasic( $som, $path, \%response ) ) {
|
|
|
|
$self->getErrors( $som, $path, \%response );
|
2009-12-02 16:44:18 +00:00
|
|
|
return %response;
|
|
|
|
}
|
|
|
|
|
2014-03-23 02:31:50 +00:00
|
|
|
$self->getFields( $som, $path, \%response, { Token => 'Token' } );
|
2009-12-02 16:44:18 +00:00
|
|
|
|
|
|
|
return %response;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub GetBillingAgreementCustomerDetails {
|
|
|
|
my $self = shift;
|
|
|
|
my $token = shift;
|
|
|
|
|
2014-03-23 02:31:50 +00:00
|
|
|
my $request = SOAP::Data->name(
|
|
|
|
GetBillingAgreementCustomerDetailsRequest => \SOAP::Data->value(
|
|
|
|
$self->version_req,
|
|
|
|
SOAP::Data->name( Token => $token )->type( 'xs:string' )
|
|
|
|
->attr( { xmlns => $self->C_xmlns_ebay } ),
|
|
|
|
)
|
|
|
|
)->type( 'ns:GetBillingAgreementCustomerDetailsResponseType' );
|
2009-12-02 16:44:18 +00:00
|
|
|
|
2014-03-23 02:31:50 +00:00
|
|
|
my $som
|
|
|
|
= $self->doCall( GetBillingAgreementCustomerDetailsReq => $request )
|
|
|
|
or return;
|
2009-12-02 16:44:18 +00:00
|
|
|
|
|
|
|
my $path = '/Envelope/Body/GetBillingAgreementCustomerDetailsResponse';
|
|
|
|
|
|
|
|
my %details = ();
|
2014-03-23 02:31:50 +00:00
|
|
|
unless ( $self->getBasic( $som, $path, \%details ) ) {
|
|
|
|
$self->getErrors( $som, $path, \%details );
|
2009-12-02 16:44:18 +00:00
|
|
|
return %details;
|
|
|
|
}
|
|
|
|
|
2014-03-23 02:31:50 +00:00
|
|
|
$self->getFields(
|
|
|
|
$som,
|
|
|
|
"$path/GetBillingAgreementCustomerDetailsResponseDetails",
|
|
|
|
\%details,
|
|
|
|
{ Token => 'Token',
|
|
|
|
|
|
|
|
Payer => 'PayerInfo/Payer',
|
|
|
|
PayerID => 'PayerInfo/PayerID',
|
|
|
|
PayerStatus => 'PayerInfo/PayerStatus', ## 'unverified'
|
|
|
|
PayerBusiness => 'PayerInfo/PayerBusiness',
|
|
|
|
|
|
|
|
Name => 'PayerInfo/Address/Name',
|
|
|
|
AddressOwner => 'PayerInfo/Address/AddressOwner', ## 'PayPal'
|
|
|
|
AddressStatus => 'PayerInfo/Address/AddressStatus', ## 'none'
|
|
|
|
Street1 => 'PayerInfo/Address/Street1',
|
|
|
|
Street2 => 'PayerInfo/Address/Street2',
|
|
|
|
StateOrProvince => 'PayerInfo/Address/StateOrProvince',
|
|
|
|
PostalCode => 'PayerInfo/Address/PostalCode',
|
|
|
|
CountryName => 'PayerInfo/Address/CountryName',
|
|
|
|
|
|
|
|
Salutation => 'PayerInfo/PayerName/Salutation',
|
|
|
|
FirstName => 'PayerInfo/PayerName/FirstName',
|
|
|
|
MiddleName => 'PayerInfo/PayerName/MiddleName',
|
|
|
|
LastName => 'PayerInfo/PayerName/LastName',
|
|
|
|
Suffix => 'PayerInfo/PayerName/Suffix',
|
|
|
|
}
|
|
|
|
);
|
2009-12-02 16:44:18 +00:00
|
|
|
|
|
|
|
return %details;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub CreateRecurringPaymentsProfile {
|
|
|
|
my $self = shift;
|
|
|
|
my %args = @_;
|
|
|
|
|
|
|
|
## RecurringPaymentProfileDetails
|
2014-03-23 02:31:50 +00:00
|
|
|
my %profiledetailstype = (
|
|
|
|
SubscriberName => 'xs:string',
|
|
|
|
SubscriberShipperAddress => 'ns:AddressType',
|
|
|
|
BillingStartDate => 'xs:dateTime', ## MM-DD-YY
|
|
|
|
ProfileReference => 'xs:string',
|
|
|
|
);
|
2009-12-02 16:44:18 +00:00
|
|
|
|
|
|
|
## ScheduleDetailsType
|
2014-03-23 02:31:50 +00:00
|
|
|
my %schedtype = (
|
|
|
|
Description => 'xs:string',
|
|
|
|
ActivationDetails => 'ns:ActivationDetailsType',
|
|
|
|
TrialPeriod => 'ns:BillingPeriodDetailsType',
|
|
|
|
PaymentPeriod => 'ns:BillingPeriodDetailsType',
|
|
|
|
MaxFailedPayments => 'xs:int',
|
|
|
|
AutoBillOutstandingAmount => 'ns:AutoBillType',
|
|
|
|
); ## NoAutoBill or AddToNextBilling
|
2009-12-02 16:44:18 +00:00
|
|
|
|
|
|
|
## activation details
|
2014-03-23 02:31:50 +00:00
|
|
|
my %activationdetailstype = (
|
|
|
|
InitialAmount => 'cc:BasicAmountType',
|
|
|
|
FailedInitialAmountAction => 'ns:FailedPaymentAction',
|
|
|
|
); ## ContinueOnFailure or CancelOnFailure
|
2009-12-02 16:44:18 +00:00
|
|
|
|
|
|
|
## BillingPeriodDetailsType
|
2014-03-23 02:31:50 +00:00
|
|
|
my %trialbilltype = (
|
|
|
|
TrialBillingPeriod => 'xs:string', ##'ns:BillingPeriodType',
|
|
|
|
TrialBillingFrequency => 'xs:int',
|
|
|
|
TrialTotalBillingCycles => 'xs:int',
|
|
|
|
TrialAmount => 'cc:AmountType',
|
|
|
|
TrialShippingAmount => 'cc:AmountType',
|
|
|
|
TrialTaxAmount => 'cc:AmountType',
|
|
|
|
);
|
|
|
|
|
|
|
|
my %paymentbilltype = (
|
|
|
|
PaymentBillingPeriod => 'xs:string', ##'ns:BillingPeriodType',
|
|
|
|
PaymentBillingFrequency => 'xs:int',
|
|
|
|
PaymentTotalBillingCycles => 'xs:int',
|
|
|
|
PaymentAmount => 'cc:AmountType',
|
|
|
|
PaymentShippingAmount => 'cc:AmountType',
|
|
|
|
PaymentTaxAmount => 'cc:AmountType',
|
|
|
|
);
|
2009-12-02 16:44:18 +00:00
|
|
|
|
|
|
|
## AddressType
|
2014-03-23 02:31:50 +00:00
|
|
|
my %payaddrtype = (
|
|
|
|
CCPayerName => 'xs:string',
|
|
|
|
CCPayerStreet1 => 'xs:string',
|
|
|
|
CCPayerStreet2 => 'xs:string',
|
|
|
|
CCPayerCityName => 'xs:string',
|
|
|
|
CCPayerStateOrProvince => 'xs:string',
|
|
|
|
CCPayerCountry => 'xs:string', ## ebl:CountryCodeType
|
|
|
|
CCPayerPostalCode => 'xs:string',
|
|
|
|
CCPayerPhone => 'xs:string',
|
|
|
|
);
|
|
|
|
|
|
|
|
my %shipaddrtype = (
|
|
|
|
SubscriberShipperName => 'xs:string',
|
|
|
|
SubscriberShipperStreet1 => 'xs:string',
|
|
|
|
SubscriberShipperStreet2 => 'xs:string',
|
|
|
|
SubscriberShipperCityName => 'xs:string',
|
|
|
|
SubscriberShipperStateOrProvince => 'xs:string',
|
|
|
|
SubscriberShipperCountry => 'xs:string', ## ebl:CountryCodeType
|
|
|
|
SubscriberShipperPostalCode => 'xs:string',
|
|
|
|
SubscriberShipperPhone => 'xs:string',
|
|
|
|
);
|
2009-12-02 16:44:18 +00:00
|
|
|
|
|
|
|
## credit card payer
|
2014-03-23 02:31:50 +00:00
|
|
|
my %payerinfotype = (
|
|
|
|
CCPayer => 'ns:EmailAddressType',
|
|
|
|
CCPayerID => 'ebl:UserIDType',
|
|
|
|
CCPayerStatus => 'xs:string',
|
|
|
|
CCPayerName => 'xs:string',
|
|
|
|
CCPayerCountry => 'xs:string',
|
|
|
|
CCPayerPhone => 'xs:string',
|
|
|
|
CCPayerBusiness => 'xs:string',
|
|
|
|
CCAddress => 'xs:string',
|
|
|
|
);
|
2009-12-02 16:44:18 +00:00
|
|
|
|
|
|
|
## credit card details
|
2014-03-23 02:31:50 +00:00
|
|
|
my %creditcarddetailstype = (
|
|
|
|
CardOwner => 'ns:PayerInfoType',
|
|
|
|
CreditCardType => 'ebl:CreditCardType'
|
|
|
|
, ## Visa, MasterCard, Discover, Amex, Switch, Solo
|
|
|
|
CreditCardNumber => 'xs:string',
|
|
|
|
ExpMonth => 'xs:int',
|
|
|
|
ExpYear => 'xs:int',
|
|
|
|
CVV2 => 'xs:string',
|
|
|
|
StartMonth => 'xs:string',
|
|
|
|
StartYear => 'xs:string',
|
|
|
|
IssueNumber => 'xs:int',
|
|
|
|
);
|
2009-12-02 16:44:18 +00:00
|
|
|
|
|
|
|
## this gets pushed onto scheduledetails
|
|
|
|
my @activationdetailstype = ();
|
|
|
|
for my $field ( keys %activationdetailstype ) {
|
|
|
|
next unless exists $args{$field};
|
|
|
|
my $real_field = $field;
|
2014-03-23 02:31:50 +00:00
|
|
|
push @activationdetailstype,
|
|
|
|
SOAP::Data->name( $real_field => $args{$field} )
|
|
|
|
->type( $activationdetailstype{$field} );
|
2009-12-02 16:44:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
## this gets pushed onto scheduledetails
|
|
|
|
my @trialbilltype = ();
|
|
|
|
for my $field ( keys %trialbilltype ) {
|
|
|
|
next unless exists $args{$field};
|
2014-03-23 02:31:50 +00:00
|
|
|
( my $real_field = $field ) =~ s/^Trial//;
|
|
|
|
push @trialbilltype,
|
|
|
|
SOAP::Data->name( $real_field => $args{$field} )
|
|
|
|
->type( $trialbilltype{$field} );
|
2009-12-02 16:44:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
## this gets pushed onto scheduledetails
|
|
|
|
my @paymentbilltype = ();
|
|
|
|
for my $field ( keys %paymentbilltype ) {
|
|
|
|
next unless exists $args{$field};
|
2014-03-23 02:31:50 +00:00
|
|
|
( my $real_field = $field ) =~ s/^Payment//;
|
|
|
|
push @paymentbilltype,
|
|
|
|
SOAP::Data->name( $real_field => $args{$field} )
|
|
|
|
->type( $paymentbilltype{$field} );
|
2009-12-02 16:44:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
## this gets pushed onto the top
|
|
|
|
my @sched = ();
|
|
|
|
for my $field ( keys %schedtype ) {
|
|
|
|
next unless exists $args{$field};
|
2014-03-23 02:31:50 +00:00
|
|
|
push @sched,
|
|
|
|
SOAP::Data->name( $field => $args{$field} )
|
|
|
|
->type( $schedtype{$field} );
|
2009-12-02 16:44:18 +00:00
|
|
|
}
|
2014-03-23 02:31:50 +00:00
|
|
|
push @sched,
|
|
|
|
SOAP::Data->name(
|
|
|
|
TrialPeriod => \SOAP::Data->value( @trialbilltype ) )
|
|
|
|
; #->type( 'ns:BillingPeriodDetailsType' );
|
|
|
|
push @sched,
|
|
|
|
SOAP::Data->name(
|
|
|
|
PaymentPeriod => \SOAP::Data->value( @paymentbilltype ) )
|
|
|
|
; #->type( 'ns:BillingPeriodDetailsType' );
|
2009-12-02 16:44:18 +00:00
|
|
|
|
|
|
|
## this gets pushed into profile details
|
|
|
|
my @shipaddr = ();
|
|
|
|
for my $field ( keys %shipaddrtype ) {
|
|
|
|
next unless exists $args{$field};
|
2014-03-23 02:31:50 +00:00
|
|
|
( my $real_field = $field ) =~ s/^SubscriberShipper//;
|
|
|
|
push @shipaddr,
|
|
|
|
SOAP::Data->name( $real_field => $args{$field} )
|
|
|
|
->type( $shipaddrtype{$field} );
|
2009-12-02 16:44:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
## this gets pushed into payerinfo (from creditcarddetails)
|
|
|
|
my @payeraddr = ();
|
|
|
|
for my $field ( keys %payaddrtype ) {
|
|
|
|
next unless $args{$field};
|
2014-03-23 02:31:50 +00:00
|
|
|
( my $real_field = $field ) =~ s/^CCPayer//;
|
|
|
|
push @payeraddr,
|
|
|
|
SOAP::Data->name( $real_field => $args{$field} )
|
|
|
|
->type( payaddrtype {$field} );
|
2009-12-02 16:44:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
## credit card type
|
|
|
|
my @creditcarddetails = ();
|
|
|
|
for my $field ( keys %creditcarddetailstype ) {
|
|
|
|
next unless $args{$field};
|
2014-03-23 02:31:50 +00:00
|
|
|
( my $real_field = $field ) =~ s/^CC//;
|
|
|
|
push @payeraddr,
|
|
|
|
SOAP::Data->name( $real_field => $args{$field} )
|
|
|
|
->type( payaddrtype {$field} );
|
2009-12-02 16:44:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
## this gets pushed onto the top
|
|
|
|
my @profdetail = ();
|
|
|
|
for my $field ( keys %profiledetailstype ) {
|
|
|
|
next unless exists $args{$field};
|
2014-03-23 02:31:50 +00:00
|
|
|
push @profdetail,
|
|
|
|
SOAP::Data->name( $field => $args{$field} )
|
|
|
|
->type( $profiledetailstype{$field} );
|
2009-12-02 16:44:18 +00:00
|
|
|
}
|
2014-03-23 02:31:50 +00:00
|
|
|
push @profdetail,
|
|
|
|
SOAP::Data->name(
|
|
|
|
SubscriberShipperAddress => \SOAP::Data->value( @shipaddr ) );
|
2009-12-02 16:44:18 +00:00
|
|
|
|
|
|
|
## crappard?
|
|
|
|
my @crpprd = ();
|
|
|
|
push @crpprd, SOAP::Data->name( Token => $args{Token} );
|
2014-03-23 02:31:50 +00:00
|
|
|
push @crpprd,
|
|
|
|
SOAP::Data->name(
|
|
|
|
CreditCardDetails => \SOAP::Data->value( @creditcarddetails ) )
|
|
|
|
; #->type( 'ns:CreditCardDetailsType' );
|
|
|
|
push @crpprd,
|
|
|
|
SOAP::Data->name(
|
|
|
|
RecurringPaymentProfileDetails => \SOAP::Data->value( @profdetail ) )
|
|
|
|
; #->type( 'ns:RecurringPaymentProfileDetailsType' );
|
|
|
|
push @crpprd,
|
|
|
|
SOAP::Data->name( ScheduleDetails => \SOAP::Data->value( @sched ) )
|
|
|
|
; #->type( 'ns:ScheduleDetailsType' );
|
|
|
|
|
|
|
|
my $request = SOAP::Data->name(
|
|
|
|
CreateRecurringPaymentsProfileRequest => \SOAP::Data->value
|
|
|
|
|
|
|
|
# ( $API_VERSION,
|
|
|
|
(
|
|
|
|
$self->version_req,
|
|
|
|
SOAP::Data->name(
|
|
|
|
CreateRecurringPaymentsProfileRequestDetails =>
|
|
|
|
\SOAP::Data->value( @crpprd )
|
|
|
|
)->attr( { xmlns => $self->C_xmlns_ebay } )
|
|
|
|
)
|
|
|
|
); #->type( 'ns:CreateRecurringPaymentsProfileRequestType' );
|
2009-12-02 16:44:18 +00:00
|
|
|
|
|
|
|
my $som = $self->doCall( CreateRecurringPaymentsProfileReq => $request )
|
2014-03-23 02:31:50 +00:00
|
|
|
or return;
|
2009-12-02 16:44:18 +00:00
|
|
|
|
|
|
|
my $path = '/Envelope/Body/CreateRecurringPaymentsProfileResponse';
|
|
|
|
|
|
|
|
my %response = ();
|
2014-03-23 02:31:50 +00:00
|
|
|
unless ( $self->getBasic( $som, $path, \%response ) ) {
|
|
|
|
$self->getErrors( $som, $path, \%response );
|
2009-12-02 16:44:18 +00:00
|
|
|
return %response;
|
|
|
|
}
|
|
|
|
|
2014-03-23 02:31:50 +00:00
|
|
|
$self->getFields( $som, $path, \%response, { Token => 'Token' } );
|
2009-12-02 16:44:18 +00:00
|
|
|
|
|
|
|
return %response;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub DoReferenceTransaction {
|
|
|
|
my $self = shift;
|
|
|
|
my %args = @_;
|
|
|
|
|
2014-03-23 02:31:50 +00:00
|
|
|
my %types = (
|
|
|
|
ReferenceID => 'xs:string',
|
|
|
|
PaymentAction => '', ## NOTA BENE!
|
|
|
|
currencyID => '',
|
|
|
|
);
|
2009-12-02 16:44:18 +00:00
|
|
|
|
|
|
|
## PaymentDetails
|
2014-03-23 02:31:50 +00:00
|
|
|
my %pd_types = (
|
|
|
|
OrderTotal => 'ebl:BasicAmountType',
|
|
|
|
OrderDescription => 'xs:string',
|
|
|
|
ItemTotal => 'ebl:BasicAmountType',
|
|
|
|
ShippingTotal => 'ebl:BasicAmountType',
|
|
|
|
HandlingTotal => 'ebl:BasicAmountType',
|
|
|
|
TaxTotal => 'ebl:BasicAmountType',
|
|
|
|
Custom => 'xs:string',
|
|
|
|
InvoiceID => 'xs:string',
|
|
|
|
ButtonSource => 'xs:string',
|
|
|
|
NotifyURL => 'xs:string',
|
|
|
|
);
|
2009-12-02 16:44:18 +00:00
|
|
|
|
|
|
|
## ShipToAddress
|
2014-03-23 02:31:50 +00:00
|
|
|
my %st_types = (
|
|
|
|
ST_Name => 'xs:string',
|
|
|
|
ST_Street1 => 'xs:string',
|
|
|
|
ST_Street2 => 'xs:string',
|
|
|
|
ST_CityName => 'xs:string',
|
|
|
|
ST_StateOrProvince => 'xs:string',
|
|
|
|
ST_Country => 'xs:string',
|
|
|
|
ST_PostalCode => 'xs:string',
|
|
|
|
ST_Phone => 'xs:string',
|
|
|
|
);
|
2009-12-02 16:44:18 +00:00
|
|
|
|
|
|
|
##PaymentDetailsItem
|
2014-03-23 02:31:50 +00:00
|
|
|
my %pdi_types = (
|
|
|
|
PDI_Name => 'xs:string',
|
|
|
|
PDI_Description => 'xs:string',
|
|
|
|
PDI_Amount => 'ebl:BasicAmountType',
|
|
|
|
PDI_Number => 'xs:string',
|
|
|
|
PDI_Quantity => 'xs:string',
|
|
|
|
PDI_Tax => 'ebl:BasicAmountType',
|
|
|
|
);
|
2009-12-02 16:44:18 +00:00
|
|
|
|
|
|
|
$args{PaymentAction} ||= 'Sale';
|
|
|
|
$args{currencyID} ||= 'USD';
|
|
|
|
|
2014-03-23 02:31:50 +00:00
|
|
|
my @payment_details = ();
|
2009-12-02 16:44:18 +00:00
|
|
|
|
|
|
|
## push OrderTotal here and delete it (i.e., and all others that have special attrs)
|
2014-03-23 02:31:50 +00:00
|
|
|
push @payment_details,
|
|
|
|
SOAP::Data->name( OrderTotal => $args{OrderTotal} )
|
|
|
|
->type( $pd_types{OrderTotal} )->attr(
|
|
|
|
{ currencyID => $args{currencyID},
|
|
|
|
xmlns => $self->C_xmlns_ebay
|
|
|
|
}
|
|
|
|
);
|
2009-12-02 16:44:18 +00:00
|
|
|
|
|
|
|
## don't process it again
|
|
|
|
delete $pd_types{OrderTotal};
|
|
|
|
|
|
|
|
for my $field ( keys %pd_types ) {
|
2014-03-23 02:31:50 +00:00
|
|
|
if ( $args{$field} ) {
|
|
|
|
push @payment_details,
|
|
|
|
SOAP::Data->name( $field => $args{$field} )
|
|
|
|
->type( $pd_types{$field} );
|
|
|
|
}
|
2009-12-02 16:44:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
##
|
|
|
|
## ShipToAddress
|
|
|
|
##
|
|
|
|
my @ship_types = ();
|
|
|
|
for my $field ( keys %st_types ) {
|
2014-03-23 02:31:50 +00:00
|
|
|
if ( $args{$field} ) {
|
|
|
|
( my $name = $field ) =~ s/^ST_//;
|
|
|
|
push @ship_types,
|
|
|
|
SOAP::Data->name( $name => $args{$field} )
|
|
|
|
->type( $st_types{$field} );
|
|
|
|
}
|
2009-12-02 16:44:18 +00:00
|
|
|
}
|
|
|
|
|
2014-03-23 02:31:50 +00:00
|
|
|
if ( scalar @ship_types ) {
|
|
|
|
push @payment_details,
|
|
|
|
SOAP::Data->name( ShipToAddress =>
|
|
|
|
\SOAP::Data->value( @ship_types )->type( 'ebl:AddressType' )
|
|
|
|
->attr( { xmlns => $self->C_xmlns_ebay } ), );
|
2009-12-02 16:44:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
##
|
|
|
|
## PaymentDetailsItem
|
|
|
|
##
|
|
|
|
my @payment_details_item = ();
|
|
|
|
for my $field ( keys %pdi_types ) {
|
2014-03-23 02:31:50 +00:00
|
|
|
if ( $args{$field} ) {
|
|
|
|
( my $name = $field ) =~ s/^PDI_//;
|
|
|
|
push @payment_details_item,
|
|
|
|
SOAP::Data->name( $name => $args{$field} )
|
|
|
|
->type( $pdi_types{$field} );
|
|
|
|
}
|
2009-12-02 16:44:18 +00:00
|
|
|
}
|
|
|
|
|
2014-03-23 02:31:50 +00:00
|
|
|
if ( scalar @payment_details_item ) {
|
|
|
|
push @payment_details,
|
|
|
|
SOAP::Data->name(
|
|
|
|
PaymentDetailsItem => \SOAP::Data->value( @payment_details_item )
|
|
|
|
->type( 'ebl:PaymentDetailsItemType' )
|
|
|
|
->attr( { xmlns => $self->C_xmlns_ebay } ), );
|
2009-12-02 16:44:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
##
|
|
|
|
## ReferenceTransactionPaymentDetails
|
|
|
|
##
|
|
|
|
my @reference_details = (
|
2014-03-23 02:31:50 +00:00
|
|
|
SOAP::Data->name( ReferenceID => $args{ReferenceID} )
|
|
|
|
->type( $types{ReferenceID} )
|
|
|
|
->attr( { xmlns => $self->C_xmlns_ebay } ),
|
|
|
|
SOAP::Data->name( PaymentAction => $args{PaymentAction} )
|
|
|
|
->type( $types{PaymentAction} )
|
|
|
|
->attr( { xmlns => $self->C_xmlns_ebay } ),
|
|
|
|
SOAP::Data->name(
|
|
|
|
PaymentDetails => \SOAP::Data->value( @payment_details )
|
|
|
|
->type( 'ebl:PaymentDetailsType' )
|
|
|
|
->attr( { xmlns => $self->C_xmlns_ebay } ),
|
|
|
|
),
|
|
|
|
);
|
2009-12-02 16:44:18 +00:00
|
|
|
|
|
|
|
##
|
|
|
|
## the main request object
|
|
|
|
##
|
2014-03-23 02:31:50 +00:00
|
|
|
my $request = SOAP::Data->name(
|
|
|
|
DoReferenceTransactionRequest => \SOAP::Data->value(
|
|
|
|
$self->version_req,
|
|
|
|
SOAP::Data->name(
|
|
|
|
DoReferenceTransactionRequestDetails =>
|
|
|
|
\SOAP::Data->value( @reference_details )
|
|
|
|
->type( 'ns:DoReferenceTransactionRequestDetailsType' )
|
|
|
|
)->attr( { xmlns => $self->C_xmlns_ebay } ),
|
|
|
|
)
|
|
|
|
);
|
2009-12-02 16:44:18 +00:00
|
|
|
|
|
|
|
my $som = $self->doCall( DoReferenceTransactionReq => $request )
|
2014-03-23 02:31:50 +00:00
|
|
|
or return;
|
2009-12-02 16:44:18 +00:00
|
|
|
|
|
|
|
my $path = '/Envelope/Body/DoReferenceTransactionResponse';
|
|
|
|
|
|
|
|
my %response = ();
|
2014-03-23 02:31:50 +00:00
|
|
|
unless ( $self->getBasic( $som, $path, \%response ) ) {
|
|
|
|
$self->getErrors( $som, $path, \%response );
|
2009-12-02 16:44:18 +00:00
|
|
|
return %response;
|
|
|
|
}
|
|
|
|
|
2014-03-23 02:31:50 +00:00
|
|
|
$self->getFields(
|
|
|
|
$som,
|
|
|
|
"$path/DoReferenceTransactionResponseDetails",
|
|
|
|
\%response,
|
|
|
|
{ BillingAgreementID => 'BillingAgreementID',
|
|
|
|
TransactionID => 'PaymentInfo/TransactionID',
|
|
|
|
TransactionType => 'PaymentInfo/TransactionType',
|
|
|
|
PaymentType => 'PaymentInfo/PaymentType',
|
|
|
|
PaymentDate => 'PaymentInfo/PaymentDate',
|
|
|
|
GrossAmount => 'PaymentInfo/GrossAmount',
|
|
|
|
FeeAmount => 'PaymentInfo/FeeAmount',
|
|
|
|
SettleAmount => 'PaymentInfo/SettleAmount',
|
|
|
|
TaxAmount => 'PaymentInfo/TaxAmount',
|
|
|
|
ExchangeRate => 'PaymentInfo/ExchangeRate',
|
|
|
|
PaymentStatus => 'PaymentInfo/PaymentStatus',
|
|
|
|
PendingReason => 'PaymentInfo/PendingReason',
|
|
|
|
ReasonCode => 'PaymentInfor/ReasonCode',
|
|
|
|
}
|
|
|
|
);
|
2009-12-02 16:44:18 +00:00
|
|
|
|
|
|
|
return %response;
|
|
|
|
}
|
|
|
|
|
|
|
|
1;
|
|
|
|
__END__
|
|
|
|
|
|
|
|
=head1 NAME
|
|
|
|
|
|
|
|
Business::PayPal::API::RecurringPayments - PayPal RecurringPayments API
|
|
|
|
|
|
|
|
=head1 SYNOPSIS
|
|
|
|
|
|
|
|
use Business::PayPal::API::RecurringPayments;
|
|
|
|
|
|
|
|
my $pp = new Business::PayPal::API::RecurringPayments( ... );
|
|
|
|
|
|
|
|
my %resp = $pp->FIXME
|
|
|
|
|
|
|
|
## Ask PayPal to charge a new transaction from the ReferenceID
|
|
|
|
## This method is used both for Recurring Transactions as well
|
|
|
|
## as for Express Checkout's MerchantInitiatedBilling, where
|
|
|
|
## ReferenceID is the BillingAgreementID returned from
|
|
|
|
## ExpressCheckout->DoExpressCheckoutPayment
|
|
|
|
|
|
|
|
my %payinfo = $pp->DoReferenceTransaction( ReferenceID => $details{ReferenceID},
|
|
|
|
PaymentAction => 'Sale',
|
|
|
|
OrderTotal => '55.43' );
|
|
|
|
|
|
|
|
|
|
|
|
=head1 DESCRIPTION
|
|
|
|
|
|
|
|
THIS MODULE IS NOT COMPLETE YET. PLEASE DO NOT REPORT ANY BUGS RELATED
|
|
|
|
TO IT.
|
|
|
|
|
|
|
|
=head2 DoReferenceTransaction
|
|
|
|
|
|
|
|
Implements PayPal's WPP B<DoReferenceTransaction> API call. Supported
|
|
|
|
parameters include:
|
|
|
|
|
|
|
|
ReferenceID (aka BillingAgreementID)
|
|
|
|
PaymentAction (defaults to 'Sale' if not supplied)
|
|
|
|
currencyID (defaults to 'USD' if not supplied)
|
|
|
|
|
|
|
|
OrderTotal
|
|
|
|
OrderDescription
|
|
|
|
ItemTotal
|
|
|
|
ShippingTotal
|
|
|
|
HandlingTotal
|
|
|
|
TaxTotal
|
|
|
|
Custom
|
|
|
|
InvoiceID
|
|
|
|
ButtonSource
|
|
|
|
NotifyURL
|
|
|
|
|
|
|
|
ST_Name
|
|
|
|
ST_Street1
|
|
|
|
ST_Street2
|
|
|
|
ST_CityName
|
|
|
|
ST_StateOrProvince
|
|
|
|
ST_Country
|
|
|
|
ST_PostalCode
|
|
|
|
ST_Phone
|
|
|
|
|
|
|
|
PDI_Name
|
|
|
|
PDI_Description
|
|
|
|
PDI_Amount
|
|
|
|
PDI_Number
|
|
|
|
PDI_Quantity
|
|
|
|
PDI_Tax
|
|
|
|
|
|
|
|
as described in the PayPal "Web Services API Reference" document.
|
|
|
|
|
|
|
|
Returns a hash with the following keys:
|
|
|
|
|
|
|
|
BillingAgreementID
|
|
|
|
TransactionID
|
|
|
|
TransactionType
|
|
|
|
PaymentType
|
|
|
|
PaymentDate
|
|
|
|
GrossAmount
|
|
|
|
FeeAmount
|
|
|
|
SettleAmount
|
|
|
|
TaxAmount
|
|
|
|
ExchangeRate
|
|
|
|
PaymentStatus
|
|
|
|
PendingReason
|
|
|
|
ReasonCode
|
|
|
|
|
|
|
|
Required fields:
|
|
|
|
|
|
|
|
ReferenceID, OrderTotal
|
|
|
|
|
|
|
|
=head1 SEE ALSO
|
|
|
|
|
|
|
|
L<https://developer.paypal.com/en_US/pdf/PP_APIReference.pdf>
|
|
|
|
|
|
|
|
=head1 AUTHOR
|
|
|
|
|
|
|
|
Scot Wiersdorf E<lt>scott@perlcode.orgE<gt>
|
|
|
|
|
|
|
|
=head1 COPYRIGHT AND LICENSE
|
|
|
|
|
|
|
|
Copyright (C) 2007 by Scott Wiersdorf
|
|
|
|
|
|
|
|
This library is free software; you can redistribute it and/or modify
|
|
|
|
it under the same terms as Perl itself, either Perl version 5.8.5 or,
|
|
|
|
at your option, any later version of Perl 5 you may have available.
|
|
|
|
|
|
|
|
=cut
|