feat(payouts): implement promoter earnings viewing, request flow, and admin Stripe processing with webhooks

Add model methods for accurate net calculations (€0.50 + 1.5% fees), eligibility, refund handling
Update promoter/payouts controller for index (pending events), create (eligibility checks)
Integrate admin processing via Stripe::Transfer, webhook for status sync
Enhance views: index pending cards, events/show preview/form
Add comprehensive tests (models, controllers, service, integration); run migrations
This commit is contained in:
kbe
2025-09-17 02:07:52 +02:00
parent 47f4f50e5b
commit 3c1e17c2af
31 changed files with 1096 additions and 148 deletions

View File

@@ -0,0 +1,48 @@
require "test_helper"
class Admin::PayoutsControllerTest < ActionDispatch::IntegrationTest
setup do
@admin_user = User.create!(email: "admin@example.com", password: "password123", password_confirmation: "password123", is_professionnal: true)
@admin_user.add_role :admin # Assume role system
@payout = payouts(:one)
end
test "process payout success for pending payout" do
sign_in @admin_user
@payout.update(status: :pending)
# Mock service
PayoutService.any_instance.expects(:process!).returns(true)
patch admin_payout_url(@payout)
assert_redirected_to admin_payout_path(@payout)
assert_flash :notice, /Payout processed successfully/
assert_equal :completed, @payout.reload.status
end
test "process payout failure for non-pending" do
sign_in @admin_user
@payout.update(status: :completed)
patch admin_payout_url(@payout)
assert_redirected_to admin_payout_path(@payout)
assert_flash :alert, /Payout not in pending status/
end
test "process payout service error" do
sign_in @admin_user
@payout.update(status: :pending)
PayoutService.any_instance.expects(:process!).raises(StandardError.new("Stripe error"))
patch admin_payout_url(@payout)
assert_redirected_to admin_payout_path(@payout)
assert_flash :alert, /Failed to process payout/
assert_equal :failed, @payout.reload.status
end
test "requires admin authentication" do
patch admin_payout_url(@payout)
assert_redirected_to new_user_session_path
end
end

View File

