pay 8.3.0 → 9.0.1

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: 8db3bdd142f76a9bd53d2a5e490c8209570c0b56e29a285c7d261bfab89a8fd8
4
- data.tar.gz: cb518172aa52336942cd612bdf3f4df5a6145109748cfa58b2d07b62361aab83
3
+ metadata.gz: 351dfe4e2001a3d892ba872ca305a44c49fde456d78fa1aaebf505fe603d3bc1
4
+ data.tar.gz: a9bdd0f9c7b81325bba64cf63e610a520867678a51c6ea1c30c29a43b47c90e4
5
5
  SHA512:
6
- metadata.gz: fd1d1207c6babf29d763320cced7aa7d595c49d55125c5655132c021a2c836cae85f1c51909ad3e2141545ec0d44fd9575db14ca3dc78dfb25c2a16fa5446606
7
- data.tar.gz: b0c7e0b96b6d7734fb967fd93b328e890c1085eb8c2b159b2ea6ccdd4093e073af39c0785fad9c75bcdfe918ee77a8028f4b9a931cdc72b3571f644844dfd449
6
+ metadata.gz: 9bf920b25c5c845e31aa639dfa725ae5e3beaa226892518bd6c025b7805cb70cb6e1c211e743204d9b5fea14b93c6506c5d58ec59725a189457b8db3dee97893
7
+ data.tar.gz: 1cdd1a4d26048b9683fc4a83df5f327c7ef4792b66b831ed0464dc9f035df9a8e72e35b8592cf19c103f56faa5b1777b2cee8b97e07f054a88b79b2bd177c3e0
data/README.md CHANGED
@@ -49,6 +49,7 @@ Want to add a new payment provider? Contributions are welcome.
49
49
  * [Paddle](docs/paddle_billing/1_overview.md)
50
50
  * [Lemon Squeezy](docs/lemon_squeezy/1_overview.md)
51
51
  * [Fake Processor](docs/fake_processor/1_overview.md)
52
+ * [Asaas (Community)](https://github.com/PedroAugustoRamalhoDuarte/pay-asaas)
52
53
  * **Marketplaces**
53
54
  * [Stripe Connect](docs/marketplaces/stripe_connect.md)
54
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
@@ -36,6 +36,7 @@ module Pay
36
36
  store_accessor :data, :total_tax_amounts # array of tax details for each jurisdiction
37
37
  store_accessor :data, :credit_notes # array of credit notes for the Stripe Invoice
38
38
  store_accessor :data, :refunds # array of refunds
39
+ store_accessor :data, :balance_transaction
39
40
 
40
41
  # Helpers for payment processors
41
42
  %w[braintree stripe paddle_billing paddle_classic lemon_squeezy fake_processor].each do |processor_name|
@@ -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
@@ -26,6 +26,7 @@ module Pay
26
26
  amount_captured: object.amount_captured,
27
27
  amount_refunded: object.amount_refunded,
28
28
  application_fee_amount: object.application_fee_amount,
29
+ balance_transaction: object.balance_transaction,
29
30
  bank: payment_method.try(:bank_name) || payment_method.try(:bank), # eps, fpx, ideal, p24, acss_debit, etc
30
31
  brand: payment_method.try(:brand)&.capitalize,
31
32
  created_at: Time.at(object.created),
@@ -81,13 +82,11 @@ module Pay
81
82
  end
82
83
 
83
84
  # 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
85
+ if (pay_charge = find_by(customer: pay_customer, processor_id: object.id))
86
+ pay_charge.with_lock { pay_charge.update!(attrs) }
88
87
  pay_charge
89
88
  else
90
- pay_customer.charges.create!(attrs.merge(processor_id: object.id))
89
+ create!(attrs.merge(customer: pay_customer, processor_id: object.id))
91
90
  end
92
91
  rescue ActiveRecord::RecordInvalid, ActiveRecord::RecordNotUnique
93
92
  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
@@ -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.3.0"
2
+ VERSION = "9.0.1"
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.3.0
4
+ version: 9.0.1
5
5
  platform: ruby
6
6
  authors:
7
7
  - Jason Charnes
8
8
  - Chris Oliver
9
9
  - Collin Jilbert
10
- autorequire:
11
10
  bindir: bin
12
11
  cert_chain: []
13
- date: 2024-12-20 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
@@ -165,7 +164,6 @@ homepage: https://github.com/pay-rails/pay
165
164
  licenses:
166
165
  - MIT
167
166
  metadata: {}
168
- post_install_message:
169
167
  rdoc_options: []
170
168
  require_paths:
171
169
  - lib
@@ -180,8 +178,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
180
178
  - !ruby/object:Gem::Version
181
179
  version: '0'
182
180
  requirements: []
183
- rubygems_version: 3.5.23
184
- signing_key:
181
+ rubygems_version: 3.6.6
185
182
  specification_version: 4
186
183
  summary: Payments engine for Ruby on Rails
187
184
  test_files: []