perl-business-paypal-api/lib/Business/PayPal/API/RecurringPayments.pm

661 lines
21 KiB
Perl
Raw Normal View History

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