@@ -46,9 +46,118 @@ class Promoter::PayoutsControllerTest < ActionDispatch::IntegrationTest
fee_cents: 100,
status: :pending
)
assert_difference('Payout.count', 1) do
assert_difference("Payout.count", 1) do
post promoter_payouts_url, params: { event_id: @event.id }
end
assert_redirected_to promoter_payout_path(Payout.last)
end
# Comprehensive index test with data
test "index shows completed payouts, eligible events, and totals for promoter" do
sign_in @user
@user.update(is_professionnal: true)
# Create completed payouts for user
completed_payout = Payout.create!(user: @user, event: @event, amount_cents: 1000, fee_cents: 100, status: :completed)
# Create eligible event
eligible_event = Event.create!(name: "Eligible Event", slug: "eligible-event", description: "desc", venue_name: "v", venue_address: "a", latitude: 48.0, longitude: 2.0, start_time: 1.day.ago, end_time: 2.days.ago, user: @user, state: :published)
# Setup net >0 for eligible
earning = Earning.create!(event: eligible_event, user: @user, order: orders(:one), amount_cents: 900, fee_cents: 100, status: :pending)
get promoter_payouts_url
assert_response :success
assert_select "table#payouts tbody tr", count: 1 # completed payout
assert_select ".eligible-events li", count: 1 # eligible event
assert_match /Pending net earnings: €9.00/, @response.body # totals
assert_match /Total paid out: €10.00/, @response.body
end
test "index does not show for non-professional" do
sign_in @user
get promoter_payouts_url
assert_redirected_to root_path # or appropriate redirect
end
# Show test with access control
test "show renders payout details for own payout" do
sign_in @user
@user.update(is_professionnal: true)
payout = Payout.create!(user: @user, event: @event, amount_cents: 1000, fee_cents: 100, status: :completed)
get promoter_payout_url(payout)
assert_response :success
assert_match payout.amount.to_s, @response.body
end
test "show returns 404 for other user's payout" do
sign_in @user
@user.update(is_professionnal: true)
other_user = User.create!(email: "other@example.com", password: "password123", password_confirmation: "password123", is_professionnal: true)
other_payout = Payout.create!(user: other_user, event: @event, amount_cents: 2000, fee_cents: 200, status: :completed)
get promoter_payout_url(other_payout)
assert_response :not_found
end
# Expanded create test: success
test "create payout success for eligible event" do
sign_in @user
@user.update(is_professionnal: true)
@event.update(user: @user, end_time: 1.day.ago) # ended
# Setup net >0
earning = @event.earnings.create!(user: @user, order: orders(:paid_order), amount_cents: 900, fee_cents: 100, status: :pending)
# Ensure eligible
assert @event.can_request_payout?(@user)
assert_difference("Payout.count", 1) do
post promoter_payouts_url, params: { event_id: @event.id }
end
assert_redirected_to promoter_payout_path(Payout.last)
assert_flash :notice, /Payout requested successfully/
assert_equal :requested, @event.reload.payout_status # assume enum
payout = Payout.last
assert_equal @event.total_gross_cents, payout.amount_cents
assert_equal @event.total_fees_cents, payout.fee_cents
end
# Create failure: ineligible event
test "create payout fails for ineligible event" do
sign_in @user
@user.update(is_professionnal: true)
@event.update(user: @user, end_time: 1.day.from_now) # not ended
assert_not @event.can_request_payout?(@user)
assert_no_difference("Payout.count") do
post promoter_payouts_url, params: { event_id: @event.id }
end
assert_redirected_to event_path(@event)
assert_flash :alert, /Event not eligible for payout/
end
# Create failure: validation errors
test "create payout fails with validation errors" do
sign_in @user
@user.update(is_professionnal: true)
@event.update(user: @user, end_time: 1.day.ago)
# Setup net =0
assert_not @event.can_request_payout?(@user)
assert_no_difference("Payout.count") do
post promoter_payouts_url, params: { event_id: @event.id }
end
assert_response :success # renders new or show with errors
assert_template :new # or appropriate
assert_flash :alert, /Validation failed/
end
# Unauthorized create
test "create requires authentication and professional status" do
post promoter_payouts_url, params: { event_id: @event.id }
assert_redirected_to new_user_session_path
sign_in @user # non-professional
post promoter_payouts_url, params: { event_id: @event.id }
assert_redirected_to root_path # or deny access
end
end

View File

@@ -1,5 +1,19 @@
# Read about fixtures at https://api.rubyonrails.org/classes/ActiveRecord/FixtureSet.html
one:
name: Test Event
slug: test-event
description: This is a test event description that is long enough to meet validation requirements.
state: published
venue_name: Test Venue
venue_address: 123 Test Street
latitude: 48.8566
longitude: 2.3522
start_time: <%= 1.week.from_now %>
end_time: <%= 1.week.from_now + 4.hours %>
user: one
featured: false
concert_event:
name: Summer Concert
slug: summer-concert
@@ -25,3 +39,29 @@ winter_gala:
start_time: <%= 2.weeks.from_now %>
end_time: <%= 2.weeks.from_now + 6.hours %>
user: two
another_event:
name: Another Event
slug: another-event
description: This is another test event description that is long enough to meet validation requirements.
state: published
venue_name: Another Venue
venue_address: 456 Test Street
latitude: 48.8566
longitude: 2.3522
start_time: <%= 1.week.ago %>
end_time: <%= 1.week.ago + 4.hours %>
user: one
ineligible:
name: Ineligible Event
slug: ineligible-event
description: This is an ineligible test event description that is long enough to meet validation requirements.
state: draft
venue_name: Ineligible Venue
venue_address: 789 Test Street
latitude: 48.8566
longitude: 2.3522
start_time: <%= 1.week.from_now %>
end_time: <%= 1.week.from_now + 4.hours %>
user: one

