Net-Stripe
view release on metacpan or search on metacpan
lib/Net/Stripe.pm view on Meta::CPAN
package Net::Stripe;
$Net::Stripe::VERSION = '0.42';
use Moose;
use Class::Load;
use Type::Tiny 1.008004;
use Kavorka;
use LWP::UserAgent;
use HTTP::Request::Common qw/GET POST DELETE/;
use MIME::Base64 qw/encode_base64/;
use URI::Escape qw/uri_escape/;
use JSON qw/decode_json/;
use URI qw//;
use DateTime qw//;
use Net::Stripe::TypeConstraints;
use Net::Stripe::Constants;
use Net::Stripe::Token;
use Net::Stripe::Invoiceitem;
use Net::Stripe::Invoice;
use Net::Stripe::Card;
use Net::Stripe::Source;
use Net::Stripe::Plan;
use Net::Stripe::Product;
use Net::Stripe::Coupon;
use Net::Stripe::Charge;
use Net::Stripe::Customer;
use Net::Stripe::Discount;
use Net::Stripe::Subscription;
use Net::Stripe::Error;
use Net::Stripe::BalanceTransaction;
use Net::Stripe::List;
use Net::Stripe::LineItem;
use Net::Stripe::Refund;
use Net::Stripe::PaymentMethod;
use Net::Stripe::PaymentIntent;
# ABSTRACT: API client for Stripe.com
has 'debug' => (is => 'rw', isa => 'Bool', default => 0, documentation => "The debug flag");
has 'debug_network' => (is => 'rw', isa => 'Bool', default => 0, documentation => "The debug network request flag");
has 'api_key' => (is => 'ro', isa => 'Str', required => 1, documentation => "You get this from your Stripe Account settings");
has 'api_base' => (is => 'ro', isa => 'Str', lazy_build => 1, documentation => "This is the base part of the URL for every request made");
has 'ua' => (is => 'ro', isa => 'Object', lazy_build => 1, documentation => "The LWP::UserAgent that is used for requests");
has 'api_version' => (is => 'ro', isa => 'StripeAPIVersion', documentation => "This is the value of the Stripe-Version header you wish to transmit for API calls");
has 'force_api_version' => (is => 'ro', isa => 'Bool', default => 0, documentation => "Set this to true to bypass the safety checks for API version compatibility.");
sub BUILD {
my ( $self, $args ) = @_;
$self->_validate_api_version_range();
$self->_validate_api_version_value();
}
Charges: {
method post_charge(Int :$amount,
Str :$currency,
StripeCustomerId :$customer?,
StripeTokenId|StripeCardId :$card?,
StripeTokenId|StripeCardId|StripeSourceId :$source?,
Str :$description?,
HashRef :$metadata?,
Bool :$capture?,
Str :$statement_descriptor?,
Int :$application_fee?,
Str :$receipt_email?
) {
if ( defined( $card ) ) {
my $card_id_type = Moose::Util::TypeConstraints::find_type_constraint( 'StripeCardId' );
if ( defined( $customer ) && ! $card_id_type->check( $card ) ) {
die Net::Stripe::Error->new(
type => "post_charge error",
message => sprintf(
"Invalid value '%s' passed for parameter 'card'. Charges for an existing customer can only accept a card id.",
$card,
),
);
}
my $token_id_type = Moose::Util::TypeConstraints::find_type_constraint( 'StripeTokenId' );
if ( ! defined( $customer ) && ! $token_id_type->check( $card ) ) {
die Net::Stripe::Error->new(
type => "post_charge error",
message => sprintf(
"Invalid value '%s' passed for parameter 'card'. Charges without an existing customer can only accept a token id.",
$card,
),
);
}
}
if ( defined( $source ) ) {
my $card_id_type = Moose::Util::TypeConstraints::find_type_constraint( 'StripeCardId' );
if ( defined( $customer ) && ! $card_id_type->check( $source ) ) {
die Net::Stripe::Error->new(
type => "post_charge error",
message => sprintf(
"Invalid value '%s' passed for parameter 'source'. Charges for an existing customer can only accept a card id.",
$source,
),
);
}
my $token_id_type = Moose::Util::TypeConstraints::find_type_constraint( 'StripeTokenId' );
my $source_id_type = Moose::Util::TypeConstraints::find_type_constraint( 'StripeSourceId' );
if ( ! defined( $customer ) && ! $token_id_type->check( $source ) && ! $source_id_type->check( $source ) ) {
die Net::Stripe::Error->new(
type => "post_charge error",
message => sprintf(
"Invalid value '%s' passed for parameter 'source'. Charges without an existing customer can only accept a token id or source id.",
$source,
),
);
}
}
my $charge = Net::Stripe::Charge->new(amount => $amount,
currency => $currency,
customer => $customer,
card => $card,
source => $source,
description => $description,
metadata => $metadata,
capture => $capture,
statement_descriptor => $statement_descriptor,
application_fee => $application_fee,
receipt_email => $receipt_email
);
return $self->_post('charges', $charge);
}
method get_charge(Str :$charge_id) {
return $self->_get("charges/" . $charge_id);
}
method refund_charge(Net::Stripe::Charge|Str :$charge, Int :$amount?) {
if (ref($charge)) {
$charge = $charge->id;
}
my $refund = Net::Stripe::Refund->new(id => $charge,
amount => $amount
);
return $self->_post("charges/$charge/refunds", $refund);
}
method get_charges(HashRef :$created?,
Net::Stripe::Customer|Str :$customer?,
Str :$ending_before?,
Int :$limit?,
Str :$starting_after?) {
if (ref($customer)) {
$customer = $customer->id;
}
my %args = (
path => 'charges',
created => $created,
customer => $customer,
ending_before => $ending_before,
limit => $limit,
starting_after => $starting_after,
);
$self->_get_all(%args);
}
method capture_charge(
Net::Stripe::Charge|Str :$charge!,
Int :$amount?,
) {
if (ref($charge)) {
$charge = $charge->id;
}
my %args = (
amount => $amount,
);
return $self->_post("charges/$charge/capture", \%args);
}
}
PaymentIntents: {
method create_payment_intent(
Int :$amount!,
Str :$currency!,
Int :$application_fee_amount?,
StripeCaptureMethod :$capture_method?,
Bool :$confirm?,
StripeConfirmationMethod :$confirmation_method?,
StripeCustomerId :$customer?,
Str :$description?,
Bool :$error_on_requires_action?,
Str :$mandate?,
HashRef :$mandate_data?,
HashRef[Str] :$metadata?,
Bool :$off_session?,
Str :$on_behalf_of?,
StripePaymentMethodId :$payment_method?,
HashRef :$payment_method_options?,
ArrayRef[StripePaymentMethodType] :$payment_method_types?,
Str :$receipt_email?,
Str :$return_url?,
Bool :$save_payment_method?,
Str :$setup_future_usage?,
HashRef :$shipping?,
Str :$statement_descriptor?,
Str :$statement_descriptor_suffix?,
HashRef :$transfer_data?,
Str :$transfer_group?,
Bool :$use_stripe_sdk?,
) {
my %args = (
amount => $amount,
currency => $currency,
application_fee_amount => $application_fee_amount,
capture_method => $capture_method,
confirm => $confirm,
confirmation_method => $confirmation_method,
customer => $customer,
description => $description,
error_on_requires_action => $error_on_requires_action,
mandate => $mandate,
mandate_data => $mandate_data,
metadata => $metadata,
off_session => $off_session,
on_behalf_of => $on_behalf_of,
payment_method => $payment_method,
payment_method_options => $payment_method_options,
payment_method_types => $payment_method_types,
receipt_email => $receipt_email,
return_url => $return_url,
save_payment_method => $save_payment_method,
setup_future_usage => $setup_future_usage,
shipping => $shipping,
statement_descriptor => $statement_descriptor,
statement_descriptor_suffix => $statement_descriptor_suffix,
transfer_data => $transfer_data,
transfer_group => $transfer_group,
use_stripe_sdk => $use_stripe_sdk,
);
my $payment_intent_obj = Net::Stripe::PaymentIntent->new( %args );
return $self->_post("payment_intents", $payment_intent_obj);
}
method get_payment_intent(
StripePaymentIntentId :$payment_intent_id!,
Str :$client_secret?,
) {
my %args = (
client_secret => $client_secret,
);
return $self->_get("payment_intents/$payment_intent_id", \%args);
}
method update_payment_intent(
StripePaymentIntentId :$payment_intent_id!,
Int :$amount?,
Int :$application_fee_amount?,
Str :$currency?,
StripeCustomerId :$customer?,
Str :$description?,
HashRef[Str]|EmptyStr :$metadata?,
StripePaymentMethodId :$payment_method?,
HashRef :$payment_method_options?,
ArrayRef[StripePaymentMethodType] :$payment_method_types?,
Str :$receipt_email?,
Bool :$save_payment_method?,
Str :$setup_future_usage?,
HashRef :$shipping?,
Str :$statement_descriptor?,
Str :$statement_descriptor_suffix?,
HashRef :$transfer_data?,
Str :$transfer_group?,
) {
my %args = (
amount => $amount,
application_fee_amount => $application_fee_amount,
currency => $currency,
customer => $customer,
description => $description,
metadata => $metadata,
payment_method => $payment_method,
payment_method_options => $payment_method_options,
payment_method_types => $payment_method_types,
receipt_email => $receipt_email,
save_payment_method => $save_payment_method,
setup_future_usage => $setup_future_usage,
shipping => $shipping,
statement_descriptor => $statement_descriptor,
statement_descriptor_suffix => $statement_descriptor_suffix,
transfer_data => $transfer_data,
transfer_group => $transfer_group,
);
my $payment_intent_obj = Net::Stripe::PaymentIntent->new( %args );
return $self->_post("payment_intents/$payment_intent_id", $payment_intent_obj);
}
method confirm_payment_intent(
StripePaymentIntentId :$payment_intent_id!,
Str :$client_secret?,
Bool :$error_on_requires_action?,
Str :$mandate?,
HashRef :$mandate_data?,
Bool :$off_session?,
StripePaymentMethodId :$payment_method?,
HashRef :$payment_method_options?,
ArrayRef[StripePaymentMethodType] :$payment_method_types?,
Str :$receipt_email?,
Str :$return_url?,
Bool :$save_payment_method?,
Str :$setup_future_usage?,
HashRef :$shipping?,
Bool :$use_stripe_sdk?,
) {
my %args = (
client_secret => $client_secret,
error_on_requires_action => $error_on_requires_action,
mandate => $mandate,
mandate_data => $mandate_data,
off_session => $off_session,
payment_method => $payment_method,
payment_method_options => $payment_method_options,
payment_method_types => $payment_method_types,
receipt_email => $receipt_email,
return_url => $return_url,
save_payment_method => $save_payment_method,
setup_future_usage => $setup_future_usage,
shipping => $shipping,
use_stripe_sdk => $use_stripe_sdk,
);
return $self->_post("payment_intents/$payment_intent_id/confirm", \%args);
}
method capture_payment_intent(
StripePaymentIntentId :$payment_intent_id!,
Int :$amount_to_capture?,
Int :$application_fee_amount?,
Str :$statement_descriptor?,
Str :$statement_descriptor_suffix?,
HashRef :$transfer_data?,
) {
my %args = (
amount_to_capture => $amount_to_capture,
application_fee_amount => $application_fee_amount,
statement_descriptor => $statement_descriptor,
statement_descriptor_suffix => $statement_descriptor_suffix,
transfer_data => $transfer_data,
);
return $self->_post("payment_intents/$payment_intent_id/capture", \%args);
}
method cancel_payment_intent(
StripePaymentIntentId :$payment_intent_id!,
StripeCancellationReason :$cancellation_reason?,
) {
my %args = (
cancellation_reason => $cancellation_reason,
);
return $self->_post("payment_intents/$payment_intent_id/cancel", \%args);
}
method list_payment_intents(
StripeCustomerId :$customer?,
HashRef[Int] :$created?,
Str :$ending_before?,
Int :$limit?,
Str :$starting_after?,
) {
my %args = (
customer => $customer,
created => $created,
ending_before => $ending_before,
limit => $limit,
starting_after => $starting_after,
);
return $self->_get("payment_intents", \%args);
}
}
BalanceTransactions: {
method get_balance_transaction(Str :$id) {
return $self->_get("balance/history/$id");
}
}
Customers: {
method post_customer(Net::Stripe::Customer|StripeCustomerId :$customer?,
Int :$account_balance?,
Int :$balance?,
Net::Stripe::Token|StripeTokenId :$card?,
Str :$coupon?,
Str :$default_card?,
StripeCardId|StripeSourceId :$default_source?,
Str :$description?,
Str :$email?,
HashRef :$metadata?,
Str :$plan?,
Int :$quantity?,
StripeTokenId|StripeSourceId :$source?,
Int|Str :$trial_end?) {
my $customer_obj;
if ( ref( $customer ) ) {
$customer_obj = $customer;
} else {
my %args = (
account_balance => $account_balance,
balance => $balance,
card => $card,
coupon => $coupon,
default_card => $default_card,
default_source => $default_source,
email => $email,
metadata => $metadata,
plan => $plan,
quantity => $quantity,
source => $source,
trial_end => $trial_end,
);
$args{id} = $customer if defined( $customer );
$customer_obj = Net::Stripe::Customer->new( %args );
}
if ( my $customer_id = $customer_obj->id ) {
return $self->_post("customers/$customer_id", $customer_obj);
} else {
return $self->_post('customers', $customer_obj);
}
}
method list_subscriptions(Net::Stripe::Customer|Str :$customer,
Str :$ending_before?,
Int :$limit?,
Str :$starting_after?) {
if (ref($customer)) {
$customer = $customer->id;
}
my %args = (
path => "customers/$customer/subscriptions",
ending_before => $ending_before,
limit => $limit,
starting_after => $starting_after,
);
return $self->_get_all(%args);
}
method get_customer(Str :$customer_id) {
return $self->_get("customers/$customer_id");
}
method delete_customer(Net::Stripe::Customer|Str :$customer) {
if (ref($customer)) {
$customer = $customer->id;
}
$self->_delete("customers/$customer");
}
method get_customers(HashRef :$created?, Str :$ending_before?, Int :$limit?, Str :$starting_after?, Str :$email?) {
my %args = (
path => 'customers',
created => $created,
ending_before => $ending_before,
limit => $limit,
starting_after => $starting_after,
email => $email,
);
$self->_get_all(%args);
}
}
Cards: {
method get_card(Net::Stripe::Customer|Str :$customer,
Str :$card_id) {
if (ref($customer)) {
$customer = $customer->id;
}
return $self->_get("customers/$customer/sources/$card_id");
}
method get_cards(Net::Stripe::Customer|Str :$customer,
HashRef :$created?,
Str :$ending_before?,
Int :$limit?,
Str :$starting_after?) {
if (ref($customer)) {
$customer = $customer->id;
}
my %args = (
path => "customers/$customer/sources",
object => "card",
created => $created,
ending_before => $ending_before,
limit => $limit,
starting_after => $starting_after,
);
$self->_get_all(%args);
}
method post_card(
Net::Stripe::Customer|StripeCustomerId :$customer!,
Net::Stripe::Token|StripeTokenId :$card?,
StripeTokenId :$source?,
) {
die Net::Stripe::Error->new(
type => "post_card error",
message => "One of parameters 'source' or 'card' is required.",
) unless defined( $card ) || defined( $source );
my $customer_id = ref( $customer ) ? $customer->id : $customer;
if ( defined( $card ) ) {
# card here is either Net::Stripe::Token or StripeTokenId
my $token_id = ref( $card ) ? $card->id : $card;
return $self->_post("customers/$customer_id/cards", {card=> $token_id});
}
if ( defined( $source ) ) {
return $self->_post("customers/$customer_id/cards", { source=> $source });
}
}
method update_card(StripeCustomerId :$customer_id!,
StripeCardId :$card_id!,
HashRef :$card!) {
return $self->_post("customers/$customer_id/cards/$card_id", $card);
}
method delete_card(Net::Stripe::Customer|Str :$customer, Net::Stripe::Card|Str :$card) {
if (ref($customer)) {
$customer = $customer->id;
}
if (ref($card)) {
$card = $card->id;
}
return $self->_delete("customers/$customer/sources/$card");
}
}
Sources: {
method create_source(
StripeSourceType :$type!,
Int :$amount?,
Str :$currency?,
StripeSourceFlow :$flow?,
HashRef :$mandate?,
HashRef :$metadata?,
HashRef :$owner?,
HashRef :$receiver?,
HashRef :$redirect?,
HashRef :$source_order?,
Str :$statement_descriptor?,
StripeTokenId :$token?,
StripeSourceUsage :$usage?,
) {
die Net::Stripe::Error->new(
type => "create_source error",
message => "Parameter 'token' is required for source type 'card'",
param => 'token',
) if defined( $type ) && $type eq 'card' && ! defined( $token );
my %args = (
amount => $amount,
currency => $currency,
flow => $flow,
mandate => $mandate,
metadata => $metadata,
owner => $owner,
receiver => $receiver,
redirect => $redirect,
source_order => $source_order,
statement_descriptor => $statement_descriptor,
token => $token,
type => $type,
usage => $usage,
);
my $source_obj = Net::Stripe::Source->new( %args );
return $self->_post("sources", $source_obj);
}
method get_source(
StripeSourceId :$source_id!,
Str :$client_secret?,
) {
my %args = (
client_secret => $client_secret,
);
return $self->_get("sources/$source_id", \%args);
}
method update_source(
StripeSourceId :$source_id!,
Int :$amount?,
HashRef :$mandate?,
HashRef|EmptyStr :$metadata?,
HashRef :$owner?,
HashRef :$source_order?,
) {
my %args = (
amount => $amount,
mandate => $mandate,
metadata => $metadata,
owner => $owner,
source_order => $source_order,
);
my $source_obj = Net::Stripe::Source->new( %args );
my @one_of = qw/ amount mandate metadata owner source_order /;
my @defined = grep { defined( $source_obj->$_ ) } @one_of;
die Net::Stripe::Error->new(
type => "update_source error",
message => sprintf( "at least one of: %s is required to update a source",
join( ', ', @one_of ),
),
) if ! @defined;
return $self->_post("sources/$source_id", $source_obj);
}
method attach_source (
StripeCustomerId :$customer_id!,
StripeSourceId :$source_id!,
) {
my %args = (
source => $source_id,
);
return $self->_post("customers/$customer_id/sources", \%args);
}
method detach_source(
StripeCustomerId :$customer_id!,
StripeSourceId :$source_id!,
) {
return $self->_delete("customers/$customer_id/sources/$source_id");
}
# undocumented API endpoint
method list_sources(
StripeCustomerId :$customer_id!,
Str :$object!,
Str :$ending_before?,
Int :$limit?,
Str :$starting_after?,
) {
my %args = (
ending_before => $ending_before,
limit => $limit,
object => $object,
starting_after => $starting_after,
);
return $self->_get("customers/$customer_id/sources", \%args);
}
}
Subscriptions: {
method get_subscription(Net::Stripe::Customer|Str :$customer) {
if (ref($customer)) {
$customer = $customer->id;
}
return $self->_get("customers/$customer/subscription");
}
# adds a subscription, keeping any existing subscriptions unmodified
method post_subscription(Net::Stripe::Customer|Str :$customer,
Net::Stripe::Subscription|Str :$subscription?,
Net::Stripe::Plan|Str :$plan?,
Str :$coupon?,
Int|Str :$trial_end?,
Net::Stripe::Card|Net::Stripe::Token|Str :$card?,
Net::Stripe::Card|Net::Stripe::Token|Str :$source?,
Int :$quantity? where { $_ >= 0 },
Num :$application_fee_percent?,
Bool :$prorate? = 1,
Bool :$cancel_at_period_end?,
) {
if (ref($customer)) {
$customer = $customer->id;
}
if (ref($plan)) {
$plan = $plan->id;
}
if (ref($subscription) ne 'Net::Stripe::Subscription') {
my %args = (plan => $plan,
coupon => $coupon,
trial_end => $trial_end,
card => $card,
source => $source,
prorate => $prorate,
quantity => $quantity,
application_fee_percent => $application_fee_percent,
cancel_at_period_end => $cancel_at_period_end);
if (defined($subscription)) {
$args{id} = $subscription;
}
$subscription = Net::Stripe::Subscription->new( %args );
}
if (defined($subscription->id)) {
return $self->_post("customers/$customer/subscriptions/" . $subscription->id, $subscription);
} else {
return $self->_post("customers/$customer/subscriptions", $subscription);
}
}
method delete_subscription(Net::Stripe::Customer|Str :$customer,
Net::Stripe::Subscription|Str :$subscription,
Bool :$at_period_end?
) {
if (ref($customer)) {
$customer = $customer->id;
}
if (ref($subscription)) {
$subscription = $subscription->id;
}
my %args;
$args{at_period_end} = 'true' if $at_period_end;
return $self->_delete("customers/$customer/subscriptions/$subscription", \%args);
}
}
PaymentMethods: {
method create_payment_method(
StripePaymentMethodType :$type!,
HashRef :$billing_details?,
StripeTokenId :$card?,
HashRef :$fpx?,
HashRef :$ideal?,
HashRef[Str] :$metadata?,
StripePaymentMethodId :$payment_method?,
HashRef :$sepa_debit?,
) {
my %args = (
type => $type,
billing_details => $billing_details,
card => $card,
fpx => $fpx,
ideal => $ideal,
metadata => $metadata,
payment_method => $payment_method,
sepa_debit => $sepa_debit,
);
my $payment_method_obj = Net::Stripe::PaymentMethod->new( %args );
return $self->_post("payment_methods", $payment_method_obj);
}
method get_payment_method(
StripePaymentMethodId :$payment_method_id!,
) {
return $self->_get("payment_methods/$payment_method_id");
}
method update_payment_method(
StripePaymentMethodId :$payment_method_id!,
HashRef :$billing_details?,
HashRef[Int] :$card?,
HashRef[Str]|EmptyStr :$metadata?,
HashRef :$sepa_debit?,
) {
my %args = (
billing_details => $billing_details,
card => $card,
metadata => $metadata,
sepa_debit => $sepa_debit,
);
my $payment_method_obj = Net::Stripe::PaymentMethod->new( %args );
return $self->_post("payment_methods/$payment_method_id", $payment_method_obj);
}
method list_payment_methods(
StripeCustomerId :$customer!,
StripePaymentMethodType :$type!,
Str :$ending_before?,
Int :$limit?,
Str :$starting_after?,
) {
my %args = (
customer => $customer,
type => $type,
ending_before => $ending_before,
limit => $limit,
starting_after => $starting_after,
);
return $self->_get("payment_methods", \%args);
}
method attach_payment_method(
StripeCustomerId :$customer!,
StripePaymentMethodId :$payment_method_id!,
) {
my %args = (
customer => $customer,
);
return $self->_post("payment_methods/$payment_method_id/attach", \%args);
}
method detach_payment_method(
StripePaymentMethodId :$payment_method_id!,
) {
return $self->_post("payment_methods/$payment_method_id/detach");
}
}
Tokens: {
method get_token(Str :$token_id) {
return $self->_get("tokens/$token_id");
}
}
Products: {
method create_product(
Str :$name!,
Bool :$active?,
ArrayRef[Str] :$attributes?,
Str :$caption?,
ArrayRef[Str] :$deactivate_on?,
Str :$description?,
StripeProductId|Str :$id?,
ArrayRef[Str] :$images?,
HashRef[Str] :$metadata?,
HashRef[Num] :$package_dimensions?,
Bool :$shippable?,
Str :$statement_descriptor?,
StripeProductType :$type?,
Str :$unit_label?,
Str :$url?,
) {
my %args = (
name => $name,
active => $active,
attributes => $attributes,
caption => $caption,
deactivate_on => $deactivate_on,
description => $description,
id => $id,
images => $images,
metadata => $metadata,
package_dimensions => $package_dimensions,
shippable => $shippable,
statement_descriptor => $statement_descriptor,
type => $type,
unit_label => $unit_label,
url => $url,
);
my $product_obj = Net::Stripe::Product->new( %args );
return $self->_post('products', $product_obj);
}
method get_product(
StripeProductId|Str :$product_id!,
) {
return $self->_get("products/$product_id");
}
method update_product(
StripeProductId|Str :$product_id!,
Bool :$active?,
ArrayRef[Str] :$attributes?,
Str :$caption?,
ArrayRef[Str] :$deactivate_on?,
Str :$description?,
ArrayRef[Str] :$images?,
HashRef[Str]|EmptyStr :$metadata?,
Str :$name?,
HashRef[Num] :$package_dimensions?,
Bool :$shippable?,
Str :$statement_descriptor?,
StripeProductType :$type?,
Str :$unit_label?,
Str :$url?,
) {
my %args = (
active => $active,
attributes => $attributes,
caption => $caption,
deactivate_on => $deactivate_on,
description => $description,
images => $images,
metadata => $metadata,
name => $name,
package_dimensions => $package_dimensions,
shippable => $shippable,
statement_descriptor => $statement_descriptor,
type => $type,
unit_label => $unit_label,
url => $url,
);
my $product_obj = Net::Stripe::Product->new( %args );
return $self->_post("products/$product_id", $product_obj);
}
method list_products(
Bool :$active?,
ArrayRef[StripeProductId|Str] :$ids,
Bool :$shippable?,
StripeProductType :$type?,
Str :$url?,
HashRef[Str] :$created?,
Str :$ending_before?,
Int :$limit?,
Str :$starting_after?,
) {
my %args = (
path => "products",
active => _encode_boolean( $active ),
ids => $ids,
shippable => _encode_boolean( $shippable ),
type => $type,
url => $url,
created => $created,
ending_before => $ending_before,
limit => $limit,
starting_after => $starting_after,
);
return $self->_get_all( %args );
}
method delete_product(
StripeProductId|Str :$product_id!,
) {
$self->_delete("products/$product_id");
}
}
Plans: {
method post_plan(Str :$id,
Int :$amount,
Str :$currency,
Str :$interval,
Int :$interval_count?,
Str :$name,
StripeProductId|Str :$product,
Int :$trial_period_days?,
HashRef :$metadata?,
Str :$statement_descriptor?) {
my $plan = Net::Stripe::Plan->new(id => $id,
amount => $amount,
currency => $currency,
interval => $interval,
interval_count => $interval_count,
name => $name,
product => $product,
trial_period_days => $trial_period_days,
metadata => $metadata,
statement_descriptor => $statement_descriptor);
return $self->_post('plans', $plan);
}
method get_plan(Str :$plan_id) {
return $self->_get("plans/" . uri_escape($plan_id));
}
method delete_plan(Str|Net::Stripe::Plan $plan) {
if (ref($plan)) {
$plan = $plan->id;
}
$self->_delete("plans/$plan");
}
method get_plans(
StripeProductId|Str :$product?,
Str :$ending_before?,
Int :$limit?,
Str :$starting_after?,
) {
my %args = (
path => 'plans',
product => $product,
ending_before => $ending_before,
limit => $limit,
starting_after => $starting_after,
);
$self->_get_all(%args);
}
}
Coupons: {
method post_coupon(Str :$id?,
Str :$duration,
Int :$amount_off?,
Str :$currency?,
Int :$duration_in_months?,
Int :$max_redemptions?,
HashRef :$metadata?,
Int :$percent_off?,
Int :$redeem_by?) {
my $coupon = Net::Stripe::Coupon->new(id => $id,
duration => $duration,
amount_off => $amount_off,
currency => $currency,
duration_in_months => $duration_in_months,
max_redemptions => $max_redemptions,
metadata => $metadata,
percent_off => $percent_off,
redeem_by => $redeem_by
);
return $self->_post('coupons', $coupon);
}
method get_coupon(Str :$coupon_id) {
return $self->_get("coupons/" . uri_escape($coupon_id));
}
method delete_coupon($id) {
$id = $id->id if ref($id);
$self->_delete("coupons/$id");
}
method get_coupons(Str :$ending_before?, Int :$limit?, Str :$starting_after?) {
my %args = (
path => 'coupons',
ending_before => $ending_before,
limit => $limit,
starting_after => $starting_after,
);
$self->_get_all(%args);
}
}
Discounts: {
method delete_customer_discount(Net::Stripe::Customer|Str :$customer) {
if (ref($customer)) {
$customer = $customer->id;
}
$self->_delete("customers/$customer/discount");
}
}
Invoices: {
method create_invoice(Net::Stripe::Customer|Str :$customer,
Int :$application_fee?,
Str :$description?,
HashRef :$metadata?,
Net::Stripe::Subscription|Str :$subscription?,
Bool :$auto_advance?) {
if (ref($customer)) {
$customer = $customer->id;
}
if (ref($subscription)) {
$subscription = $subscription->id;
}
return $self->_post("invoices",
{
customer => $customer,
application_fee => $application_fee,
description => $description,
metadata => $metadata,
subscription => $subscription,
auto_advance => $auto_advance,
});
}
method post_invoice(Net::Stripe::Invoice|Str :$invoice,
Int :$application_fee?,
Bool :$closed?,
Bool :$auto_advance?,
Str :$description?,
HashRef :$metadata?) {
if (ref($invoice)) {
$invoice = $invoice->id;
}
return $self->_post("invoices/$invoice",
{
application_fee => $application_fee,
closed => $closed,
auto_advance => $auto_advance,
description => $description,
metadata => $metadata
});
}
method get_invoice(Str :$invoice_id) {
return $self->_get("invoices/$invoice_id");
}
method pay_invoice(Str :$invoice_id) {
return $self->_post("invoices/$invoice_id/pay");
}
method get_invoices(Net::Stripe::Customer|Str :$customer?,
Int|HashRef :$date?,
Str :$ending_before?,
Int :$limit?,
Str :$starting_after?) {
if (ref($customer)) {
$customer = $customer->id
}
my %args = (
customer => $customer,
date => $date,
ending_before => $ending_before,
limit => $limit,
starting_after => $starting_after,
);
$self->_get('invoices', \%args);
}
method get_upcominginvoice(Net::Stripe::Customer|Str $customer) {
if (ref($customer)) {
$customer = $customer->id;
}
my %args = (
path => 'invoices/upcoming',
customer => $customer,
);
return $self->_get_all(%args);
}
}
InvoiceItems: {
method create_invoiceitem(Net::Stripe::Customer|Str :$customer,
Int :$amount,
Str :$currency,
Net::Stripe::Invoice|Str :$invoice?,
Net::Stripe::Subscription|Str :$subscription?,
Str :$description?,
HashRef :$metadata?) {
if (ref($customer)) {
$customer = $customer->id;
}
if (ref($invoice)) {
$invoice = $invoice->id;
}
if (ref($subscription)) {
$subscription = $subscription->id;
}
my $invoiceitem = Net::Stripe::Invoiceitem->new(customer => $customer,
amount => $amount,
currency => $currency,
invoice => $invoice,
subscription => $subscription,
description => $description,
metadata => $metadata);
return $self->_post('invoiceitems', $invoiceitem);
}
method post_invoiceitem(Net::Stripe::Invoiceitem|Str :$invoice_item,
Int :$amount?,
Str :$description?,
HashRef :$metadata?) {
if (!defined($amount) && !defined($description) && !defined($metadata)) {
my $item = $invoice_item->clone; $item->clear_currency;
return $self->_post("invoiceitems/" . $item->id, $item);
}
if (ref($invoice_item)) {
$invoice_item = $invoice_item->id;
}
return $self->_post("invoiceitems/" . $invoice_item,
{
amount => $amount,
description => $description,
metadata => $metadata
});
}
method get_invoiceitem(Str :$invoice_item) {
return $self->_get("invoiceitems/$invoice_item");
}
method delete_invoiceitem(Net::Stripe::Invoiceitem|Str :$invoice_item) {
if (ref($invoice_item)) {
$invoice_item = $invoice_item->id;
}
$self->_delete("invoiceitems/$invoice_item");
}
method get_invoiceitems(HashRef :$created?,
Net::Stripe::Customer|Str :$customer?,
Str :$ending_before?,
Int :$limit?,
Str :$starting_after?) {
if (ref($customer)) {
$customer = $customer->id;
}
my %args = (
path => 'invoiceitems',
customer => $customer,
created => $created,
ending_before => $ending_before,
limit => $limit,
starting_after => $starting_after,
);
$self->_get_all(%args);
}
}
# Helper methods
method _get(Str $path!, HashRef|StripeResourceObject $obj?) {
my $uri_obj = URI->new( $self->api_base . '/' . $path );
if ( $obj ) {
my %form_fields = %{ convert_to_form_fields( $obj ) };
$uri_obj->query_form( \%form_fields ) if %form_fields;
}
my $req = GET $uri_obj->as_string;
return $self->_make_request($req);
}
method _delete(Str $path!, HashRef|StripeResourceObject $obj?) {
my $uri_obj = URI->new( $self->api_base . '/' . $path );
if ( $obj ) {
my %form_fields = %{ convert_to_form_fields( $obj ) };
$uri_obj->query_form( \%form_fields ) if %form_fields;
}
my $req = DELETE $uri_obj->as_string;
return $self->_make_request($req);
}
sub convert_to_form_fields {
my $hash = shift;
my $stripe_resource_object_type = Moose::Util::TypeConstraints::find_type_constraint( 'StripeResourceObject' );
if (ref($hash) eq 'HASH') {
my $r = {};
foreach my $key (grep { defined($hash->{$_}) }keys %$hash) {
if ( $stripe_resource_object_type->check( $hash->{$key} ) ) {
%{$r} = ( %{$r}, %{ convert_to_form_fields($hash->{$key}) } );
} elsif (ref($hash->{$key}) eq 'HASH') {
foreach my $fn (keys %{$hash->{$key}}) {
$r->{$key . '[' . $fn . ']'} = $hash->{$key}->{$fn};
}
} elsif (ref($hash->{$key}) eq 'ARRAY') {
$r->{$key . '[]'} = $hash->{$key};
} else {
$r->{$key} = $hash->{$key};
}
}
return $r;
} elsif ($stripe_resource_object_type->check($hash)) {
return { $hash->form_fields };
}
return $hash;
}
method _post(Str $path!, HashRef|StripeResourceObject $obj?) {
my %headers;
if ( $obj ) {
my %form_fields = %{ convert_to_form_fields( $obj ) };
$headers{Content} = [ %form_fields ] if %form_fields;
}
my $req = POST $self->api_base . '/' . $path, %headers;
return $self->_make_request($req);
}
method _get_response(
HTTP::Request :$req!,
Bool :$suppress_api_version? = 0,
) {
$req->header( Authorization =>
"Basic " . encode_base64($self->api_key . ':'));
if ($self->api_version && ! $suppress_api_version) {
$req->header( 'Stripe-Version' => $self->api_version );
}
if ($self->debug_network) {
print STDERR "Sending to Stripe:\n------\n" . $req->as_string() . "------\n";
}
my $resp = $self->ua->request($req);
if ($self->debug_network) {
print STDERR "Received from Stripe:\n------\n" . $resp->as_string() . "------\n";
}
if ($resp->code == 200) {
return $resp;
} elsif ($resp->code == 500) {
die Net::Stripe::Error->new(
type => "HTTP request error",
code => $resp->code,
message => $resp->status_line . " - " . $resp->content,
);
}
my $e = eval {
my $hash = decode_json($resp->content);
Net::Stripe::Error->new($hash->{error})
};
if ($@) {
Net::Stripe::Error->new(
type => "Could not decode HTTP response: $@",
message => $resp->status_line . " - " . $resp->content,
);
};
warn "$e\n" if $self->debug;
die $e;
}
method _make_request(HTTP::Request $req!) {
my $resp = $self->_get_response(
req => $req,
);
my $ref = decode_json( $resp->content );
if ( ref( $ref ) eq 'ARRAY' ) {
# some list-type data structures are arrayrefs in API versions 2012-09-24 and earlier.
# if those data structures are at the top level, such as when
# we request 'GET /charges/cus_.../', we need to coerce that
# arrayref into the form that Net::Stripe::List expects.
return _array_to_object( $ref, $req->uri );
} elsif ( ref( $ref ) eq 'HASH' ) {
# all top-level data structures are hashes in API versions 2012-10-26 and later
return _hash_to_object( $ref );
} else {
die Net::Stripe::Error->new(
type => "HTTP request error",
message => sprintf(
"Invalid object type returned: '%s'",
ref( $ref ) || 'NONREF',
),
);
}
}
sub _hash_to_object {
my $hash = shift;
if ( exists( $hash->{deleted} ) && exists( $hash->{object} ) && $hash->{object} ne 'customer' ) {
delete( $hash->{object} );
}
# coerce pre-2011-08-01 API arrayref list format into a hashref
# compatible with Net::Stripe::List
$hash = _pre_2011_08_01_processing( $hash );
# coerce pre-2012-10-26 API invoice lines format into a hashref
# compatible with Net::Stripe::List
$hash = _pre_2012_10_26_processing( $hash );
# coerce post-2015-02-18 source-type args to to card-type args
$hash = _post_2015_02_18_processing( $hash );
# coerce post-2019-10-17 balance to account_balance
$hash = _post_2019_10_17_processing( $hash );
foreach my $k (grep { ref($hash->{$_}) } keys %$hash) {
my $v = $hash->{$k};
if (ref($v) eq 'HASH' && defined($v->{object})) {
$hash->{$k} = _hash_to_object($v);
} elsif (ref($v) =~ /^(JSON::XS::Boolean|JSON::PP::Boolean)$/) {
$hash->{$k} = $v ? 1 : 0;
}
}
if (defined($hash->{object})) {
if ($hash->{object} eq 'list') {
$hash->{data} = [map { _hash_to_object($_) } @{$hash->{data}}];
return Net::Stripe::List->new($hash);
}
my @words = map { ucfirst($_) } split('_', $hash->{object});
my $object = join('', @words);
my $class = 'Net::Stripe::' . $object;
if (Class::Load::is_class_loaded($class)) {
return $class->new($hash);
}
}
return $hash;
}
sub _array_to_object {
my ( $array, $uri ) = @_;
my $list = _array_to_list( $array );
# strip the protocol, domain and query args in order to mimic the
# url returned with Stripe lists in API versions 2012-10-26 and later
$uri =~ s#\?.*$##;
$uri =~ s#^https://[^/]+##;
$list->{url} = $uri;
return _hash_to_object( $list );
}
sub _array_to_list {
my $array = shift;
my $count = scalar( @$array );
my $list = {
object => 'list',
count => $count,
has_more => 0,
data => $array,
total_count => $count,
};
return $list;
}
# coerce pre-2011-08-01 API arrayref list format into a hashref
# compatible with Net::Stripe::List
sub _pre_2011_08_01_processing {
my $hash = shift;
foreach my $type ( qw/ cards sources subscriptions / ) {
if ( exists( $hash->{$type} ) && ref( $hash->{$type} ) eq 'ARRAY' ) {
$hash->{$type} = _array_to_list( delete( $hash->{$type} ) );
my $customer_id;
if ( exists( $hash->{object} ) && $hash->{object} eq 'customer' && exists( $hash->{id} ) ) {
$customer_id = $hash->{id};
} elsif ( exists( $hash->{customer} ) ) {
$customer_id = $hash->{customer};
}
# Net::Stripe::List->new() will fail without url, but we
# can make debugging easier by providing a message here
die Net::Stripe::Error->new(
type => "object coercion error",
message => sprintf(
"Could not determine customer id while coercing %s list into Net::Stripe::List.",
$type,
),
) unless $customer_id;
# mimic the url sent with standard Stripe lists
$hash->{$type}->{url} = "/v1/customers/$customer_id/$type";
}
}
foreach my $type ( qw/ refunds / ) {
if ( exists( $hash->{$type} ) && ref( $hash->{$type} ) eq 'ARRAY' ) {
$hash->{$type} = _array_to_list( delete( $hash->{$type} ) );
my $charge_id;
if ( exists( $hash->{object} ) && $hash->{object} eq 'charge' && exists( $hash->{id} ) ) {
$charge_id = $hash->{id};
}
# Net::Stripe::List->new() will fail without url, but we
# can make debugging easier by providing a message here
die Net::Stripe::Error->new(
type => "object coercion error",
message => sprintf(
"Could not determine charge id while coercing %s list into Net::Stripe::List.",
$type,
),
) unless $charge_id;
# mimic the url sent with standard Stripe lists
$hash->{$type}->{url} = "/v1/charges/$charge_id/$type";
}
}
foreach my $type ( qw/ charges / ) {
if ( exists( $hash->{$type} ) && ref( $hash->{$type} ) eq 'ARRAY' ) {
$hash->{$type} = _array_to_list( delete( $hash->{$type} ) );
my $payment_intent_id;
if ( exists( $hash->{object} ) && $hash->{object} eq 'payment_intent' && exists( $hash->{id} ) ) {
$payment_intent_id = $hash->{id};
} elsif ( exists( $hash->{payment_intent} ) ) {
$payment_intent_id = $hash->{payment_intent};
}
# Net::Stripe::List->new() will fail without url, but we
# can make debugging easier by providing a message here
die Net::Stripe::Error->new(
type => "object coercion error",
message => sprintf(
"Could not determine payment_intent id while coercing %s list into Net::Stripe::List.",
$type,
),
) unless $payment_intent_id;
# mimic the url sent with standard Stripe lists
$hash->{$type}->{url} = "/v1/charges?payment_intent=$payment_intent_id";
}
}
return $hash;
}
# coerce pre-2012-10-26 API invoice lines format into a hashref
# compatible with Net::Stripe::List
sub _pre_2012_10_26_processing {
my $hash = shift;
if (
exists( $hash->{object} ) && $hash->{object} eq 'invoice' &&
exists( $hash->{lines} ) && ref( $hash->{lines} ) eq 'HASH' &&
! exists( $hash->{lines}->{object} )
) {
my $data = [];
my $lines = delete( $hash->{lines} );
foreach my $key ( sort( keys( %$lines ) ) ) {
my $ref = $lines->{$key};
unless ( ref( $ref ) eq 'ARRAY' ) {
die Net::Stripe::Error->new(
type => "object coercion error",
message => sprintf(
"Found invalid subkey type '%s' while coercing invoice lines into a Net::Stripe::List.",
ref( $ref ),
),
);
}
foreach my $item ( @$ref ) {
push @$data, $item;
}
}
$hash->{lines} = _array_to_list( $data );
my $customer_id;
if ( exists( $hash->{customer} ) ) {
$customer_id = $hash->{customer};
}
# Net::Stripe::List->new() will fail without url, but we
# can make debugging easier by providing a message here
die Net::Stripe::Error->new(
type => "object coercion error",
message => "Could not determine customer id while coercing invoice lines into Net::Stripe::List.",
) unless $customer_id;
# mimic the url sent with standard Stripe lists
$hash->{lines}->{url} = "/v1/invoices/upcoming/lines?customer=$customer_id";
}
return $hash;
}
# coerce post-2015-02-18 source-type args to to card-type args
sub _post_2015_02_18_processing {
my $hash = shift;
if (
exists( $hash->{object} ) &&
( $hash->{object} eq 'charge' || $hash->{object} eq 'customer' )
) {
if (
! exists( $hash->{card} ) &&
exists( $hash->{source} ) && ref( $hash->{source} ) eq 'HASH' &&
exists( $hash->{source}->{object} ) && $hash->{source}->{object} eq 'card'
) {
$hash->{card} = Storable::dclone( $hash->{source} );
}
if (
! exists( $hash->{cards} ) &&
exists( $hash->{sources} ) && ref( $hash->{sources} ) eq 'HASH' &&
exists( $hash->{sources}->{object} ) && $hash->{sources}->{object} eq 'list'
) {
$hash->{cards} = Storable::dclone( $hash->{sources} );
}
my $card_id_type = Moose::Util::TypeConstraints::find_type_constraint( 'StripeCardId' );
if (
! exists( $hash->{default_card} ) &&
exists( $hash->{default_source} ) &&
$card_id_type->check( $hash->{default_source} )
) {
$hash->{default_card} = $hash->{default_source};
}
}
return $hash;
}
# coerce post-2019-10-17 balance to account_balance
fun _post_2019_10_17_processing(
HashRef $hash,
) {
if ( exists( $hash->{object} ) && $hash->{object} eq 'customer' ) {
if ( ! exists( $hash->{account_balance} ) && exists( $hash->{balance} ) ) {
$hash->{account_balance} = $hash->{balance};
}
}
return $hash;
}
method _get_all(
Str :$path!,
Maybe[Str] :$ending_before?,
Maybe[Int] :$limit?,
Maybe[Str] :$starting_after?,
%object_filters,
) {
# minimize the number of API calls by retrieving as many results as
# possible per call. the API currently returns a maximum of 100 results.
my $API_PAGE_SIZE = 100;
my $PAGE_SIZE = $limit;
my $GET_MORE;
if ( defined( $limit ) && ( $limit eq '0' || $limit > $API_PAGE_SIZE ) ) {
$PAGE_SIZE = $API_PAGE_SIZE;
$GET_MORE = 1;
}
my %args = (
%object_filters,
ending_before => $ending_before,
limit => $PAGE_SIZE,
starting_after => $starting_after,
);
my $list = $self->_get($path, \%args);
if ( $GET_MORE && $list->elements() > 0 ) {
# passing 'ending_before' causes the API to start with the oldest
# records. so in order to always provide records in reverse-chronological
# order, we must prepend these to the existing records.
my $REVERSE = defined( $ending_before ) && ! defined( $starting_after );
my $MAX_COUNT = $limit eq '0' ? undef : $limit;
while ( 1 ) {
my $PAGE_SIZE = $API_PAGE_SIZE;
if ( defined( $MAX_COUNT ) ) {
my $TO_FETCH = $MAX_COUNT - scalar( $list->elements );
last if $TO_FETCH <= 0;
$PAGE_SIZE = $TO_FETCH if $TO_FETCH < $PAGE_SIZE;
}
my %args = (
%object_filters,
limit => $PAGE_SIZE,
( $REVERSE ? $list->_previous_page_args() : $list->_next_page_args() ),
);
my $page = $self->_get($path, \%args);
last if $page->is_empty;
$list = Net::Stripe::List::_merge_lists(
lists => [ $REVERSE ?
( $page, $list ) :
( $list, $page )
],
);
}
}
return $list;
}
fun _encode_boolean(
Bool $value!,
) {
# a bare `return` with no arguemnts evaluates to an empty list, resulting
# in 'odd number of elements in hash assignment, so we must return undef
return undef unless defined( $value );
return $value ? 'true' : 'false';
}
method _build_api_base { 'https://api.stripe.com/v1' }
method _build_ua {
my $ua = LWP::UserAgent->new(keep_alive => 4);
$ua->agent("Net::Stripe/" . ($Net::Stripe::VERSION || 'dev'));
return $ua;
}
# since the Stripe API does not have a ping-like method, we have to perform
# an extraneous request in order to retrieve the Stripe-Version header with
# the response. for now, we will use the 'balance' endpoint because it one of
# the simplest and least-privileged.
method _get_stripe_verison_header(
Bool :$suppress_api_version? = 0,
) {
my $path = 'balance';
my $req = GET $self->api_base . '/' . $path;
# swallow the possible invalid API version warning
local $SIG{__WARN__} = sub {};
my $resp = $self->_get_response(
req => $req,
suppress_api_version => $suppress_api_version,
);
my $stripe_version = $resp->header( 'Stripe-Version' );
my $stripe_api_version_type = Moose::Util::TypeConstraints::find_type_constraint( 'StripeAPIVersion' );
die Net::Stripe::Error->new(
type => "API version validation error",
message => sprintf( "Failed to retrieve the Stripe-Version header: '%s'",
defined( $stripe_version ) ? $stripe_version : 'undefined',
),
) unless defined( $stripe_version ) && $stripe_api_version_type->check( $stripe_version );
return $stripe_version;
}
method _get_account_api_version {
my $stripe_version = $self->_get_stripe_verison_header(
suppress_api_version => 1,
);
return $stripe_version;
}
# if we have set an explicit API version, confirm that it is valid. if
# it is invalid, _get_response() dies with an invalid_request_error.
method _validate_api_version_value {
return unless defined( $self->api_version );
my $stripe_version = $self->_get_stripe_verison_header();
die Net::Stripe::Error->new(
type => "API version validation error",
message => sprintf( "Stripe API version mismatch. Sent: '%s'. Received: '%s'.",
$self->api_version,
$stripe_version,
),
) unless $stripe_version eq $self->api_version;
return 1;
}
# if we have set an explicit API version, confirm that it is within the
# appropriate range. otherwise, retrieve the default value for this
# account and confirm that it is within the appropriate range.
method _validate_api_version_range {
if ( $self->force_api_version ) {
warn "bypassing API version range safety check" if $self->debug;
return 1;
}
my $api_version = defined( $self->api_version ) ? $self->api_version : $self->_get_account_api_version();
my @api_version = split( '-', $api_version );
my $api_version_dt;
eval {
$api_version_dt = DateTime->new(
year => $api_version[0],
month => $api_version[1],
day => $api_version[2],
time_zone => 'UTC',
);
};
if ( my $error = $@ ) {
die Net::Stripe::Error->new(
type => "API version validation error",
message => sprintf( "Invalid date string '%s' provided for api_version: %s",
$api_version,
$error,
),
);
}
my @min_api_version = split( '-', Net::Stripe::Constants::MIN_API_VERSION );
my $min_api_version_dt = DateTime->new(
year => $min_api_version[0],
month => $min_api_version[1],
day => $min_api_version[2],
time_zone => 'UTC',
);
my @max_api_version = split( '-', Net::Stripe::Constants::MAX_API_VERSION );
my $max_api_version_dt = DateTime->new(
year => $max_api_version[0],
month => $max_api_version[1],
day => $max_api_version[2],
time_zone => 'UTC',
);
my $format = "Stripe API version %s is not supported by this version of Net::Stripe. " .
"This version of Net::Stripe only supports Stripe API versions from %s to %s. " .
"Please check for a version-appropriate branch at https://github.com/lukec/stripe-perl/branches.";
my $message = sprintf( $format,
$api_version,
Net::Stripe::Constants::MIN_API_VERSION,
Net::Stripe::Constants::MAX_API_VERSION,
);
die Net::Stripe::Error->new(
type => "API version validation error",
message => $message,
) unless $min_api_version_dt <= $api_version_dt && $api_version_dt <= $max_api_version_dt;
return 1;
}
__PACKAGE__->meta->make_immutable;
1;
__END__
=pod
=head1 NAME
Net::Stripe - API client for Stripe.com
=head1 VERSION
version 0.42
=head1 SYNOPSIS
my $stripe = Net::Stripe->new(api_key => $API_KEY);
my $card_token = 'a token';
my $charge = $stripe->post_charge( # Net::Stripe::Charge
amount => 12500,
currency => 'usd',
source => $card_token,
description => 'YAPC Registration',
);
print "Charge was not paid!\n" unless $charge->paid;
my $card = $charge->card; # Net::Stripe::Card
# look up a charge by id
my $same_charge = $stripe->get_charge(charge_id => $charge->id);
# ... and the API mirrors https://stripe.com/docs/api
# Charges: post_charge() get_charge() refund_charge() get_charges()
# Customer: post_customer()
=head1 DESCRIPTION
This module is a wrapper around the Stripe.com HTTP API. Methods are
generally named after the HTTP method and the object name.
This method returns Moose objects for responses from the API.
=head2 VERSIONING
Because of occasional non-backward-compatible changes in the Stripe API, a
given version of the SDK is only guaranteed to support Stripe API versions
within the range defined by C<Net::Stripe::Constants::MIN_API_VERSION> and
C<Net::Stripe::Constants::MAX_API_VERSION>.
If you need a version of the SDK supporting a specific older Stripe API
version, you can check for available versions at
L<https://github.com/lukec/stripe-perl/branches>, or by cloning this
repository, located at <https://github.com/lukec/stripe-perl> and using
<git branch> to view available version-specific branches.
=head3 DEFAULT VERSIONING
If you do not set the Stripe API version on object instantiation, API
calls will default to the API version setting for your Stripe account.
=head3 SPECIFIC VERSIONING
If you set the Stripe API version on object instantiation you are telling
Stripe to use that version of the API instead of the default for your account,
and therefore the available API request and response parameters, the names of
those parameters and the structure of the format of the returned data will all
be dictated by the version that you specify. You can read more about the
details of specific API versions at
<https://stripe.com/docs/upgrades#api-changelog>.
=head3 OUT OF SCOPE VERSIONING
If you are wearing a cowboy hat and think - although your specified account
version is outside the range defined in C<Net::Stripe::Constants> - that your
use case is simple enough that it should "just work", you can create your
object instance with C<force_api_version =E<gt> 1>, but don't say we didn't
warn you!
=head1 METHODS
=head2 new PARAMHASH
This creates a new stripe API object. The following parameters are accepted:
=over
=item api_key
This is required. You get this from your Stripe Account settings.
=item api_version
This is the value of the Stripe-Version header <https://stripe.com/docs/api/versioning>
you wish to transmit for API calls.
=item force_api_version
Set this to true to bypass the safety checks for API version compatibility with
a given version of the SDK. Please see the warnings above, and remember that
you are handling the financial data of your users. Use with extreme caution!
=item debug
You can set this to true to see extra debug info.
=item debug_network
You can set this to true to see the actual network requests.
=back
=head1 ATTRIBUTES
=head2 api_base
Reader: api_base
Type: Str
Additional documentation: This is the base part of the URL for every request made
=head2 api_key
Reader: api_key
Type: Str
This attribute is required.
Additional documentation: You get this from your Stripe Account settings
=head2 api_version
Reader: api_version
Type: StripeAPIVersion
Additional documentation: This is the value of the Stripe-Version header you wish to transmit for API calls
=head2 debug
Reader: debug
Writer: debug
Type: Bool
Additional documentation: The debug flag
=head2 debug_network
Reader: debug_network
Writer: debug_network
Type: Bool
Additional documentation: The debug network request flag
=head2 force_api_version
Reader: force_api_version
Type: Bool
Additional documentation: Set this to true to bypass the safety checks for API version compatibility.
=head2 ua
Reader: ua
Type: Object
Additional documentation: The LWP::UserAgent that is used for requests
=head1 Balance Transaction Methods
=head2 get_balance_transaction
Retrieve a balance transaction.
L<https://stripe.com/docs/api#retrieve_balance_transaction>
=over
=item * id - Str - balance transaction ID to retrieve.
=back
Returns a L<Net::Stripe::BalanceTransaction>.
$stripe->get_balance_transaction(id => 'id');
=head1 Charge Methods
=head2 post_charge
Create a new charge.
L<https://stripe.com/docs/api/charges/create#create_charge>
=over
=item * amount - Int - amount to charge
=item * currency - Str - currency for charge
=item * customer - StripeCustomerId - customer to charge - optional
=item * card - StripeTokenId or StripeCardId - card to use - optional
=item * source - StripeTokenId or StripeCardId - source to use - optional
=item * description - Str - description for the charge - optional
=item * metadata - HashRef - metadata for the charge - optional
=item * capture - Bool - optional
=item * statement_descriptor - Str - descriptor for statement - optional
=item * application_fee - Int - optional
=item * receipt_email - Str - The email address to send this charge's receipt to - optional
=back
Returns L<Net::Stripe::Charge>.
$stripe->post_charge(currency => 'USD', amount => 500, customer => 'testcustomer');
=head2 get_charge
Retrieve a charge.
L<https://stripe.com/docs/api#retrieve_charge>
=over
=item * charge_id - Str - charge id to retrieve
=back
Returns L<Net::Stripe::Charge>.
$stripe->get_charge(charge_id => 'chargeid');
=head2 refund_charge
Refunds a charge.
L<https://stripe.com/docs/api#create_refund>
=over
=item * charge - L<Net::Stripe::Charge> or Str - charge or charge_id to refund
=item * amount - Int - amount to refund in cents, optional
=back
Returns a new L<Net::Stripe::Refund>.
$stripe->refund_charge(charge => $charge, amount => 500);
=head2 get_charges
Returns a L<Net::Stripe::List> object containing L<Net::Stripe::Charge> objects.
L<https://stripe.com/docs/api#list_charges>
=over
=item * created - HashRef - created conditions to match, optional
=item * customer - L<Net::Stripe::Customer> or Str - customer to match
=item * ending_before - Str - ending before condition, optional
=item * limit - Int - maximum number of charges to return, optional
=item * starting_after - Str - starting after condition, optional
=back
Returns a list of L<Net::Stripe::Charge> objects.
$stripe->get_charges(customer => $customer, limit => 5);
=head2 capture_charge
L<https://stripe.com/docs/api/charges/capture#capture_charge>
=over
=item * charge - L<Net::Stripe::Charge> or Str - charge to capture
=item * amount - Int - amount to capture
=back
Returns a L<Net::Stripe::Charge>.
$stripe->capture_charge(charge => $charge_id);
=head1 Customer Methods
=head2 post_customer
Create or update a customer.
L<https://stripe.com/docs/api/customers/create#create_customer>
L<https://stripe.com/docs/api/customers/update#update_customer>
=over
=item * customer - L<Net::Stripe::Customer> or StripeCustomerId - existing customer to update, optional
=item * account_balance - Int, optional
=item * balance - Int, optional
=item * card - L<Net::Stripe::Token> or StripeTokenId, default card for the customer, optional
=item * source - StripeTokenId or StripeSourceId, source for the customer, optional
=item * coupon - Str, optional
=item * default_card - L<Net::Stripe::Token>, L<Net::Stripe::Card>, Str or HashRef, default card for the customer, optional
=item * default_source - StripeCardId or StripeSourceId, default source for the customer, optional
=item * description - Str, optional
=item * email - Str, optional
=item * metadata - HashRef, optional
=item * plan - Str, optional
=item * quantity - Int, optional
=item * trial_end - Int or Str, optional
=back
Returns a L<Net::Stripe::Customer> object.
my $customer = $stripe->post_customer(
source => $token_id,
email => 'stripe@example.com',
description => 'Test for Net::Stripe',
);
=head2 list_subscriptions
Returns the subscriptions for a customer.
L<https://stripe.com/docs/api#list_subscriptions>
=over
=item * customer - L<Net::Stripe::Customer> or Str
=item * ending_before - Str, optional
=item * limit - Int, optional
=item * starting_after - Str, optional
=back
Returns a list of L<Net::Stripe::Subscription> objects.
=head2 get_customer
Retrieve a customer.
L<https://stripe.com/docs/api#retrieve_customer>
=over
=item * customer_id - Str - the customer id to retrieve
=back
Returns a L<Net::Stripe::List> object containing L<Net::Stripe::Customer> objects.
$stripe->get_customer(customer_id => $id);
=head2 delete_customer
Delete a customer.
L<https://stripe.com/docs/api#delete_customer>
=over
=item * customer - L<Net::Stripe::Customer> or Str - the customer to delete
=back
Returns a L<Net::Stripe::Customer> object.
$stripe->delete_customer(customer => $customer);
=head2 get_customers
Returns a list of customers.
L<https://stripe.com/docs/api#list_customers>
=over
=item * created - HashRef - created conditions, optional
=item * ending_before - Str, optional
=item * limit - Int, optional
=item * starting_after - Str, optional
=back
Returns a L<Net::Stripe::List> object containing L<Net::Stripe::Customer> objects.
$stripe->get_customers(limit => 7);
=head1 Card Methods
=head2 get_card
Retrieve information about a customer's card.
L<https://stripe.com/docs/api#retrieve_card>
=over
=item * customer - L<Net::Stripe::Customer> or Str - the customer
=item * card_id - Str - the card ID to retrieve
=back
Returns a L<Net::Stripe::Card>.
$stripe->get_card(customer => 'customer_id', card_id => 'abcdef');
=head2 post_card
Create a card.
L<https://stripe.com/docs/api/cards/create#create_card>
=over
=item * customer - L<Net::Stripe::Customer> or StripeCustomerId
=item * card - L<Net::Stripe::Token> or StripeTokenId
=item * source - StripeTokenId
=back
Returns a L<Net::Stripe::Card>.
$stripe->post_card(customer => $customer, source => $token_id);
=head2 update_card
Update a card.
L<https://stripe.com/docs/api/cards/update#update_card>
=over
=item * customer_id - StripeCustomerId
=item * card_id - StripeCardId
=item * card - HashRef
=back
Returns a L<Net::Stripe::Card>.
$stripe->update_card(
customer_id => $customer_id,
card_id => $card_id,
card => {
name => $new_name,
metadata => {
'account-number' => $new_account_nunmber,
},
},
);
=head2 get_cards
Returns a list of cards.
L<https://stripe.com/docs/api#list_cards>
=over
=item * customer - L<Net::Stripe::Customer> or Str
=item * ending_before - Str, optional
=item * limit - Int, optional
=item * starting_after - Str, optional
=back
Returns a L<Net::Stripe::List> object containing L<Net::Stripe::Card> objects.
$stripe->list_cards(customer => 'abcdec', limit => 10);
=head2 delete_card
Delete a card.
L<https://stripe.com/docs/api#delete_card>
=over
=item * customer - L<Net::Stripe::Customer> or Str
=item * card - L<Net::Stripe::Card> or Str
=back
Returns a L<Net::Stripe::Card>.
$stripe->delete_card(customer => $customer, card => $card);
=head1 Source Methods
=head2 create_source
Create a new source object
L<https://stripe.com/docs/api/sources/create#create_source>
=over
=item * amount - Int - amount associated with the source
=item * currency - Str - currency associated with the source
=item * flow - StripeSourceFlow - authentication flow for the source
=item * mandate - HashRef - information about a mandate attached to the source
=item * metadata - HashRef - metadata for the source
=item * owner - HashRef - information about the owner of the payment instrument
=item * receiver - HashRef - parameters for the receiver flow
=item * redirect - HashRef - parameters required for the redirect flow
=item * source_order - HashRef - information about the items and shipping associated with the source
=item * statement_descriptor - Str - descriptor for statement
=item * token - StripeTokenId - token used to create the source
=item * type - StripeSourceType - type of source to create - required
=item * usage - StripeSourceUsage - whether the source should be reusable or not
=back
Returns a L<Net::Stripe::Source>
$stripe->create_source(
type => 'card',
token => $token_id,
);
=head2 get_source
Retrieve an existing source object
L<https://stripe.com/docs/api/sources/retrieve#retrieve_source>
=over
=item * source_id - StripeSourceId - id of source to retrieve - required
=item * client_secret - Str - client secret of the source
=back
Returns a L<Net::Stripe::Source>
$stripe->get_source(
source_id => $source_id,
);
=head2 update_source
Update the specified source by setting the values of the parameters passed
L<https://stripe.com/docs/api/sources/update#update_source>
=over
=item * source_id - StripeSourceId - id of source to update - required
=item * amount - Int - amount associated with the source
=item * metadata - HashRef - metadata for the source
=item * mandate - HashRef - information about a mandate attached to the source
=item * owner - HashRef - information about the owner of the payment instrument
=item * source_order - HashRef - information about the items and shipping associated with the source
=back
Returns a L<Net::Stripe::Source>
$stripe->update_source(
source_id => $source_id,
owner => {
email => $new_email,
phone => $new_phone,
},
);
=head2 attach_source
Attaches a Source object to a Customer
L<https://stripe.com/docs/api/sources/attach#attach_source>
=over
=item * source_id - StripeSourceId - id of source to be attached - required
=item * customer_id - StripeCustomerId - id of customer to which source should be attached - required
=back
Returns a L<Net::Stripe::Source>
$stripe->attach_source(
customer_id => $customer_id,
source_id => $source->id,
);
=head2 detach_source
Detaches a Source object from a Customer
L<https://stripe.com/docs/api/sources/detach#detach_source>
=over
=item * source_id - StripeSourceId - id of source to be detached - required
=item * customer_id - StripeCustomerId - id of customer from which source should be detached - required
=back
Returns a L<Net::Stripe::Source>
$stripe->detach_source(
customer_id => $customer_id,
source_id => $source->id,
);
=head2 list_sources
List all sources belonging to a Customer
=over
=item * customer_id - StripeCustomerId - id of customer for which source to list sources - required
=item * object - Str - object type - required
=item * ending_before - Str - ending before condition
=item * limit - Int - maximum number of charges to return
=item * starting_after - Str - starting after condition
=back
Returns a L<Net::Stripe::List> object containing objects of the requested type
$stripe->list_sources(
customer_id => $customer_id,
object => 'card',
limit => 10,
);
=head1 Subscription Methods
=head2 post_subscription
Adds or updates a subscription for a customer.
L<https://stripe.com/docs/api#create_subscription>
=over
=item * customer - L<Net::Stripe::Customer>
=item * subscription - L<Net::Stripe::Subscription> or Str
=item * card - L<Net::Stripe::Card>, L<Net::Stripe::Token> or Str, default card for the customer, optional
=item * coupon - Str, optional
=item * description - Str, optional
=item * plan - Str, optional
=item * quantity - Int, optional
=item * trial_end - Int, or Str optional
=item * application_fee_percent - Int, optional
=item * prorate - Bool, optional
=item * cancel_at_period_end - Bool, optional
=back
Returns a L<Net::Stripe::Subscription> object.
$stripe->post_subscription(customer => $customer, plan => 'testplan');
=head2 get_subscription
Returns a customer's subscription.
=over
=item * customer - L<Net::Stripe::Customer> or Str
=back
Returns a L<Net::Stripe::Subscription>.
$stripe->get_subscription(customer => 'test123');
=head2 delete_subscription
Cancel a customer's subscription.
L<https://stripe.com/docs/api#cancel_subscription>
=over
=item * customer - L<Net::Stripe::Customer> or Str
=item * subscription - L<Net::Stripe::Subscription> or Str
=item * at_period_end - Bool, optional
=back
Returns a L<Net::Stripe::Subscription> object.
$stripe->delete_subscription(customer => $customer, subscription => $subscription);
=head1 Token Methods
=head2 get_token
Retrieves an existing token.
L<https://stripe.com/docs/api#retrieve_token>
=over
=item * token_id - Str
=back
Returns a L<Net::Stripe::Token>.
$stripe->get_token(token_id => 'testtokenid');
=head1 Product Methods
=head2 create_product
Create a new Product
L<https://stripe.com/docs/api/products/create#create_product>
L<https://stripe.com/docs/api/service_products/create#create_service_product>
=over
=item * name - Str - name of the product - required
=item * active - Bool - whether the product is currently available for purchase
=item * attributes - ArrayRef[Str] - a list of attributes that each sku can provide values for
=item * caption - Str - a short description
=item * deactivate_on - ArrayRef[Str] - an list of connect application identifiers that cannot purchase this product
=item * description - Str - description
=item * id - Str - unique identifier
=item * images - ArrayRef[Str] - a list of image URLs
=item * metadata - HashRef[Str] - metadata
=item * package_dimensions - HashRef - package dimensions for shipping
=item * shippable - Bool - whether the product is a shipped good
=item * statement_descriptor - Str - descriptor for statement
=item * type - StripeProductType - the type of the product
=item * unit_label - Str - label that represents units of the product
=item * url - Str - URL of a publicly-accessible web page for the product
=back
Returns a L<Net::Stripe::Product>
$stripe->create_product(
name => $product_name,
type => 'good',
);
=head2 get_product
Retrieve an existing Product
L<https://stripe.com/docs/api/products/retrieve#retrieve_product>
L<https://stripe.com/docs/api/service_products/retrieve#retrieve_service_product>
=over
=item * product_id - StripeProductId|Str - id of product to retrieve - required
=back
Returns a L<Net::Stripe::Product>
$stripe->get_product(
product_id => $product_id,
);
=head2 update_product
Update an existing Product
L<https://stripe.com/docs/api/products/update#update_product>
L<https://stripe.com/docs/api/service_products/update#update_service_product>
=over
=item * product_id - StripeProductId|Str - id of product to retrieve - required
=item * active - Bool - whether the product is currently available for purchase
=item * attributes - ArrayRef[Str] - a list of attributes that each sku can provide values for
=item * caption - Str - a short description
=item * deactivate_on - ArrayRef[Str] - an list of connect application identifiers that cannot purchase this product
=item * description - Str - description
=item * images - ArrayRef[Str] - a list of image URLs
=item * metadata - HashRef[Str] - metadata
=item * name - Str - name of the product
=item * package_dimensions - HashRef - package dimensions for shipping
=item * shippable - Bool - whether the product is a shipped good
=item * statement_descriptor - Str - descriptor for statement
=item * type - StripeProductType - the type of the product
=item * unit_label - Str - label that represents units of the product
=item * url - Str - URL of a publicly-accessible web page for the product
=back
Returns a L<Net::Stripe::Product>
$stripe->update_product(
product_id => $product_id,
name => $new_name,
);
=head2 list_products
Retrieve a list of Products
L<https://stripe.com/docs/api/products/list#list_products>
L<https://stripe.com/docs/api/service_products/list#list_service_products>
=over
=item * active - Bool - only return products that are active or inactive
=item * ids - StripeProductId|Str - only return products with the given ids
=item * shippable - Bool - only return products that can or cannot be shipped
=item * url - Str - only return products with the given url
=item * type - StripeProductType - only return products of this type
=item * created - HashRef[Str] - created conditions to match
=item * ending_before - Str - ending before condition
=item * limit - Int - maximum number of objects to return
=item * starting_after - Str - starting after condition
=back
Returns a L<Net::Stripe::List> object containing L<Net::Stripe::Product> objects.
$stripe->list_products(
limit => 5,
);
=head2 delete_product
Delete an existing Product
L<https://stripe.com/docs/api/products/delete#delete_product>
L<https://stripe.com/docs/api/service_products/delete#delete_service_product>
=over
=item * product_id - StripeProductId|Str - id of product to delete - required
=back
Returns hashref of the form
{
deleted => <bool>,
id => <product_id>,
}
$stripe->delete_product(
product_id => $product_id,
);
=head1 Plan Methods
=head2 post_plan
Create a new plan.
L<https://stripe.com/docs/api#create_plan>
=over
=item * id - Str - identifier of the plan
=item * amount - Int - cost of the plan in cents
=item * currency - Str
=item * interval - Str
=item * interval_count - Int - optional
=item * name - Str - name of the plan
=item * trial_period_days - Int - optional
=item * statement_descriptor - Str - optional
=item * metadata - HashRef - optional
=back
Returns a L<Net::Stripe::Plan> object.
$stripe->post_plan(
id => "free-$future_ymdhms",
amount => 0,
currency => 'usd',
interval => 'year',
name => "Freeplan $future_ymdhms",
);
=head2 get_plan
Retrieves a plan.
=over
=item * plan_id - Str
=back
Returns a L<Net::Stripe::Plan>.
$stripe->get_plan(plan_id => 'plan123');
=head2 delete_plan
Delete a plan.
L<https://stripe.com/docs/api#delete_plan>
=over
=item * plan_id - L<Net::Stripe::Plan> or Str
=back
Returns a L<Net::Stripe::Plan> object.
$stripe->delete_plan(plan_id => $plan);
=head2 get_plans
Return a list of plans.
L<https://stripe.com/docs/api#list_plans>
=over
=item * product - StripeProductId|Str - only return plans for the given product
=item * ending_before - Str, optional
=item * limit - Int, optional
=item * starting_after - Str, optional
=back
Returns a L<Net::Stripe::List> object containing L<Net::Stripe::Plan> objects.
$stripe->get_plans(limit => 10);
=head1 Coupon Methods
=head2 post_coupon
Create or update a coupon.
L<https://stripe.com/docs/api#create_coupon>
=over
=item * id - Str, optional
=item * duration - Str
=item * amount_offset - Int, optional
=item * currency - Str, optional
=item * duration_in_months - Int, optional
=item * max_redemptions - Int, optional
=item * metadata - HashRef, optional
=item * percent_off - Int, optional
=item * redeem_by - Int, optional
=back
Returns a L<Net::Stripe::Coupon> object.
$stripe->post_coupon(
id => $coupon_id,
percent_off => 100,
duration => 'once',
max_redemptions => 1,
redeem_by => time() + 100,
);
=head2 get_coupon
Retrieve a coupon.
L<https://stripe.com/docs/api#retrieve_coupon>
=over
=item * coupon_id - Str
=back
Returns a L<Net::Stripe::Coupon> object.
$stripe->get_coupon(coupon_id => 'id');
=head2 delete_coupon
Delete a coupon.
L<https://stripe.com/docs/api#delete_coupon>
=over
=item * coupon_id - Str
=back
Returns a L<Net::Stripe::Coupon>.
$stripe->delete_coupon(coupon_id => 'coupon123');
=head2 get_coupons
=over
=item * ending_before - Str, optional
=item * limit - Int, optional
=item * starting_after - Str, optional
=back
Returns a L<Net::Stripe::List> object containing L<Net::Stripe::Coupon> objects.
$stripe->get_coupons(limit => 15);
=head1 Invoice Methods
=head2 post_invoice
Update an invoice.
=over
=item * invoice - L<Net::Stripe::Invoice>, Str
=item * application_fee - Int - optional
=item * closed - Bool - optional
=item * description - Str - optional
=item * metadata - HashRef - optional
=back
Returns a L<Net::Stripe::Invoice>.
$stripe->post_invoice(invoice => $invoice, closed => 1)
=head2 get_invoice
=over
=item * invoice_id - Str
=back
Returns a L<Net::Stripe::Invoice>.
$stripe->get_invoice(invoice_id => 'testinvoice');
=head2 pay_invoice
=over
=item * invoice_id - Str
=back
Returns a L<Net::Stripe::Invoice>.
$stripe->pay_invoice(invoice_id => 'testinvoice');
=head2 get_invoices
Returns a list of invoices.
L<https://stripe.com/docs/api#list_customer_invoices>
=over
=item * customer - L<Net::Stripe::Customer> or Str, optional
=item * date - Int or HashRef, optional
=item * ending_before - Str, optional
=item * limit - Int, optional
=item * starting_after - Str, optional
=back
Returns a L<Net::Stripe::List> object containing L<Net::Stripe::Invoice> objects.
$stripe->get_invoices(limit => 10);
=head2 create_invoice
Create a new invoice.
L<https://stripe.com/docs/api#create_invoice>
=over
=item * customer - L<Net::Stripe::Customer>, Str
=item * application_fee - Int - optional
=item * description - Str - optional
=item * metadata - HashRef - optional
=item * subscription - L<Net::Stripe::Subscription> or Str, optional
=back
Returns a L<Net::Stripe::Invoice>.
$stripe->create_invoice(customer => 'custid', description => 'test');
=head2 get_invoice
=over
=item * invoice_id - Str
=back
Returns a L<Net::Stripe::Invoice>.
$stripe->get_invoice(invoice_id => 'test');
=head2 get_upcominginvoice
=over
=item * customer, L<Net::Stripe::Customer> or Str
=back
Returns a L<Net::Stripe::Invoice>.
$stripe->get_upcominginvoice(customer => $customer);
=head1 Invoice Item Methods
=head2 create_invoiceitem
Create an invoice item.
L<https://stripe.com/docs/api#create_invoiceitem>
=over
=item * customer - L<Net::Stripe::Customer> or Str
=item * amount - Int
=item * currency - Str
=item * invoice - L<Net::Stripe::Invoice> or Str, optional
=item * subscription - L<Net::Stripe::Subscription> or Str, optional
=item * description - Str, optional
=item * metadata - HashRef, optional
=back
Returns a L<Net::Stripe::Invoiceitem> object.
$stripe->create_invoiceitem(customer => 'test', amount => 500, currency => 'USD');
=head2 post_invoiceitem
Update an invoice item.
L<https://stripe.com/docs/api#create_invoiceitem>
=over
=item * invoice_item - L<Net::Stripe::Invoiceitem> or Str
=item * amount - Int, optional
=item * description - Str, optional
=item * metadata - HashRef, optional
=back
Returns a L<Net::Stripe::Invoiceitem>.
$stripe->post_invoiceitem(invoice_item => 'itemid', amount => 750);
=head2 get_invoiceitem
Retrieve an invoice item.
=over
=item * invoice_item - Str
=back
Returns a L<Net::Stripe::Invoiceitem>.
$stripe->get_invoiceitem(invoice_item => 'testitemid');
=head2 delete_invoiceitem
Delete an invoice item.
=over
=item * invoice_item - L<Net::Stripe::Invoiceitem> or Str
=back
Returns a L<Net::Stripe::Invoiceitem>.
$stripe->delete_invoiceitem(invoice_item => $invoice_item);
=head2 get_invoiceitems
=over
=item * customer - L<Net::Stripe::Customer> or Str, optional
=item * date - Int or HashRef, optional
=item * ending_before - Str, optional
=item * limit - Int, optional
=item * starting_after - Str, optional
=back
Returns a L<Net::Stripe::List> object containing L<Net::Stripe::Invoiceitem> objects.
$stripe->get_invoiceitems(customer => 'test', limit => 30);
=head1 Discount Methods
=head2 delete_customer_discount
Deletes a customer-wide discount.
L<https://stripe.com/docs/api/curl#delete_discount>
=over
=item * customer - L<Net::Stripe::Customer> or Str - the customer with a discount to delete
=back
$stripe->delete_customer_discount(customer => $customer);
Returns hashref of the form
{
deleted => <bool>
}
=head1 Payment Method Methods
=head2 create_payment_method
Create a PaymentMethod
L<https://stripe.com/docs/api/payment_methods/create#create_payment_method>
=over
=item * type - StripePaymentMethodType - type of PaymentMethod - required
=item * card - StripeTokenId - Token id for card associated with the PaymentMethod
=item * billing_details - HashRef - billing information associated with the PaymentMethod
=item * fpx - HashRef - details about the FPX payment method
=item * ideal - HashRef - details about the iDEAL payment method
=item * metadata - HashRef[Str] - metadata
=item * sepa_debit - HashRef - details about the SEPA debit bank account
=back
Returns a L<Net::Stripe::PaymentMethod>.
$stripe->create_payment_method(
type => 'card',
card => $token_id,
);
=head2 get_payment_method
Retrieve an existing PaymentMethod
L<https://stripe.com/docs/api/payment_methods/retrieve#retrieve_payment_method>
=over
=item * payment_method_id - StripePaymentMethodId - id of PaymentMethod to retrieve - required
=back
Returns a L<Net::Stripe::PaymentMethod>
$stripe->get_payment_method(
payment_method_id => $payment_method_id,
);
=head2 update_payment_method
Update a PaymentMethod
L<https://stripe.com/docs/api/payment_methods/update#update_payment_method>
=over
=item * payment_method_id - StripePaymentMethodId - id of PaymentMethod to update - required
=item * billing_details - HashRef - billing information associated with the PaymentMethod
=item * card - HashRef[Int] - card details to update
=item * metadata - HashRef[Str] - metadata
=item * sepa_debit - HashRef - details about the SEPA debit bank account
=back
Returns a L<Net::Stripe::PaymentMethod>
$stripe->update_payment_method(
payment_method_id => $payment_method_id,
metadata => $metadata,
);
=head2 list_payment_methods
Retrieve a list of PaymentMethods
L<https://stripe.com/docs/api/payment_methods/list#list_payment_methods>
=over
=item * customer - StripeCustomerId - return only PaymentMethods for the specified Customer id - required
=item * type - Str - filter by type - required
=item * ending_before - Str - ending before condition
=item * limit - Int - maximum number of objects to return
=item * starting_after - Str - starting after condition
=back
Returns a L<Net::Stripe::List> object containing L<Net::Stripe::PaymentMethod> objects
$stripe->list_payment_methods(
customer => $customer_id,
type => 'card',
limit => 10,
);
=head2 attach_payment_method
Attach a PaymentMethod to a Customer
L<https://stripe.com/docs/api/payment_methods/attach#customer_attach_payment_method>
=over
=item * payment_method_id - StripePaymentMethodId - id of PaymentMethod to attach - required
=item * customer - StripeCustomerId - id of Customer to which to attach the PaymentMethod - required
=back
Returns a L<Net::Stripe::PaymentMethod>
$stripe->attach_payment_method(
payment_method_id => $payment_method_id,
customer => $customer,
);
=head2 detach_payment_method
Detach a PaymentMethod from a Customer
L<https://stripe.com/docs/api/payment_methods/detach#customer_detach_payment_method>
=over
=item * payment_method_id - StripePaymentMethodId - id of PaymentMethod to detach - required
=back
Returns a L<Net::Stripe::PaymentMethod>.
$stripe->detach_payment_method(
payment_method_id => $payment_method_id,
);
=head1 Payment Intent Methods
=head2 create_payment_intent
Create a PaymentIntent object
L<https://stripe.com/docs/api/payment_intents/create#create_payment_intent>
=over
=item * amount - Int - amount intended to be collected by this PaymentIntent - required
=item * currency - Str - currency - required
=item * application_fee_amount - Int - the amount of the application fee
=item * capture_method - StripeCaptureMethod - capture method
=item * confirm - Bool - attempt to confirm this PaymentIntent immediately
=item * confirmation_method - StripeConfirmationMethod - confirmation method
=item * customer - StripeCustomerId - id of Customer this PaymentIntent belongs to
=item * description - Str - description
=item * error_on_requires_action - Bool - fail the payment attempt if the PaymentIntent transitions into `requires_action`
=item * mandate - Str - id of the mandate to be used for this payment
=item * mandate_data - HashRef - details about the Mandate to create
=item * metadata - HashRef[Str] - metadata
=item * off_session - Bool - indicate that the customer is not in your checkout flow
=item * on_behalf_of - Str - Stripe account ID for which these funds are intended
=item * payment_method - StripePaymentMethodId - id of PaymentMethod to attach to this PaymentIntent
=item * payment_method_options - HashRef - PaymentMethod-specific configuration for this PaymentIntent
=item * payment_method_types - ArrayRef[StripePaymentMethodType] - list of PaymentMethod types that this PaymentIntent is allowed to use
=item * receipt_email - Str - email address to send the receipt to
=item * return_url - Str - URL to redirect your customer back to
=item * save_payment_method - Bool - save the payment method to the customer
=item * setup_future_usage - Str - allow future payments with this PaymentIntent's PaymentMethod
=item * shipping - HashRef - shipping information for this PaymentIntent
=item * statement_descriptor - Str - descriptor for statement
=item * statement_descriptor_suffix - Str - suffix to be concatenated with the statement descriptor
=item * transfer_data - HashRef - parameters used to automatically create a Transfer when the payment succeeds
=item * transfer_group - Str - identifies the resulting payment as part of a group
=item * use_stripe_sdk - Bool - use manual confirmation and the iOS or Android SDKs to handle additional authentication steps
=back
Returns a L<Net::Stripe::PaymentIntent>
$stripe->create_payment_intent(
amount => 3300,
currency => 'usd',
);
=head2 get_payment_intent
Retrieve an existing PaymentIntent
L<https://stripe.com/docs/api/payment_intents/retrieve#retrieve_payment_intent>
=over
=item * payment_intent_id - StripePaymentIntentId - id of PaymentIntent to retrieve - required
=item * client_secret - Str - client secret of the PaymentIntent to retrieve
=back
Returns a L<Net::Stripe::PaymentIntent>
$stripe->get_payment_intent(
payment_intent_id => $payment_intent_id,
);
=head2 update_payment_intent
Update an existing PaymentIntent
L<https://stripe.com/docs/api/payment_intents/update#update_payment_intent>
=over
=item * payment_intent_id - StripePaymentIntentId - id of PaymentIntent to update - required
=item * amount - Int - amount intended to be collected by this PaymentIntent - required
=item * application_fee_amount - Int - the amount of the application fee
=item * currency - Str - currency - required
=item * customer - StripeCustomerId - id of Customer this PaymentIntent belongs to
=item * description - Str - description
=item * metadata - HashRef[Str] - metadata
=item * payment_method - StripePaymentMethodId - id of PaymentMethod to attach to this PaymentIntent
=item * payment_method_options - HashRef - PaymentMethod-specific configuration for this PaymentIntent
=item * payment_method_types - ArrayRef[StripePaymentMethodType] - list of PaymentMethod types that this PaymentIntent is allowed to use
=item * receipt_email - Str - email address to send the receipt to
=item * save_payment_method - Bool - save the payment method to the customer
=item * setup_future_usage - Str - allow future payments with this PaymentIntent's PaymentMethod
=item * shipping - HashRef - shipping information for this PaymentIntent
=item * statement_descriptor - Str - descriptor for statement
=item * statement_descriptor_suffix - Str - suffix to be concatenated with the statement descriptor
=item * transfer_data - HashRef - parameters used to automatically create a Transfer when the payment succeeds
=item * transfer_group - Str - identifies the resulting payment as part of a group
=back
Returns a L<Net::Stripe::PaymentIntent>
$stripe->update_payment_intent(
payment_intent_id => $payment_intent_id,
description => 'Updated Description',
);
=head2 confirm_payment_intent
Confirm that customer intends to pay with provided PaymentMethod
L<https://stripe.com/docs/api/payment_intents/confirm#confirm_payment_intent>
=over
=item * payment_intent_id - StripePaymentIntentId - id of PaymentIntent to confirm - required
=item * client_secret - Str - client secret of the PaymentIntent
=item * error_on_requires_action - Bool - fail the payment attempt if the PaymentIntent transitions into `requires_action`
=item * mandate - Str - id of the mandate to be used for this payment
=item * mandate_data - HashRef - details about the Mandate to create
=item * off_session - Bool - indicate that the customer is not in your checkout flow
=item * payment_method - StripePaymentMethodId - id of PaymentMethod to attach to this PaymentIntent
=item * payment_method_options - HashRef - PaymentMethod-specific configuration for this PaymentIntent
=item * payment_method_types - ArrayRef[StripePaymentMethodType] - list of PaymentMethod types that this PaymentIntent is allowed to use
=item * receipt_email - Str - email address to send the receipt to
=item * return_url - Str - URL to redirect your customer back to
=item * save_payment_method - Bool - save the payment method to the customer
=item * setup_future_usage - Str - allow future payments with this PaymentIntent's PaymentMethod
=item * shipping - HashRef - shipping information for this PaymentIntent
=item * use_stripe_sdk - Bool - use manual confirmation and the iOS or Android SDKs to handle additional authentication steps
=back
Returns a L<Net::Stripe::PaymentIntent>
$stripe->confirm_payment_intent(
payment_intent_id => $payment_intent_id,
);
=head2 capture_payment_intent
Capture the funds for the PaymentIntent
L<https://stripe.com/docs/api/payment_intents/capture#capture_payment_intent>
=over
=item * payment_intent_id - StripePaymentIntentId - id of PaymentIntent to capture - required
=item * amount_to_capture - Int - amount to capture from the PaymentIntent
=item * application_fee_amount - Int - application fee amount
=item * statement_descriptor - Str - descriptor for statement
=item * statement_descriptor_suffix - Str - suffix to be concatenated with the statement descriptor
=item * transfer_data - HashRef - parameters used to automatically create a Transfer when the payment succeeds
=back
Returns a L<Net::Stripe::PaymentIntent>
$stripe->capture_payment_intent(
payment_intent_id => $payment_intent_id,
);
=head2 cancel_payment_intent
Cancel the PaymentIntent
L<https://stripe.com/docs/api/payment_intents/cancel#cancel_payment_intent>
=over
=item * payment_intent_id - StripePaymentIntentId - id of PaymentIntent to cancel - required
=item * cancellation_reason - StripeCancellationReason - reason for cancellation
=back
Returns a L<Net::Stripe::PaymentIntent>
$stripe->cancel_payment_intent(
payment_intent_id => $payment_intent_id,
cancellation_reason => 'requested_by_customer',
);
=head2 list_payment_intents
Retrieve a list of PaymentIntents
L<https://stripe.com/docs/api/payment_intents/list#list_payment_intents>
=over
=item * customer - StripeCustomerId - return only PaymentIntents for the specified Customer id
=item * created - HashRef[Int] - created conditions to match
=item * ending_before - Str - ending before condition
=item * limit - Int - maximum number of objects to return
=item * starting_after - Str - starting after condition
=back
Returns a L<Net::Stripe::List> object containing L<Net::Stripe::PaymentIntent> objects.
$stripe->list_payment_intents(
customer => $customer_id,
type => 'card',
limit => 10,
);
=head1 RELEASE NOTES
=head2 Version 0.40
=head3 BREAKING CHANGES
=over
=item deprecate direct handling of PANs
Stripe strongly discourages direct handling of PANs (primary account numbers),
even in test mode, and returns invalid_request_error when passing PANs to the
API from accounts that were created after October 2017. In live mode, all
tokenization should be performed via client-side libraries because direct
handling of PANs violates PCI compliance. So we have removed the methods and
parameter constraints that allow direct handling of PANs and updated the
unit tests appropriately.
=item updating customer card by passing Customer object to post_customer()
If you have code that updates a customer card by updating the internal values
for an existing customer object and then posting that object:
my $customer_obj = $stripe->get_customer(
customer_id => $customer_id,
);
$customer_obj->card( $new_card );
$stripe->post_customer( customer => $customer_obj );
you must unset the default_card attribute in the existing object before
posting the customer object.
$customer_obj->default_card( undef );
Otherwise there is a conflict, since the old default_card attribute in the
object is serialized in the POST stream, and it appears that you are requesting
to set default_card to the id of a card that no longer exists, but rather
is being replaced by the new value of the card attribute in the object.
=item Plan objects now linked to Product objects
For Stripe API versions after 2018-02-15 L<https://stripe.com/docs/upgrades#2018-02-05>
each Plan object is linked to a Product object with type=service. The
Plan object 'name' and 'statement_descriptor' attributes have been moved to
Product objects.
=back
=head3 DEPRECATION WARNING
=over
=item update 'card' to 'source' for Charge and Customer
While the API returns both card-related and source-related values for earlier
versions, making the update mostly backwards-compatible, Stripe API versions
after 2015-02-18 L<https://stripe.com/docs/upgrades#2015-02-18> will no longer
return the card-related values, so you should update your code where necessary
to use the 'source' argument and method for Charge objects, and the 'source',
'sources' and 'default_source' arguments and methods for Customer, in
preparation for the eventual deprecation of the card-related arguments.
=item update 'account_balance' to 'balance' for Customer
While the API returns both 'account_balance' and 'balance' for earlier
versions, making the update backwards-compatible, Stripe API versions
after 2019-10-17 L<https://stripe.com/docs/upgrades#2019-10-17> do not
accept or return 'account_balance', so you should update your code where
necessary to use the 'balance' argument and method for Customer objects in
preparation for the eventual deprecation of the 'account_balance' argument.
=item use 'cancel_at_period_end' instead of 'at_period_end' for canceling Subscriptions
For Stripe API versions after 2018-08-23
L<https://stripe.com/docs/upgrades#2018-08-23>, you can no longer use
'at_period_end' in delete_subscription(). The delete_subscription() method
is reserved for immediate canceling going forward. You should update your
code to use 'cancel_at_period_end in update_subscription() instead.
=item update 'date' to 'created' for Invoice
While the API returns both 'date' and 'created' for earlier versions, making
the update backwards-compatible, Stripe API versions after 2019-03-14
L<https://stripe.com/docs/upgrades#2019-03-14> only return 'created', so you
should update your code where necessary to use the 'created' method for
Invoice objects in preparation for the eventual deprecation of the 'date'
argument.
=item use 'auto_advance' instead of 'closed' for Invoice
The 'closed' attribute for the Invoice object controls automatic collection,
and has been deprecated in favor of the more specific 'auto_advance' attribute.
Where you might have set closed=true on Invoices in the past, set
auto_advance=false. While the API returns both 'closed' and 'auto_advance'
for earlier versions, making the update backwards-compatible, Stripe API
versions after 2018-11-08 L<https://stripe.com/docs/upgrades#2018-11-08>
only return 'auto_advance', so you should update your code where necessary to
use the 'auto_advance' argument and method for Invoice objects in preparation
for the eventual deprecation of the 'closed' argument.
=back
=head3 BUG FIXES
=over
=item fix post_charge() arguments
Some argument types for `customer` and `card` were non-functional in the
current code and have been removed from the Kavorka method signature. We
removed `Net::Stripe::Customer` and `HashRef` for `customer` and we removed
`Net::Stripe::Card` and `Net::Stripe::Token` for `card`, as none of these
forms were being serialized correctly for passing to the API call. We must
retain `Net::Stripe::Card` for the `card` attribute in `Net::Stripe::Charge`
because the `card` value returned from the API call is objectified into
a card object. We have also added TypeConstraints for the string arguments
passed and added in-method validation to ensure that the passed argument
values make sense in the context of one another.
=item cleanup post_card()
Some argument types for `card` are not legitimate, or are being deprecated
and have been removed from the Kavorka method signature. We removed
`Net::Stripe::Card` and updated the string validation to disallow card id
for `card`, as neither of these are legitimate when creating or updating a
card L<https://github.com/lukec/stripe-perl/issues/138>, and the code path
that appeared to be handling `Net::Stripe::Card` was actually unreachable
L<https://github.com/lukec/stripe-perl/issues/100>. We removed the dead code
paths and made the conditional structure more explicit, per discussion in
L<https://github.com/lukec/stripe-perl/pull/133>. We also added unit tests
for all calling forms, per L<https://github.com/lukec/stripe-perl/issues/139>.
=item fix post_customer() arguments
Some argument types for `card` are not legitimate and have been removed from
the Kavorka method signature. We removed `Net::Stripe::Card` and updated the
string validation to disallow card id for `card`, as neither of these are
legitimate when creating or updating a customer L<https://github.com/lukec/stripe-perl/issues/138>.
We have also updated the structure of the method so that we always create a
Net::Stripe::Customer object before posting L<https://github.com/lukec/stripe-perl/issues/148>
and cleaned up and centralized Net::Stripe:Token coercion code.
=item default_card not updating in post_customer()
Prior to ff84dd7, we were passing %args directly to _post(), and therefore
default_card would have been included in the POST stream. Now we are creating
a L<Net::Stripe::Customer> object and posting it, so the posted parameters
rely on the explicit list in $customer_obj->form_fields(), which was lacking
default_card. See also BREAKING CHANGES.
=back
=head3 ENHANCEMENTS
=over
=item coerce old lists
In older Stripe API versions, some list-type data structures were returned
as arrayrefs. We now coerce those old-style lists and collections into the
hashref format that newer versions of the API return, with metadata stored
top-level keys and the list elements in an arrayref with the key 'data',
which is the format that C<Net::Stripe::List> expects. This makes the SDK
compatible with the Stripe API back to the earliest documented API version
L<https://stripe.com/docs/upgrades#2011-06-21>.
=item encode card metdata in convert_to_form_fields()
When passing a hashref with a nested metadata hashref to _post(), that
metadata must be encoded properly before being passed to the Stripe API.
There is now a dedicated block in convert_to_form_fields for this operation.
This update was necessary because of the addition of update_card(), which
accepts a card hashref, which may include metadata.
=item encode objects in convert_to_form_fields()
We removed the nested tertiary operator in _post() and updated
convert_to_form_fields() so that it now handles encoding of objects, both
top-level and nested. This streamlines the hashref vs object serailizing
code, making it easy to adapt to other methods.
=item remove manual serialization in _get_collections() and _get_with_args()
We were using string contatenation to both serilize the individual query args,
in _get_collections(), and to join the individual query args together, in
_get_with_args(). This also involved some unnecessary duplication of the
logic that convert_to_form_fields() was already capable of handling. We now
use convert_to_form_fields() to process the passed data, and L<URI> to
encode and serialize the query string. Along with other updates to
convert_to_form_fields(), _get() can now easily handle the same calling
form as _post(), eliminating the need for _get_collections() and
_get_with_args(). We have also updated _delete() accordingly.
=item add _get_all()
Similar to methods provided by other SDKs, calls using this method will allow
access to all records for a given object type without having to manually
paginate through the results. It is not intended to be used directly, but
will be accessed through new and existing list-retrieval methods. In order to
maintain backwards-compatibility with existing list retrieval behavior, this
method supports passing a value of 0 for 'limit' in order to retrieve all
records. Any other positive integer value for 'limit' will attempt to retrieve
that number of records up to the maximum available. As before, not passing a
value for 'limit', or explicitly passing an undefined value, retrieves whatever
number of records the API returns by default.
=back
=head3 UPDATES
=over
=item update statement_description to statement_descriptor
The statement_description attribute is now statement_descriptor for
L<Net::Stripe::Charge> and L<Net::Stripe::Plan>. The API docs
L<https://stripe.com/docs/upgrades#2014-12-17> indicate that this change
is backwards-compatible. You must update your code to reflect this change
for parameters passed to these objects and methods called on these objects.
=item update unit tests for Charge->status
For Stripe API versions after 2015-02-18 L<https://stripe.com/docs/upgrades#2015-02-18>,
the status property on the Charge object has a value of 'succeeded' for
successful charges. Previously, the status property would be 'paid' for
successful charges. This change does not affect the API calls themselves, but
if your account is using Stripe API version 2015-02-18 or later, you should
update any code that relies on strict checking of the return value of
Charge->status.
=item add update_card()
This method allows updates to card address, expiration, metadata, etc for
existing customer cards.
=item update Token attributes
Added type and client_ip attributes for L<Net::Stripe::Token>.
=item allow capture of partial charge
Passing 'amount' to capture_charge() allows capture of a partial charge.
Per the API, any remaining amount is immediately refunded. The charge object
now also has a 'refunds' attribute, representing a L<Net::Stripe::List>
of L<Net::Stripe::Refund> objects for the charge.
=item add Source
Added a Source object. Also added 'source' attribute and argument for Charge
objects and methods, and added 'source', 'sources' and 'default_source'
attributes and arguments for Customer objects and methods.
=item add balance for Customer
Added 'balance' attribute and arguments for Customer objects and methods.
=item add cancel_at_period_end for update_subscription()
Added 'cancel_at_period_end' argument update_subscription() and added
'cancel_at_period_end' to the POST stream for Subscription objects.
=item update Invoice
Added 'created' attribute for Invoice objects, and removed the required
constraint for the deprecated 'date' attribute. Also added the 'auto_advance'
attribute and removed the required constraint for the deprecated 'closed'
attribute.
=item add Product
Added a Product object. Also added 'product' attribute and argument for Plan
objects and methods.
=item add PaymentMethod and PaymentIntent
Added PaymentMethod and PaymentIntent objects and methods.
=back
=head1 SEE ALSO
L<https://stripe.com>, L<https://stripe.com/docs/api>
=encoding UTF-8
=head1 AUTHORS
=over 4
=item *
Luke Closs
=item *
Rusty Conover
=back
=head1 CONTRIBUTORS
=for stopwords Andrew Solomon Brian Collins Devin M. Certas Dimitar Petrov Dylan Reinhold E. Choroba Florian Heyer Hermann Calabria Jonathan "Duke" Leto Luke Closs Mohammad S Anwar Olaf Alders Paul Cochrane Peter Scott Rusty Conover Sachin Sebastian ...
=over 4
=item *
Andrew Solomon <andrew@geekuni.com>
=item *
Andrew Solomon <andrew@illywhacker.net>
=item *
Brian Collins <bricollins@gmail.com>
=item *
Devin M. Certas <devin@nacredata.com>
=item *
Dimitar Petrov <mitakaa@gmail.com>
=item *
Dylan Reinhold <dylan@gasdasoftware.com>
=item *
E. Choroba <choroba@matfyz.cz>
=item *
Florian Heyer <info@heyer-it.de>
=item *
Hermann Calabria <hermann@ivouch.com>
=item *
Jonathan "Duke" Leto <jonathan@leto.net>
=item *
Luke Closs <lukec@users.noreply.github.com>
=item *
Luke Closs <me@luk.ec>
=item *
Mohammad S Anwar <mohammad.anwar@yahoo.com>
=item *
Olaf Alders <olaf@wundersolutions.com>
=item *
Paul Cochrane <paul@liekut.de>
=item *
Peter Scott <peter@shotgundriver.com>
=item *
Rusty Conover <rusty@luckydinosaur.com>
=item *
Sachin Sebastian <sachinjsk@users.noreply.github.com>
=item *
Sherrard Burton <32931314+sherrardb@users.noreply.github.com>
=item *
Sherrard Burton <sburton@allafrica.com>
=item *
Slobodan MiÅ¡koviÄ <slobodan@miskovic.ca>
=item *
Tom Eliaz <tom@tomeliaz.com>
=back
=head1 COPYRIGHT AND LICENSE
This software is copyright (c) 2015 by Prime Radiant, Inc., (c) copyright 2014 Lucky Dinosaur LLC.
This is free software; you can redistribute it and/or modify it under
the same terms as the Perl 5 programming language system itself.
=cut
( run in 0.291 second using v1.01-cache-2.11-cpan-3989ada0592 )