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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: fabadba6082c91114d763242a738a19dab355dde46f6079a4ae474b9ba64c538
4
- data.tar.gz: 4348766c53d5630613dacf04a61d1f49714076f91de1e0594625cf088072b61f
3
+ metadata.gz: 6e2755f5ff01aedf32875c2ee941d4e92efe0c044ed7b6eeb601e6d0e17d1d51
4
+ data.tar.gz: ec10df5f7c8a4d52a55f8fc533fe42bc33a773dc226a348dfee0b80be37c1536
5
5
  SHA512:
6
- metadata.gz: 8886673e5e4a1ee8d478c9214861e1b8ba2032d3f54b50664ece5cebd7d1545ef8945f167d87455ffaf739a6b758b016d297069889bb7b47efaf47028928e09a
7
- data.tar.gz: b91bd40b0de1d41b191faa0f2fccacb3de48fc2b3fa83698df75897b93de065b11a132d9958e1f31572d2cf4c57e372aa2d7033844226c5f71eeef8dfe49f68e
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
- [![Build Status](https://github.com/pay-rails/pay/workflows/Tests/badge.svg)](https://github.com/pay-rails/pay/actions) [![Gem Version](https://badge.fury.io/rb/pay.svg)](https://badge.fury.io/rb/pay)
6
-
7
- <img src="docs/images/stripe_partner_badge.svg" height="26px">
5
+ [![Build Status](https://github.com/pay-rails/pay/workflows/Tests/badge.svg)](https://github.com/pay-rails/pay/actions) [![Gem Version](https://badge.fury.io/rb/pay.svg)](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
- ⚠️ **Upgrading?** Check the [UPGRADE](UPGRADE.md) guide for required changes and/or migration when upgrading from a previous version of Pay.
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 = charges.find_or_initialize_by(processor_id: transaction.id)
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(processor: :braintree, processor_id: object.customer_id)
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 = pay_customer.subscriptions.find_by(processor_id: object.id)
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
- pay_subscription = pay_customer.subscriptions.create!(attributes.merge(name: name, processor_id: object.id))
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 (mimicing Stripe)
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 = pay_customer.charges.find_by(processor_id: processor_id))
26
- pay_charge.with_lock do
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
- pay_customer.charges.create!(attributes.merge(processor_id: processor_id))
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 = pay_customer.subscriptions.find_by(processor_id: object.id))
37
- pay_subscription.with_lock do
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
- pay_customer.subscriptions.create!(attributes.merge(name: name, processor_id: object.id))
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 = pay_customer.charges.find_by(processor_id: object.id))
52
- pay_charge.with_lock do
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
- pay_customer.charges.create!(attrs.merge(processor_id: object.id))
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 = pay_customer.subscriptions.find_by(processor_id: subscription_id))
58
- pay_subscription.with_lock do
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
- pay_customer.subscriptions.create!(attributes.merge(name: name, processor_id: subscription_id))
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 = pay_customer.charges.find_by(processor_id: object.id))
85
- pay_charge.with_lock do
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
- pay_customer.charges.create!(attrs.merge(processor_id: object.id))
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
- defaults = {
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.payment_methods.update_all(default: false) if default
40
- pay_payment_method = pay_customer.payment_methods.where(processor_id: object.id).first_or_initialize
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 = pay_customer.subscriptions.find_by(processor_id: object.id)
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 = pay_customer.subscriptions.create!(attributes.merge(name: name, processor_id: object.id))
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
- if Pay.send_email?(:subscription_renewing, pay_subscription, price)
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(event.data.object.next_payment_attempt)
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? ? true : ActiveModel::Type::Boolean.new.cast(value)
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
@@ -1,3 +1,3 @@
1
1
  module Pay
2
- VERSION = "8.2.2"
2
+ VERSION = "9.0.0"
3
3
  end
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
- if (session_id = params[:stripe_checkout_session_id] || params[:session_id])
137
- Pay::Stripe.sync_checkout_session(session_id)
138
- elsif (transaction_id = params[:paddle_billing_transaction_id] || params[:transaction_id])
139
- Pay::PaddleBilling.sync_transaction(transaction_id)
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.customer.payment_methods.find(&:default?)
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.customer.invoice_settings.default_payment_method
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: 8.2.2
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: 2024-12-13 00:00:00.000000000 Z
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.0.dev
181
+ rubygems_version: 3.6.6
183
182
  specification_version: 4
184
183
  summary: Payments engine for Ruby on Rails
185
184
  test_files: []