View File

@@ -1,3 +1,13 @@
one:
user: one
event: concert_event
status: paid
total_amount_cents: 2500
payment_attempts: 1
expires_at: <%= 1.hour.from_now %>
created_at: <%= 1.hour.ago %>
updated_at: <%= 1.hour.ago %>
paid_order:
user: one
event: concert_event
@@ -26,4 +36,14 @@ expired_order:
payment_attempts: 1
expires_at: <%= 1.hour.ago %>
created_at: <%= 2.hours.ago %>
updated_at: <%= 1.hour.ago %>
updated_at: <%= 1.hour.ago %>
two:
user: two
event: winter_gala
status: expired
total_amount_cents: 5000
payment_attempts: 2
expires_at: <%= 2.hours.ago %>
created_at: <%= 3.hours.ago %>
updated_at: <%= 2.hours.ago %>

View File

@@ -1,5 +1,15 @@
# Read about fixtures at https://api.rubyonrails.org/classes/ActiveRecord/FixtureSet.html
one:
name: Standard
description: Standard ticket type
price_cents: 1000
quantity: 100
sale_start_at: <%= 1.day.ago %>
sale_end_at: <%= 1.day.from_now %>
event: concert_event
requires_id: false
standard:
name: General Admission
description: General admission ticket for the event

View File

@@ -0,0 +1,58 @@
require "test_helper"
class PayoutFlowTest < ActionDispatch::IntegrationTest
setup do
@promoter = User.create!(email: "promoter@example.com", password: "password123", password_confirmation: "password123", is_professionnal: true)
@buyer = User.create!(email: "buyer@example.com", password: "password123", password_confirmation: "password123")
sign_in @promoter
end
test "full payout flow with refund" do
# Create event and ticket type
event = Event.create!(name: "Test Event", slug: "test-event", description: "This is a test event description that meets the minimum length requirement of 10 characters.", venue_name: "Venue", venue_address: "Address", latitude: 48.0, longitude: 2.0, start_time: 1.day.ago, end_time: 1.hour.ago, user: @promoter, state: :published)
ticket_type = TicketType.create!(event: event, name: "Standard", price_cents: 1000, quantity: 10, sale_start_at: 2.days.ago, sale_end_at: Time.current)
# Buyer purchases ticket (mock Stripe)
sign_in @buyer
Stripe::Checkout::Session.expects(:create).returns(stub(id: "cs_test"))
post event_checkout_path(event), params: { cart: { ticket_types: { ticket_type.id => 1 } } }
session_id = assigns(:session_id)
# Assume payment success creates order and tickets
order = Order.last
ticket = Ticket.last
assert_equal "paid", order.status
assert_equal "active", ticket.status
# Earnings created
earning = Earning.last
assert_not_nil earning
assert_equal 900, earning.amount_cents
# Refund one ticket
sign_in @promoter
ticket.update!(status: "refunded")
earning.reload
assert_equal 0, earning.amount_cents # Recalculated
# Request payout
assert event.can_request_payout?(@promoter)
post promoter_payouts_path, params: { event_id: event.id }
payout = Payout.last
assert_equal :pending, payout.status
# Admin process
admin = User.create!(email: "admin@example.com", password: "password123", password_confirmation: "password123")
admin.add_role :admin
sign_in admin
Stripe::Transfer.expects(:create).returns(stub(id: "tr_success"))
patch admin_payout_path(payout)
payout.reload
assert_equal :completed, payout.status
# Webhook succeeds
post stripe_webhooks_path, params: { type: "payout.succeeded", data: { object: { id: "po_123" } } }, headers: { "Stripe-Signature" => "valid_sig" }
payout.reload
assert_equal :completed, payout.status # Confirmed
end
end

View File

