pay 8.2.2 → 9.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/README.md +6 -4
- data/app/models/pay/braintree/customer.rb +1 -1
- data/app/models/pay/braintree/payment_method.rb +1 -1
- data/app/models/pay/braintree/subscription.rb +2 -5
- data/app/models/pay/fake_processor/subscription.rb +5 -1
- data/app/models/pay/lemon_squeezy/charge.rb +3 -5
- data/app/models/pay/lemon_squeezy/subscription.rb +3 -5
- data/app/models/pay/paddle_billing/charge.rb +3 -5
- data/app/models/pay/paddle_billing/subscription.rb +3 -5
- data/app/models/pay/stripe/charge.rb +4 -6
- data/app/models/pay/stripe/merchant.rb +1 -9
- data/app/models/pay/stripe/payment_method.rb +2 -2
- data/app/models/pay/stripe/subscription.rb +2 -2
- data/lib/pay/stripe/webhooks/payment_action_required.rb +2 -1
- data/lib/pay/stripe/webhooks/payment_failed.rb +2 -1
- data/lib/pay/stripe/webhooks/subscription_renewing.rb +4 -2
- data/lib/pay/stripe.rb +1 -1
- data/lib/pay/version.rb +1 -1
- data/lib/pay.rb +13 -6
- data/lib/tasks/pay.rake +2 -2
- metadata +3 -4
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 6e2755f5ff01aedf32875c2ee941d4e92efe0c044ed7b6eeb601e6d0e17d1d51
|
4
|
+
data.tar.gz: ec10df5f7c8a4d52a55f8fc533fe42bc33a773dc226a348dfee0b80be37c1536
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: bbd81b0e97724c907bb691dc7576b9e458d467064219311feb2e6a0628c31e02b835f99e83daa92a5718f6189859daaa92851d32fe7395d5b2e140815cbe7d59
|
7
|
+
data.tar.gz: 5f143cde69bdc489d423aa7619770b5d1256303a890ea350e2c40090f29af4c4baea39094a9e2e7c854e80ee12ea3d0c360ded743f15b5432fa5881807fa7074
|
data/README.md
CHANGED
@@ -2,13 +2,14 @@
|
|
2
2
|
|
3
3
|
# 💳 Pay - Payments engine for Ruby on Rails
|
4
4
|
|
5
|
-
[](https://github.com/pay-rails/pay/actions) [](https://badge.fury.io/rb/pay)
|
6
|
-
|
7
|
-
<img src="docs/images/stripe_partner_badge.svg" height="26px">
|
5
|
+
[](https://github.com/pay-rails/pay/actions) [](https://badge.fury.io/rb/pay) <img src="docs/images/stripe_partner_badge.svg" height="26px">
|
8
6
|
|
9
7
|
Pay is a payments engine for Ruby on Rails 6.0 and higher.
|
10
8
|
|
11
|
-
|
9
|
+
> [!TIP]
|
10
|
+
> Check out [Jumpstart](https://jumpstartrails.com) for Rails Starter Kit with Pay already integrated!
|
11
|
+
|
12
|
+
**Upgrading?** Check the [UPGRADE](UPGRADE.md) guide for required changes and/or migration when upgrading from a previous version of Pay.
|
12
13
|
|
13
14
|
## 🧑💻 Tutorial
|
14
15
|
|
@@ -48,6 +49,7 @@ Want to add a new payment provider? Contributions are welcome.
|
|
48
49
|
* [Paddle](docs/paddle_billing/1_overview.md)
|
49
50
|
* [Lemon Squeezy](docs/lemon_squeezy/1_overview.md)
|
50
51
|
* [Fake Processor](docs/fake_processor/1_overview.md)
|
52
|
+
* [Asaas (Community)](https://github.com/PedroAugustoRamalhoDuarte/pay-asaas)
|
51
53
|
* **Marketplaces**
|
52
54
|
* [Stripe Connect](docs/marketplaces/stripe_connect.md)
|
53
55
|
* **Contributing**
|
@@ -147,7 +147,7 @@ module Pay
|
|
147
147
|
end
|
148
148
|
end
|
149
149
|
|
150
|
-
charge =
|
150
|
+
charge = Pay::Braintree::Charge.find_or_initialize_by(customer: self, processor_id: transaction.id)
|
151
151
|
charge.update!(attrs)
|
152
152
|
charge
|
153
153
|
end
|
@@ -4,7 +4,7 @@ module Pay
|
|
4
4
|
def self.sync(id, object: nil, try: 0, retries: 1)
|
5
5
|
object ||= Pay.braintree_gateway.payment_method.find(id)
|
6
6
|
|
7
|
-
pay_customer = Pay::Customer.find_by(
|
7
|
+
pay_customer = Pay::Braintree::Customer.find_by(processor_id: object.customer_id)
|
8
8
|
return unless pay_customer
|
9
9
|
|
10
10
|
pay_customer.save_payment_method(object, default: object.default?)
|
@@ -31,15 +31,12 @@ module Pay
|
|
31
31
|
attributes[:ends_at] = object.paid_through_date.end_of_day
|
32
32
|
end
|
33
33
|
|
34
|
-
pay_subscription =
|
35
|
-
if pay_subscription
|
34
|
+
if (pay_subscription = find_by(customer: pay_customer, processor_id: object.id))
|
36
35
|
pay_subscription.with_lock { pay_subscription.update!(attributes) }
|
37
36
|
else
|
38
37
|
name ||= Pay.default_product_name
|
39
|
-
|
38
|
+
create!(attributes.merge(customer: pay_customer, name: name, processor_id: object.id))
|
40
39
|
end
|
41
|
-
|
42
|
-
pay_subscription
|
43
40
|
rescue ActiveRecord::RecordInvalid, ActiveRecord::RecordNotUnique
|
44
41
|
try += 1
|
45
42
|
if try <= retries
|
@@ -1,11 +1,15 @@
|
|
1
1
|
module Pay
|
2
2
|
module FakeProcessor
|
3
3
|
class Subscription < Pay::Subscription
|
4
|
+
def self.sync(processor_id, **options)
|
5
|
+
# Bypass sync operation for FakeProcessor
|
6
|
+
end
|
7
|
+
|
4
8
|
def api_record(**options)
|
5
9
|
self
|
6
10
|
end
|
7
11
|
|
8
|
-
# With trial, sets end to trial end (
|
12
|
+
# With trial, sets end to trial end (mimicking Stripe)
|
9
13
|
# Without trial, sets can ends_at to end of month
|
10
14
|
def cancel(**options)
|
11
15
|
return if canceled?
|
@@ -22,13 +22,11 @@ module Pay
|
|
22
22
|
}
|
23
23
|
|
24
24
|
# Update or create the charge
|
25
|
-
if (pay_charge =
|
26
|
-
pay_charge.with_lock
|
27
|
-
pay_charge.update!(attributes)
|
28
|
-
end
|
25
|
+
if (pay_charge = find_by(customer: pay_customer, processor_id: processor_id))
|
26
|
+
pay_charge.with_lock { pay_charge.update!(attributes) }
|
29
27
|
pay_charge
|
30
28
|
else
|
31
|
-
|
29
|
+
create!(attributes.merge(customer: pay_customer, processor_id: processor_id))
|
32
30
|
end
|
33
31
|
end
|
34
32
|
|
@@ -33,13 +33,11 @@ module Pay
|
|
33
33
|
end
|
34
34
|
|
35
35
|
# Update or create the subscription
|
36
|
-
if (pay_subscription =
|
37
|
-
pay_subscription.with_lock
|
38
|
-
pay_subscription.update!(attributes)
|
39
|
-
end
|
36
|
+
if (pay_subscription = find_by(customer: pay_customer, processor_id: object.id))
|
37
|
+
pay_subscription.with_lock { pay_subscription.update!(attributes) }
|
40
38
|
pay_subscription
|
41
39
|
else
|
42
|
-
|
40
|
+
create!(attributes.merge(customer: pay_customer, name: name, processor_id: object.id))
|
43
41
|
end
|
44
42
|
end
|
45
43
|
|
@@ -48,13 +48,11 @@ module Pay
|
|
48
48
|
end
|
49
49
|
|
50
50
|
# Update or create the charge
|
51
|
-
if (pay_charge =
|
52
|
-
pay_charge.with_lock
|
53
|
-
pay_charge.update!(attrs)
|
54
|
-
end
|
51
|
+
if (pay_charge = find_by(customer: pay_customer, processor_id: object.id))
|
52
|
+
pay_charge.with_lock { pay_charge.update!(attrs) }
|
55
53
|
pay_charge
|
56
54
|
else
|
57
|
-
|
55
|
+
create!(attrs.merge(customer: pay_customer, processor_id: object.id))
|
58
56
|
end
|
59
57
|
rescue ActiveRecord::RecordInvalid, ActiveRecord::RecordNotUnique
|
60
58
|
try += 1
|
@@ -54,13 +54,11 @@ module Pay
|
|
54
54
|
end
|
55
55
|
|
56
56
|
# Update or create the subscription
|
57
|
-
if (pay_subscription =
|
58
|
-
pay_subscription.with_lock
|
59
|
-
pay_subscription.update!(attributes)
|
60
|
-
end
|
57
|
+
if (pay_subscription = find_by(customer: pay_customer, processor_id: subscription_id))
|
58
|
+
pay_subscription.with_lock { pay_subscription.update!(attributes) }
|
61
59
|
pay_subscription
|
62
60
|
else
|
63
|
-
|
61
|
+
create!(attributes.merge(customer: pay_customer, name: name, processor_id: subscription_id))
|
64
62
|
end
|
65
63
|
rescue ActiveRecord::RecordInvalid, ActiveRecord::RecordNotUnique
|
66
64
|
try += 1
|
@@ -5,7 +5,7 @@ module Pay
|
|
5
5
|
|
6
6
|
def self.sync(charge_id, object: nil, stripe_account: nil, try: 0, retries: 1)
|
7
7
|
# Skip loading the latest charge details from the API if we already have it
|
8
|
-
object ||= ::Stripe::Charge.retrieve({id: charge_id, expand: ["invoice.total_discount_amounts.discount", "invoice.total_tax_amounts.tax_rate", "refunds"]}, {stripe_account: stripe_account}.compact)
|
8
|
+
object ||= ::Stripe::Charge.retrieve({id: charge_id, expand: ["balance_transaction", "invoice.total_discount_amounts.discount", "invoice.total_tax_amounts.tax_rate", "refunds"]}, {stripe_account: stripe_account}.compact)
|
9
9
|
if object.customer.blank?
|
10
10
|
Rails.logger.debug "Stripe Charge #{object.id} does not have a customer"
|
11
11
|
return
|
@@ -81,13 +81,11 @@ module Pay
|
|
81
81
|
end
|
82
82
|
|
83
83
|
# Update or create the charge
|
84
|
-
if (pay_charge =
|
85
|
-
pay_charge.with_lock
|
86
|
-
pay_charge.update!(attrs)
|
87
|
-
end
|
84
|
+
if (pay_charge = find_by(customer: pay_customer, processor_id: object.id))
|
85
|
+
pay_charge.with_lock { pay_charge.update!(attrs) }
|
88
86
|
pay_charge
|
89
87
|
else
|
90
|
-
|
88
|
+
create!(attrs.merge(customer: pay_customer, processor_id: object.id))
|
91
89
|
end
|
92
90
|
rescue ActiveRecord::RecordInvalid, ActiveRecord::RecordNotUnique
|
93
91
|
try += 1
|
@@ -2,15 +2,7 @@ module Pay
|
|
2
2
|
module Stripe
|
3
3
|
class Merchant < Pay::Merchant
|
4
4
|
def create_account(**options)
|
5
|
-
|
6
|
-
type: "express",
|
7
|
-
capabilities: {
|
8
|
-
card_payments: {requested: true},
|
9
|
-
transfers: {requested: true}
|
10
|
-
}
|
11
|
-
}
|
12
|
-
|
13
|
-
stripe_account = ::Stripe::Account.create(defaults.merge(options))
|
5
|
+
stripe_account = ::Stripe::Account.create(options)
|
14
6
|
update(processor_id: stripe_account.id)
|
15
7
|
stripe_account
|
16
8
|
rescue ::Stripe::StripeError => e
|
@@ -36,8 +36,8 @@ module Pay
|
|
36
36
|
|
37
37
|
attributes = extract_attributes(object).merge(default: default, stripe_account: stripe_account)
|
38
38
|
|
39
|
-
pay_customer.
|
40
|
-
pay_payment_method =
|
39
|
+
where(customer: pay_customer).update_all(default: false) if default
|
40
|
+
pay_payment_method = where(customer: pay_customer, processor_id: object.id).first_or_initialize
|
41
41
|
pay_payment_method.update!(attributes)
|
42
42
|
pay_payment_method
|
43
43
|
rescue ActiveRecord::RecordInvalid, ActiveRecord::RecordNotUnique
|
@@ -81,7 +81,7 @@ module Pay
|
|
81
81
|
end
|
82
82
|
|
83
83
|
# Update or create the subscription
|
84
|
-
pay_subscription =
|
84
|
+
pay_subscription = find_by(customer: pay_customer, processor_id: object.id)
|
85
85
|
if pay_subscription
|
86
86
|
# If pause behavior is changing to `void`, record the pause start date
|
87
87
|
# Any other pause status (or no pause at all) should have nil for start
|
@@ -95,7 +95,7 @@ module Pay
|
|
95
95
|
else
|
96
96
|
# Allow setting the subscription name in metadata, otherwise use the default
|
97
97
|
name ||= object.metadata["pay_name"] || Pay.default_product_name
|
98
|
-
pay_subscription =
|
98
|
+
pay_subscription = create!(attributes.merge(customer: pay_customer, name: name, processor_id: object.id))
|
99
99
|
end
|
100
100
|
|
101
101
|
# Cache the Stripe subscription on the Pay::Subscription that we return
|
@@ -8,8 +8,9 @@ module Pay
|
|
8
8
|
|
9
9
|
object = event.data.object
|
10
10
|
|
11
|
+
# Don't send email on incomplete Stripe subscriptions since they're just getting created and the JavaScript will handle SCA
|
11
12
|
pay_subscription = Pay::Subscription.find_by_processor_and_id(:stripe, object.subscription)
|
12
|
-
return if pay_subscription.nil?
|
13
|
+
return if pay_subscription.nil? || pay_subscription.status == "incomplete"
|
13
14
|
|
14
15
|
if Pay.send_email?(:payment_action_required, pay_subscription)
|
15
16
|
Pay.mailer.with(
|
@@ -8,8 +8,9 @@ module Pay
|
|
8
8
|
|
9
9
|
object = event.data.object
|
10
10
|
|
11
|
+
# Don't send email on incomplete Stripe subscriptions since they're just getting created and the JavaScript will handle SCA
|
11
12
|
pay_subscription = Pay::Subscription.find_by_processor_and_id(:stripe, object.subscription)
|
12
|
-
return if pay_subscription.nil?
|
13
|
+
return if pay_subscription.nil? || pay_subscription.status == "incomplete"
|
13
14
|
|
14
15
|
if Pay.send_email?(:payment_failed, pay_subscription)
|
15
16
|
Pay.mailer.with(
|
@@ -14,11 +14,13 @@ module Pay
|
|
14
14
|
# Stripe subscription items all have the same interval
|
15
15
|
price = event.data.object.lines.data.first.price
|
16
16
|
|
17
|
-
|
17
|
+
# For collection_method=send_invoice, Stripe will send an email and next_payment_attempt will be null
|
18
|
+
# https://docs.stripe.com/api/invoices/object#invoice_object-collection_method
|
19
|
+
if Pay.send_email?(:subscription_renewing, pay_subscription, price) && (next_payment_attempt = event.data.object.next_payment_attempt)
|
18
20
|
Pay.mailer.with(
|
19
21
|
pay_customer: pay_subscription.customer,
|
20
22
|
pay_subscription: pay_subscription,
|
21
|
-
date: Time.zone.at(
|
23
|
+
date: Time.zone.at(next_payment_attempt)
|
22
24
|
).subscription_renewing.deliver_later
|
23
25
|
end
|
24
26
|
end
|
data/lib/pay/stripe.rb
CHANGED
@@ -65,7 +65,7 @@ module Pay
|
|
65
65
|
|
66
66
|
def self.webhook_receive_test_events
|
67
67
|
value = find_value_by_name(:stripe, :webhook_receive_test_events)
|
68
|
-
value.blank?
|
68
|
+
value.blank? || ActiveModel::Type::Boolean.new.cast(value)
|
69
69
|
end
|
70
70
|
|
71
71
|
def self.configure_webhooks
|
data/lib/pay/version.rb
CHANGED
data/lib/pay.rb
CHANGED
@@ -132,13 +132,20 @@ module Pay
|
|
132
132
|
end
|
133
133
|
end
|
134
134
|
|
135
|
+
SYNC_HANDLERS = {
|
136
|
+
lemon_squeezy_order_id: ->(param) { Pay::LemonSqueezy.sync_order(param) },
|
137
|
+
paddle_billing_transaction_id: ->(param) { Pay::PaddleBilling.sync_transaction(param) },
|
138
|
+
stripe_checkout_session_id: ->(param) { Pay::Stripe.sync_checkout_session(param) },
|
139
|
+
transaction_id: ->(param) { Pay::PaddleBilling.sync_transaction(param) },
|
140
|
+
session_id: ->(param) { Pay::Stripe.sync_checkout_session(param) }
|
141
|
+
}
|
142
|
+
|
135
143
|
def self.sync(params)
|
136
|
-
|
137
|
-
|
138
|
-
|
139
|
-
|
140
|
-
elsif (order_id = params[:lemon_squeezy_order_id])
|
141
|
-
Pay::LemonSqueezy.sync_order(order_id)
|
144
|
+
SYNC_HANDLERS.each do |param_name, handler|
|
145
|
+
if (param = params[param_name])
|
146
|
+
return handler.call(param)
|
147
|
+
end
|
142
148
|
end
|
149
|
+
nil
|
143
150
|
end
|
144
151
|
end
|
data/lib/tasks/pay.rake
CHANGED
@@ -15,10 +15,10 @@ def sync_default_payment_method(pay_customer, retries: 2)
|
|
15
15
|
puts "Syncing Pay::Customer ##{pay_customer.id} attempt #{try + 1}: #{pay_customer.processor.titleize} #{pay_customer.processor_id}"
|
16
16
|
case pay_customer.processor
|
17
17
|
when "braintree"
|
18
|
-
payment_method = pay_customer.
|
18
|
+
payment_method = pay_customer.api_record.payment_methods.find(&:default?)
|
19
19
|
Pay::Braintree::PaymentMethod.sync(payment_method.token, object: payment_method) if payment_method
|
20
20
|
when "stripe"
|
21
|
-
payment_method_id = pay_customer.
|
21
|
+
payment_method_id = pay_customer.api_record.invoice_settings.default_payment_method
|
22
22
|
Pay::Stripe::PaymentMethod.sync(payment_method_id) if payment_method_id
|
23
23
|
when "paddle_classic"
|
24
24
|
Pay::PaddleClassic::PaymentMethod.sync(pay_customer: pay_customer)
|
metadata
CHANGED
@@ -1,16 +1,15 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: pay
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version:
|
4
|
+
version: 9.0.0
|
5
5
|
platform: ruby
|
6
|
-
original_platform: ''
|
7
6
|
authors:
|
8
7
|
- Jason Charnes
|
9
8
|
- Chris Oliver
|
10
9
|
- Collin Jilbert
|
11
10
|
bindir: bin
|
12
11
|
cert_chain: []
|
13
|
-
date:
|
12
|
+
date: 2025-03-27 00:00:00.000000000 Z
|
14
13
|
dependencies:
|
15
14
|
- !ruby/object:Gem::Dependency
|
16
15
|
name: rails
|
@@ -179,7 +178,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
|
|
179
178
|
- !ruby/object:Gem::Version
|
180
179
|
version: '0'
|
181
180
|
requirements: []
|
182
|
-
rubygems_version: 3.6.
|
181
|
+
rubygems_version: 3.6.6
|
183
182
|
specification_version: 4
|
184
183
|
summary: Payments engine for Ruby on Rails
|
185
184
|
test_files: []
|