Files
sure/test/controllers/goal_pledges_controller_test.rb
Guillem Arias ad101f619a fix(goals): test pledge new across both turbo-frame and full-page paths
CI failure on the prior commit: `GoalPledgesControllerTest#
test_new_renders_the_pledge_form` expected 200 but got a 302 to
the goal show page. The recently-added non-frame guard on
`GoalPledgesController#new` redirects direct GETs (F5, bookmark)
back to the goal so the dialog doesn't render standalone, and the
test wasn't sending the `Turbo-Frame` header that the modal flow
uses in production.

Split the test into the two paths the controller actually serves:

- `new renders the pledge form inside a turbo frame` passes a
  `Turbo-Frame: modal` header and asserts 200 — the real modal
  flow.
- `new redirects to the goal show page on a non-frame GET` asserts
  the 302 to `goal_path(@goal)` — the guard's intended branch.

Together they cover the controller's actual contract.
2026-05-14 22:54:05 +02:00

84 lines
2.7 KiB
Ruby

require "test_helper"
class GoalPledgesControllerTest < ActionDispatch::IntegrationTest
setup do
sign_in users(:family_admin)
@goal = goals(:vacation_italy)
@account = accounts(:depository)
@pledge = goal_pledges(:open_transfer)
ensure_tailwind_build
end
test "new renders the pledge form inside a turbo frame" do
get new_goal_pledge_url(@goal), headers: { "Turbo-Frame" => "modal" }
assert_response :success
end
test "new redirects to the goal show page on a non-frame GET" do
get new_goal_pledge_url(@goal)
assert_redirected_to goal_path(@goal)
end
test "create opens a pledge with default kind" do
assert_difference -> { GoalPledge.count } => 1 do
post goal_pledges_url(@goal), params: {
goal_pledge: {
amount: "150",
account_id: @account.id
}
}
end
pledge = GoalPledge.order(created_at: :desc).first
assert_equal "open", pledge.status
assert_equal @goal.id, pledge.goal_id
assert_redirected_to goal_path(@goal)
end
test "create rejects amount <= 0" do
assert_no_difference "GoalPledge.count" do
post goal_pledges_url(@goal), params: {
goal_pledge: { amount: "0", account_id: @account.id }
}
end
assert_response :unprocessable_entity
end
test "extend pushes expires_at forward" do
before = @pledge.expires_at
patch renew_goal_pledge_url(@goal, @pledge)
assert_redirected_to goal_path(@goal)
assert @pledge.reload.expires_at > before
end
test "extend on non-open pledge flashes alert" do
pledge = goal_pledges(:matched_transfer)
patch renew_goal_pledge_url(@goal, pledge)
assert_redirected_to goal_path(@goal)
assert flash[:alert].present?
end
test "destroy cancels an open pledge" do
delete goal_pledge_url(@goal, @pledge)
assert_redirected_to goal_path(@goal)
assert @pledge.reload.status_cancelled?
end
test "destroy on non-open pledge flashes alert" do
pledge = goal_pledges(:matched_transfer)
delete goal_pledge_url(@goal, pledge)
assert_redirected_to goal_path(@goal)
assert flash[:alert].present?
end
test "another family's goal returns redirect" do
other_family = Family.create!(name: "Other", currency: "USD", locale: "en", country: "US", timezone: "UTC")
other_account = Account.create!(family: other_family, accountable: Depository.new, name: "Foreign", currency: "USD", balance: 100)
other_goal = other_family.goals.new(name: "Foreign goal", target_amount: 100, currency: "USD")
other_goal.goal_accounts.build(account: other_account)
other_goal.save!
get new_goal_pledge_url(other_goal)
assert_redirected_to goals_path
end
end