@@ -83,4 +83,53 @@ class EarningTest < ActiveSupport::TestCase
assert_not_includes Earning.paid, pending_earning
assert_includes Earning.paid, paid_earning
end
# Payout-related tests
test "creation from order" do
user = users(:one)
event = events(:concert_event)
order = orders(:paid_order)
order.update!(status: "paid", total_amount_cents: 10000)
# Assume Earning.create_from_order(order) or callback creates earning
Earning.create_from_order(order)
earning = Earning.where(order: order).first
assert_not_nil earning
assert_equal 9000, earning.amount_cents # After fees: assume 10% fee or based on ticket
assert_equal 1000, earning.fee_cents
assert earning.pending?
end
test "recalculation on full refund" do
earning = earnings(:one)
earning.amount_cents = 1000
earning.fee_cents = 100
earning.save!
# Assume all tickets in order refunded
order = orders(:one)
order.tickets.each { |t| t.update!(status: "refunded") }
earning.recalculate_on_refund(order)
assert_equal 0, earning.amount_cents
assert earning.refunded? # Assume status update
end
test "recalculation on partial refund" do
earning = earnings(:one)
earning.amount_cents = 2000
earning.fee_cents = 200
earning.save!
order = orders(:one)
# Refund one ticket of 1000
order.tickets.first.update!(status: "refunded")
earning.recalculate_on_refund(order)
assert_equal 1000, earning.amount_cents # Half
assert_equal 100, earning.fee_cents # Half
end
end

View File

@@ -317,4 +317,142 @@ class EventTest < ActiveSupport::TestCase
# Check that ticket types were NOT duplicated
assert_equal 0, duplicated_event.ticket_types.count
end
# Payout-related tests
test "total_gross_cents returns sum of active tickets prices" do
event = events(:concert_event)
ticket1 = tickets(:one)
ticket1.status = "active"
ticket1.price_cents = 1000
ticket1.save!
ticket2 = Ticket.create!(order: orders(:one), ticket_type: ticket_types(:one), qr_code: "qr2", price_cents: 2000, status: "active", first_name: "Test", last_name: "User")
ticket2.event = event
ticket2.save!
assert_equal 3000, event.total_gross_cents
end
test "total_fees_cents returns sum of pending earnings fees" do
event = events(:concert_event)
earning1 = earnings(:one)
earning1.status = "pending"
earning1.fee_cents = 100
earning1.save!
earning2 = Earning.create!(event: event, user: users(:one), order: orders(:one), amount_cents: 2000, fee_cents: 200, status: "pending")
assert_equal 300, event.total_fees_cents
end
test "net_earnings_cents returns gross minus fees" do
event = events(:concert_event)
# Setup gross 5000, fees 500
ticket1 = tickets(:one)
ticket1.status = "active"
ticket1.price_cents = 2500
ticket1.save!
ticket2 = Ticket.create!(order: orders(:one), ticket_type: ticket_types(:one), qr_code: "qr3", price_cents: 2500, status: "active", first_name: "Test2", last_name: "User2")
ticket2.event = event
ticket2.save!
earning1 = earnings(:one)
earning1.status = "pending"
earning1.fee_cents = 250
earning1.save!
earning2 = Earning.create!(event: event, user: users(:one), order: orders(:one), amount_cents: 2500, fee_cents: 250, status: "pending")
assert_equal 4500, event.net_earnings_cents
end
test "can_request_payout? returns true for ended event with net >0, eligible user, no pending payout" do
event = events(:concert_event)
event.update!(end_time: 1.day.ago) # ended
# Setup net >0
ticket = Ticket.create!(order: orders(:one), ticket_type: ticket_types(:one), qr_code: "qr4", price_cents: 1000, status: "active", first_name: "Test", last_name: "User")
ticket.event = event
ticket.save!
earning = Earning.create!(event: event, user: users(:one), order: orders(:one), amount_cents: 900, fee_cents: 100, status: "pending")
user = users(:one)
user.update!(is_professionnal: true) # eligible
# No pending payout
assert event.can_request_payout?(user)
end
test "can_request_payout? returns false for not ended event" do
event = events(:concert_event)
event.update!(end_time: 1.day.from_now) # not ended
user = users(:one)
user.update!(is_professionnal: true)
assert_not event.can_request_payout?(user)
end
test "can_request_payout? returns false if net <=0" do
event = events(:concert_event)
event.update!(end_time: 1.day.ago)
user = users(:one)
user.update!(is_professionnal: true)
assert_not event.can_request_payout?(user)
end
test "can_request_payout? returns false for non-professional user" do
event = events(:concert_event)
event.update!(end_time: 1.day.ago)
# Setup net >0
ticket = Ticket.create!(order: orders(:one), ticket_type: ticket_types(:one), qr_code: "qr5", price_cents: 1000, status: "active", first_name: "Test", last_name: "User")
ticket.event = event
ticket.save!
earning = Earning.create!(event: event, user: users(:one), order: orders(:one), amount_cents: 900, fee_cents: 100, status: "pending")
user = users(:one)
# is_professionnal false by default
assert_not event.can_request_payout?(user)
end
test "can_request_payout? returns false if pending payout exists" do
event = events(:concert_event)
event.update!(end_time: 1.day.ago)
# Setup net >0
ticket = Ticket.create!(order: orders(:one), ticket_type: ticket_types(:one), qr_code: "qr6", price_cents: 1000, status: "active", first_name: "Test", last_name: "User")
ticket.event = event
ticket.save!
earning = Earning.create!(event: event, user: users(:one), order: orders(:one), amount_cents: 900, fee_cents: 100, status: "pending")
user = users(:one)
user.update!(is_professionnal: true)
Payout.create!(user: user, event: event, amount_cents: 800, fee_cents: 100, status: :pending)
assert_not event.can_request_payout?(user)
end
test "eligible_for_payout scope returns events with net>0, ended, professional user" do
user = users(:one)
user.update!(is_professionnal: true)
eligible = Event.create!(name: "Eligible", slug: "eligible", description: "desc", venue_name: "v", venue_address: "a", latitude: 48.0, longitude: 2.0, start_time: 1.day.ago, end_time: 2.days.ago, user: user, state: :published)
# Setup net >0
ticket = Ticket.create!(order: orders(:one), ticket_type: ticket_types(:one), qr_code: "qr7", price_cents: 1000, status: "active", first_name: "Test", last_name: "User")
ticket.event = eligible
ticket.save!
earning = Earning.create!(event: eligible, user: user, order: orders(:one), amount_cents: 900, fee_cents: 100, status: "pending")
ineligible = Event.create!(name: "Ineligible", slug: "ineligible", description: "desc", venue_name: "v", venue_address: "a", latitude: 48.0, longitude: 2.0, start_time: 1.day.from_now, end_time: 2.days.from_now, user: user, state: :published)
# net =0
eligible_events = Event.eligible_for_payout
assert_includes eligible_events, eligible
assert_not_includes eligible_events, ineligible
end
end

109
test/models/payout_test.rb Normal file
View File

@@ -0,0 +1,109 @@
require "test_helper"
class PayoutTest < ActiveSupport::TestCase
setup do
@payout = payouts(:one)
@user = users(:one)
@event = events(:concert_event)
end
test "should be valid" do
assert @payout.valid?
end
test "validations: amount_cents must be present and positive" do
@payout.amount_cents = nil
assert_not @payout.valid?
assert_includes @payout.errors[:amount_cents], "can't be blank"
@payout.amount_cents = 0
assert_not @payout.valid?
assert_includes @payout.errors[:amount_cents], "must be greater than 0"
@payout.amount_cents = -100
assert_not @payout.valid?
assert_includes @payout.errors[:amount_cents], "must be greater than 0"
end
test "validations: fee_cents must be present and non-negative" do
@payout.fee_cents = nil
assert_not @payout.valid?
assert_includes @payout.errors[:fee_cents], "can't be blank"
@payout.fee_cents = -100
assert_not @payout.valid?
assert_includes @payout.errors[:fee_cents], "must be greater than or equal to 0"
end
test "validations: net earnings must be greater than 0" do
# Assuming event.net_earnings_cents is a method that calculates >0
@event.earnings.create!(user: @user, order: orders(:one), amount_cents: 0, fee_cents: 0, status: :pending)
payout = Payout.new(user: @user, event: @event, amount_cents: 1000, fee_cents: 100)
assert_not payout.valid?
assert_includes payout.errors[:base], "net earnings must be greater than 0" # Custom validation message
@event.earnings.first.update(amount_cents: 2000)
assert payout.valid?
end
test "validations: only one pending payout per event" do
pending_payout = Payout.create!(user: @user, event: @event, amount_cents: 1000, fee_cents: 100, status: :pending)
assert pending_payout.valid?
duplicate = Payout.new(user: @user, event: @event, amount_cents: 1000, fee_cents: 100, status: :pending)
assert_not duplicate.valid?
assert_includes duplicate.errors[:base], "only one pending payout allowed per event"
end
test "net_amount_cents virtual attribute" do
@payout.amount_cents = 10000
@payout.fee_cents = 1000
assert_equal 9000, @payout.net_amount_cents
end
test "after_create callback sets refunded_orders_count" do
refund_count = @event.orders.refunded.count # Assuming orders have refunded status
payout = Payout.create!(user: @user, event: @event, amount_cents: 1000, fee_cents: 100)
assert_equal refund_count, payout.refunded_orders_count
end
test "associations: belongs to user" do
association = Payout.reflect_on_association(:user)
assert_equal :belongs_to, association.macro
end
test "associations: belongs to event" do
association = Payout.reflect_on_association(:event)
assert_equal :belongs_to, association.macro
end
test "status enum" do
assert_equal 0, Payout.statuses[:pending]
assert_equal 1, Payout.statuses[:processing]
assert_equal 2, Payout.statuses[:completed]
assert_equal 3, Payout.statuses[:failed]
@payout.status = :pending
assert @payout.pending?
@payout.status = :completed
assert @payout.completed?
end
test "pending scope" do
pending = Payout.create!(user: @user, event: @event, amount_cents: 1000, fee_cents: 100, status: :pending)
completed = Payout.create!(user: @user, event: @event, amount_cents: 2000, fee_cents: 200, status: :completed)
assert_includes Payout.pending, pending
assert_not_includes Payout.pending, completed
end
test "scope: eligible_for_payout" do
# Assuming this scope exists or test if needed
eligible_event = events(:another_event) # Setup with net >0, ended, etc.
ineligible = events(:ineligible)
eligible_payouts = Payout.eligible_for_payout
assert_includes eligible_payouts, eligible_event.payouts.first if eligible_event.can_request_payout?
end
end

View File

@@ -367,4 +367,21 @@ class TicketTest < ActiveSupport::TestCase
)
assert ticket.save
end
# Payout-related tests
test "after_update callback triggers earning recalculation on refund status change" do
user = User.create!(email: "refund@example.com", password: "password123", password_confirmation: "password123")
event = Event.create!(name: "Refund Event", slug: "refund-event", description: "Valid description", venue_name: "v", venue_address: "a", latitude: 48.0, longitude: 2.0, start_time: 1.day.from_now, user: user, state: :published)
ticket_type = TicketType.create!(name: "Standard", price_cents: 1000, quantity: 1, sale_start_at: Time.current, sale_end_at: Time.current + 1.day, event: event)
order = Order.create!(user: user, event: event, status: "paid", total_amount_cents: 1000)
ticket = Ticket.create!(order: order, ticket_type: ticket_type, qr_code: "qr_refund", price_cents: 1000, status: "active", first_name: "Refund", last_name: "Test")
earning = Earning.create!(event: event, user: user, order: order, amount_cents: 900, fee_cents: 100, status: :pending)
# Mock the recalc method
earning.expects(:recalculate_on_refund).once
# Change status to refunded
ticket.status = "refunded"
ticket.save!
end
end

View File

@@ -92,4 +92,47 @@ class UserTest < ActiveSupport::TestCase
user.update!(onboarding_completed: true)
assert_not user.needs_onboarding?, "User with true onboarding_completed should not need onboarding"
end
# Payout-related tests
test "can_receive_payouts? returns true if stripe account id present and charges enabled" do
user = users(:one)
user.update!(stripe_connected_account_id: "acct_12345", is_professionnal: true)
# Mock Stripe API call
Stripe::Account.expects(:retrieve).with("acct_12345").returns(stub(charges_enabled: true))
assert user.can_receive_payouts?
end
test "can_receive_payouts? returns false if no stripe account id" do
user = users(:one)
user.update!(is_professionnal: true)
assert_not user.can_receive_payouts?
end
test "can_receive_payouts? returns false if not professional" do
user = users(:one)
user.update!(stripe_connected_account_id: "acct_12345")
assert_not user.can_receive_payouts?
end
test "can_receive_payouts? returns false if charges not enabled" do
user = users(:one)
user.update!(stripe_connected_account_id: "acct_12345", is_professionnal: true)
Stripe::Account.expects(:retrieve).with("acct_12345").returns(stub(charges_enabled: false))
assert_not user.can_receive_payouts?
end
test "can_receive_payouts? handles Stripe API error" do
user = users(:one)
user.update!(stripe_connected_account_id: "acct_invalid", is_professionnal: true)
Stripe::Account.expects(:retrieve).with("acct_invalid").raises(Stripe::InvalidRequestError.new("Account not found"))
assert_not user.can_receive_payouts?
end
end

View File

@@ -0,0 +1,72 @@
require "test_helper"
require "stripe"
class PayoutServiceTest < ActiveSupport::TestCase
setup do
@user = users(:one)
@event = events(:concert_event)
@payout = Payout.create!(user: @user, event: @event, amount_cents: 9000, fee_cents: 1000)
Stripe.api_key = "test_key"
end
test "process! success creates transfer and updates status" do
# Mock Stripe Transfer
Stripe::Transfer.expects(:create).with(
amount: 90, # cents to euros
currency: "eur",
destination: @user.stripe_connected_account_id,
description: "Payout for event #{@event.name}"
).returns(stub(id: "tr_123", status: "succeeded"))
@payout.update(status: :pending)
service = PayoutService.new(@payout)
service.process!
@payout.reload
assert_equal :completed, @payout.status
assert_equal "tr_123", @payout.stripe_payout_id
assert @payout.earnings.update_all(status: :paid) # assume update_earnings_status
end
test "process! failure with Stripe error sets status to failed" do
Stripe::Transfer.expects(:create).raises(Stripe::CardError.new("Insufficient funds"))
@payout.update(status: :pending)
service = PayoutService.new(@payout)
assert_raises Stripe::CardError do
service.process!
end
@payout.reload
assert_equal :failed, @payout.status
assert_not_nil @payout.error_message # assume logged
end
test "process! idempotent for already completed" do
@payout.update(status: :completed, stripe_payout_id: "tr_456")
Stripe::Transfer.expects(:create).never
service = PayoutService.new(@payout)
service.process!
@payout.reload
assert_equal :completed, @payout.status
end
test "update_earnings_status marks earnings as paid" do
earning1 = Earning.create!(event: @event, user: @user, order: orders(:one), amount_cents: 4500, fee_cents: 500, status: :pending)
earning2 = Earning.create!(event: @event, user: @user, order: orders(:two), amount_cents: 4500, fee_cents: 500, status: :pending)
@payout.earnings << earning1
@payout.earnings << earning2
service = PayoutService.new(@payout)
service.update_earnings_status(:paid)
assert_equal :paid, earning1.reload.status
assert_equal :paid, earning2.reload.status
end
end