# frozen_string_literal: true require 'test_helper' module Reservations; end class Reservations::CreateTest < ActionDispatch::IntegrationTest setup do @user_without_subscription = User.members.without_subscription.first @user_with_subscription = User.members.with_subscription.second end test 'user without subscription reserves a machine with success' do login_as(@user_without_subscription, scope: :user) machine = Machine.find(6) availability = machine.availabilities.first reservations_count = Reservation.count invoice_count = Invoice.count invoice_items_count = InvoiceItem.count users_credit_count = UsersCredit.count subscriptions_count = Subscription.count VCR.use_cassette('reservations_create_for_machine_without_subscription_success') do post '/api/stripe/confirm_payment', params: { payment_method_id: stripe_payment_method, cart_items: { items: [ { reservation: { reservable_id: machine.id, reservable_type: machine.class.name, slots_reservations_attributes: [ { slot_id: availability.slots.first.id } ] } } ] } }.to_json, headers: default_headers end # general assertions assert_equal 201, response.status assert_equal reservations_count + 1, Reservation.count assert_equal invoice_count + 1, Invoice.count assert_equal invoice_items_count + 1, InvoiceItem.count assert_equal users_credit_count, UsersCredit.count assert_equal subscriptions_count, Subscription.count # subscription assertions assert_equal 0, @user_without_subscription.subscriptions.count assert_nil @user_without_subscription.subscribed_plan # reservation assertions reservation = Reservation.last assert reservation.original_invoice assert_equal 1, reservation.original_invoice.invoice_items.count # invoice_items assertions invoice_item = InvoiceItem.last assert_equal machine.prices.find_by(group_id: @user_without_subscription.group_id, plan_id: nil).amount, invoice_item.amount assert invoice_item.check_footprint # invoice assertions item = InvoiceItem.find_by(object: reservation) invoice = item.invoice assert_invoice_pdf invoice assert_not_nil invoice.debug_footprint refute invoice.payment_gateway_object.blank? refute invoice.total.blank? assert invoice.check_footprint # notification assert_not_empty Notification.where(attached_object: reservation) end test 'user without subscription reserves a machine with error' do login_as(@user_without_subscription, scope: :user) machine = Machine.find(6) availability = machine.availabilities.first reservations_count = Reservation.count invoice_count = Invoice.count invoice_items_count = InvoiceItem.count notifications_count = Notification.count VCR.use_cassette('reservations_create_for_machine_without_subscription_error') do post '/api/stripe/confirm_payment', params: { payment_method_id: stripe_payment_method(error: :card_declined), cart_items: { items: [ { reservation: { reservable_id: machine.id, reservable_type: machine.class.name, slots_reservations_attributes: [ { slot_id: availability.slots.first.id } ] } } ] } }.to_json, headers: default_headers end # Check response format & status assert_equal 200, response.status, "API does not return the expected status. #{response.body}" assert_equal Mime[:json], response.content_type # Check the error was handled assert_match /Your card was declined/, response.body # Check the subscription wasn't taken assert_equal reservations_count, Reservation.count assert_equal invoice_count, Invoice.count assert_equal invoice_items_count, InvoiceItem.count assert_equal notifications_count, Notification.count # subscription assertions assert_equal 0, @user_without_subscription.subscriptions.count assert_nil @user_without_subscription.subscribed_plan end test 'user without subscription reserves a training with success' do login_as(@user_without_subscription, scope: :user) training = Training.first availability = training.availabilities.first reservations_count = Reservation.count invoice_count = Invoice.count invoice_items_count = InvoiceItem.count VCR.use_cassette('reservations_create_for_training_without_subscription_success') do post '/api/stripe/confirm_payment', params: { payment_method_id: stripe_payment_method, cart_items: { items: [ { reservation: { reservable_id: training.id, reservable_type: training.class.name, slots_reservations_attributes: [ { slot_id: availability.slots.first.id } ] } } ] } }.to_json, headers: default_headers end # general assertions assert_equal 201, response.status assert_equal reservations_count + 1, Reservation.count assert_equal invoice_count + 1, Invoice.count assert_equal invoice_items_count + 1, InvoiceItem.count # subscription assertions assert_equal 0, @user_without_subscription.subscriptions.count assert_nil @user_without_subscription.subscribed_plan # reservation assertions reservation = Reservation.last assert reservation.original_invoice assert_equal 1, reservation.original_invoice.invoice_items.count # invoice_items invoice_item = InvoiceItem.last assert_equal invoice_item.amount, training.amount_by_group(@user_without_subscription.group_id).amount assert invoice_item.check_footprint # invoice assertions item = InvoiceItem.find_by(object: reservation) invoice = item.invoice assert_invoice_pdf invoice assert_not_nil invoice.debug_footprint refute invoice.payment_gateway_object.blank? refute invoice.total.blank? assert invoice.check_footprint # notification assert_not_empty Notification.where(attached_object: reservation) end test 'user with subscription reserves a machine with success' do login_as(@user_with_subscription, scope: :user) plan = @user_with_subscription.subscribed_plan machine = Machine.find(6) availability = machine.availabilities.first reservations_count = Reservation.count invoice_count = Invoice.count invoice_items_count = InvoiceItem.count users_credit_count = UsersCredit.count VCR.use_cassette('reservations_create_for_machine_with_subscription_success') do post '/api/stripe/confirm_payment', params: { payment_method_id: stripe_payment_method, cart_items: { items: [ { reservation: { reservable_id: machine.id, reservable_type: machine.class.name, slots_reservations_attributes: [ { slot_id: availability.slots.first.id }, { slot_id: availability.slots.last.id } ] } } ] } }.to_json, headers: default_headers end # general assertions assert_equal 201, response.status assert_equal reservations_count + 1, Reservation.count assert_equal invoice_count + 1, Invoice.count assert_equal invoice_items_count + 2, InvoiceItem.count assert_equal users_credit_count + 1, UsersCredit.count # subscription assertions assert_equal 1, @user_with_subscription.subscriptions.count assert_not_nil @user_with_subscription.subscribed_plan assert_equal plan.id, @user_with_subscription.subscribed_plan.id # reservation assertions reservation = Reservation.last assert reservation.original_invoice assert_equal 2, reservation.original_invoice.invoice_items.count # invoice_items assertions invoice_items = InvoiceItem.last(2) machine_price = machine.prices.find_by(group_id: @user_with_subscription.group_id, plan_id: plan.id).amount assert(invoice_items.any? { |inv| inv.amount.zero? }) assert(invoice_items.any? { |inv| inv.amount == machine_price }) assert(invoice_items.all?(&:check_footprint)) # users_credits assertions users_credit = UsersCredit.last assert_equal @user_with_subscription, users_credit.user assert_equal [reservation.slots.count, plan.machine_credits.find_by(creditable_id: machine.id).hours].min, users_credit.hours_used # invoice assertions item = InvoiceItem.find_by(object: reservation) invoice = item.invoice assert_invoice_pdf invoice assert_not_nil invoice.debug_footprint refute invoice.payment_gateway_object.blank? refute invoice.total.blank? assert invoice.check_footprint # notification assert_not_empty Notification.where(attached_object: reservation) end test 'user with subscription reserves the FIRST training with success' do login_as(@user_with_subscription, scope: :user) plan = @user_with_subscription.subscribed_plan plan.update!(is_rolling: true) training = Training.joins(credits: :plan).where(credits: { plan: plan }).first availability = training.availabilities.first reservations_count = Reservation.count invoice_count = Invoice.count invoice_items_count = InvoiceItem.count VCR.use_cassette('reservations_create_for_training_with_subscription_success') do post '/api/local_payment/confirm_payment', params: { items: [ { reservation: { reservable_id: training.id, reservable_type: training.class.name, slots_reservations_attributes: [ { slot_id: availability.slots.first.id } ] } } ] }.to_json, headers: default_headers end # general assertions assert_equal 201, response.status assert_equal reservations_count + 1, Reservation.count assert_equal invoice_count + 1, Invoice.count assert_equal invoice_items_count + 1, InvoiceItem.count # subscription assertions assert_equal 1, @user_with_subscription.subscriptions.count assert_not_nil @user_with_subscription.subscribed_plan assert_equal plan.id, @user_with_subscription.subscribed_plan.id # reservation assertions reservation = Reservation.last assert reservation.original_invoice assert_equal 1, reservation.original_invoice.invoice_items.count # invoice_items invoice_item = InvoiceItem.last assert_equal 0, invoice_item.amount # amount is 0 because this training is a credited training with that plan assert invoice_item.check_footprint # invoice assertions item = InvoiceItem.find_by(object: reservation) invoice = item.invoice assert_invoice_pdf invoice assert_not_nil invoice.debug_footprint assert invoice.payment_gateway_object.blank? refute invoice.total.blank? assert invoice.check_footprint # notification assert_not_empty Notification.where(attached_object: reservation) # check that user subscription were extended assert_equal reservation.slots.first.start_at + plan.duration, @user_with_subscription.subscription.expired_at end test 'user reserves a machine and pay by wallet with success' do @vlonchamp = User.find_by(username: 'vlonchamp') login_as(@vlonchamp, scope: :user) machine = Machine.find(6) availability = machine.availabilities.first reservations_count = Reservation.count invoice_count = Invoice.count invoice_items_count = InvoiceItem.count users_credit_count = UsersCredit.count wallet_transactions_count = WalletTransaction.count VCR.use_cassette('reservations_create_for_machine_and_pay_wallet_success') do post '/api/stripe/confirm_payment', params: { payment_method_id: stripe_payment_method, cart_items: { customer_id: @vlonchamp.id, items: [ { reservation: { reservable_id: machine.id, reservable_type: machine.class.name, slots_reservations_attributes: [ { slot_id: availability.slots.first.id } ] } } ] } }.to_json, headers: default_headers end @vlonchamp.wallet.reload # general assertions assert_equal 201, response.status assert_equal reservations_count + 1, Reservation.count assert_equal invoice_count + 1, Invoice.count assert_equal invoice_items_count + 1, InvoiceItem.count assert_equal users_credit_count, UsersCredit.count assert_equal wallet_transactions_count + 1, WalletTransaction.count # subscription assertions assert_equal 0, @vlonchamp.subscriptions.count assert_nil @vlonchamp.subscribed_plan # reservation assertions reservation = Reservation.last assert reservation.original_invoice assert_equal 1, reservation.original_invoice.invoice_items.count # invoice_items assertions invoice_item = InvoiceItem.last assert_equal machine.prices.find_by(group_id: @vlonchamp.group_id, plan_id: nil).amount, invoice_item.amount assert invoice_item.check_footprint # invoice assertions item = InvoiceItem.find_by(object: reservation) invoice = item.invoice assert_invoice_pdf invoice assert_not_nil invoice.debug_footprint refute invoice.payment_gateway_object.blank? refute invoice.total.blank? assert invoice.check_footprint # notification assert_not_empty Notification.where(attached_object: reservation) # wallet assert_equal 0, @vlonchamp.wallet.amount assert_equal 2, @vlonchamp.wallet.wallet_transactions.count transaction = @vlonchamp.wallet.wallet_transactions.last assert_equal 'debit', transaction.transaction_type assert_equal 10, transaction.amount assert_equal invoice.wallet_amount / 100.0, transaction.amount end test 'user reserves a training and a subscription by wallet with success' do @vlonchamp = User.find_by(username: 'vlonchamp') login_as(@vlonchamp, scope: :user) training = Training.first availability = training.availabilities.first plan = Plan.find_by(group_id: @vlonchamp.group.id, type: 'Plan', base_name: 'Mensuel tarif réduit') reservations_count = Reservation.count invoice_count = Invoice.count invoice_items_count = InvoiceItem.count wallet_transactions_count = WalletTransaction.count VCR.use_cassette('reservations_create_for_training_and_plan_by_pay_wallet_success') do post '/api/stripe/confirm_payment', params: { payment_method_id: stripe_payment_method, cart_items: { items: [ { reservation: { reservable_id: training.id, reservable_type: training.class.name, slots_reservations_attributes: [ { slot_id: availability.slots.first.id } ] } }, { subscription: { plan_id: plan.id } } ] } }.to_json, headers: default_headers end @vlonchamp.wallet.reload # general assertions assert_equal 201, response.status assert_equal reservations_count + 1, Reservation.count assert_equal invoice_count + 1, Invoice.count assert_equal invoice_items_count + 2, InvoiceItem.count assert_equal wallet_transactions_count + 1, WalletTransaction.count # subscription assertions assert_equal 1, @vlonchamp.subscriptions.count assert_not_nil @vlonchamp.subscribed_plan assert_equal plan.id, @vlonchamp.subscribed_plan.id # reservation assertions reservation = Reservation.last assert reservation.original_invoice assert_equal 2, reservation.original_invoice.invoice_items.count # invoice assertions item = InvoiceItem.find_by(object: reservation) invoice = item.invoice assert_invoice_pdf invoice assert_not_nil invoice.debug_footprint refute invoice.payment_gateway_object.blank? refute invoice.total.blank? assert_equal invoice.total, 2000 assert invoice.check_footprint # notification assert_not_empty Notification.where(attached_object: reservation) # wallet assert_equal 0, @vlonchamp.wallet.amount assert_equal 2, @vlonchamp.wallet.wallet_transactions.count transaction = @vlonchamp.wallet.wallet_transactions.last assert_equal 'debit', transaction.transaction_type assert_equal 10, transaction.amount assert_equal invoice.wallet_amount / 100.0, transaction.amount end test 'user reserves a machine and a subscription using a coupon with success' do login_as(@user_without_subscription, scope: :user) machine = Machine.find(6) plan = Plan.find(4) availability = machine.availabilities.first reservations_count = Reservation.count invoice_count = Invoice.count invoice_items_count = InvoiceItem.count subscriptions_count = Subscription.count users_credit_count = UsersCredit.count VCR.use_cassette('reservations_machine_and_plan_using_coupon_success') do post '/api/stripe/confirm_payment', params: { payment_method_id: stripe_payment_method, cart_items: { items: [ { reservation: { reservable_id: machine.id, reservable_type: machine.class.name, slots_reservations_attributes: [ { slot_id: availability.slots.first.id } ] } }, { subscription: { plan_id: plan.id } } ], coupon_code: 'SUNNYFABLAB' } }.to_json, headers: default_headers end # general assertions assert_equal 201, response.status assert_equal reservations_count + 1, Reservation.count assert_equal invoice_count + 1, Invoice.count assert_equal invoice_items_count + 2, InvoiceItem.count assert_equal users_credit_count, UsersCredit.count assert_equal subscriptions_count + 1, Subscription.count # subscription assertions assert_equal 1, @user_without_subscription.subscriptions.count assert_not_nil @user_without_subscription.subscribed_plan assert_equal plan.id, @user_without_subscription.subscribed_plan.id # reservation assertions reservation = Reservation.last assert reservation.original_invoice assert_equal 2, reservation.original_invoice.invoice_items.count # invoice assertions item = InvoiceItem.find_by(object: reservation) invoice = item.invoice assert_invoice_pdf invoice assert_not_nil invoice.debug_footprint refute invoice.payment_gateway_object.blank? refute invoice.total.blank? assert invoice.check_footprint # invoice_items assertions ## reservation reservation_item = invoice.invoice_items.find_by(object: reservation) assert_not_nil reservation_item assert_equal reservation_item.amount, machine.prices.find_by(group_id: @user_without_subscription.group_id, plan_id: plan.id).amount assert reservation_item.check_footprint ## subscription subscription_item = invoice.invoice_items.find_by(object_type: Subscription.name) assert_not_nil subscription_item subscription = subscription_item.object assert_equal subscription_item.amount, plan.amount assert_equal subscription.plan_id, plan.id assert subscription_item.check_footprint VCR.use_cassette('reservations_machine_and_plan_using_coupon_retrieve_invoice_from_stripe') do stp_intent = invoice.payment_gateway_object.gateway_object.retrieve assert_equal stp_intent.amount, invoice.total end # notifications assert_not_empty Notification.where(attached_object: reservation) assert_not_empty Notification.where(attached_object: subscription) end test 'user reserves a training with an expired coupon with error' do login_as(@user_without_subscription, scope: :user) training = Training.find(1) availability = training.availabilities.first reservations_count = Reservation.count invoice_count = Invoice.count invoice_items_count = InvoiceItem.count notifications_count = Notification.count VCR.use_cassette('reservations_training_with_expired_coupon_error') do post '/api/stripe/confirm_payment', params: { payment_method_id: stripe_payment_method, cart_items: { customer_id: @user_without_subscription.id, items: [ { reservation: { reservable_id: training.id, reservable_type: training.class.name, card_token: stripe_payment_method, slots_reservations_attributes: [ { slot_id: availability.slots.first.id } ] } } ], coupon_code: 'XMAS10' } }.to_json, headers: default_headers end # general assertions assert_equal 422, response.status assert_equal reservations_count, Reservation.count assert_equal invoice_count, Invoice.count assert_equal invoice_items_count, InvoiceItem.count assert_equal notifications_count, Notification.count # subscription assertions assert_equal 0, @user_without_subscription.subscriptions.count assert_nil @user_without_subscription.subscribed_plan end test 'user reserves a training and a subscription with payment schedule' do login_as(@user_without_subscription, scope: :user) reservations_count = Reservation.count invoice_count = Invoice.count invoice_items_count = InvoiceItem.count subscriptions_count = Subscription.count users_credit_count = UsersCredit.count payment_schedule_count = PaymentSchedule.count payment_schedule_items_count = PaymentScheduleItem.count training = Training.find(1) availability = training.availabilities.first plan = Plan.find_by(group_id: @user_without_subscription.group.id, type: 'Plan', base_name: 'Abonnement mensualisable') VCR.use_cassette('reservations_training_subscription_with_payment_schedule') do post '/api/stripe/setup_subscription', params: { payment_method_id: stripe_payment_method, cart_items: { items: [ { reservation: { reservable_id: training.id, reservable_type: training.class.name, slots_reservations_attributes: [ { slot_id: availability.slots.first.id } ] } }, { subscription: { plan_id: plan.id } } ], payment_schedule: true, payment_method: 'cart' } }.to_json, headers: default_headers # Check response format & status assert_equal 201, response.status, response.body assert_equal Mime[:json], response.content_type # Check the response sub = json_response(response.body) assert_not_nil sub[:id] end # Check response format & status assert_equal 201, response.status, response.body assert_equal Mime[:json], response.content_type assert_equal reservations_count + 1, Reservation.count, 'missing the reservation' assert_equal invoice_count, Invoice.count, "an invoice was generated but it shouldn't" assert_equal invoice_items_count, InvoiceItem.count, "some invoice items were generated but they shouldn't" assert_equal users_credit_count, UsersCredit.count, "user's credits count has changed but it shouldn't" assert_equal subscriptions_count + 1, Subscription.count, 'missing the subscription' assert_equal payment_schedule_count + 1, PaymentSchedule.count, 'missing the payment schedule' assert_equal payment_schedule_items_count + 12, PaymentScheduleItem.count, 'missing some payment schedule items' # get the objects reservation = Reservation.last payment_schedule = PaymentSchedule.last # subscription assertions assert_equal 1, @user_without_subscription.subscriptions.count assert_not_nil @user_without_subscription.subscribed_plan, "user's subscribed plan was not found" assert_not_nil @user_without_subscription.subscription, "user's subscription was not found" assert_equal plan.id, @user_without_subscription.subscribed_plan.id, "user's plan does not match" # reservation assertions assert reservation.original_payment_schedule assert_equal payment_schedule.main_object.object, reservation # Check the answer result = json_response(response.body) assert_equal payment_schedule.id, result[:id], 'payment schedule id does not match' subscription = payment_schedule.payment_schedule_objects.find { |pso| pso.object_type == Subscription.name }.object assert_equal plan.id, subscription.plan_id, 'subscribed plan does not match' end test 'user reserves a machine and renew a subscription with payment schedule and coupon and wallet' do user = User.find_by(username: 'lseguin') login_as(user, scope: :user) reservations_count = Reservation.count invoice_count = Invoice.count invoice_items_count = InvoiceItem.count subscriptions_count = Subscription.count user_subscriptions_count = user.subscriptions.count payment_schedule_count = PaymentSchedule.count payment_schedule_items_count = PaymentScheduleItem.count wallet_transactions_count = WalletTransaction.count machine = Machine.find(1) availability = machine.availabilities.last plan = Plan.find_by(group_id: user.group.id, type: 'Plan', base_name: 'Abonnement mensualisable') VCR.use_cassette('reservations_machine_subscription_with_payment_schedule_coupon_wallet') do post '/api/stripe/setup_subscription', params: { payment_method_id: stripe_payment_method, cart_items: { items: [ { reservation: { reservable_id: machine.id, reservable_type: machine.class.name, slots_reservations_attributes: [ { slot_id: availability.slots.first.id } ] } }, { subscription: { plan_id: plan.id } } ], payment_schedule: true, payment_method: 'card', coupon_code: 'GIME3EUR' } }.to_json, headers: default_headers # Check response format & status assert_equal 201, response.status, response.body assert_equal Mime[:json], response.content_type # Check the response res = json_response(response.body) assert_not_nil res[:id] end assert_equal reservations_count + 1, Reservation.count, 'missing the reservation' assert_equal invoice_count, Invoice.count, "an invoice was generated but it shouldn't" assert_equal invoice_items_count, InvoiceItem.count, "some invoice items were generated but they shouldn't" assert_equal 0, UsersCredit.count, "user's credits were not reset" assert_equal subscriptions_count + 1, Subscription.count, 'missing the subscription' assert_equal payment_schedule_count + 1, PaymentSchedule.count, 'missing the payment schedule' assert_equal payment_schedule_items_count + 12, PaymentScheduleItem.count, 'missing some payment schedule items' assert_equal wallet_transactions_count + 1, WalletTransaction.count, 'missing the wallet transaction' # get the objects reservation = Reservation.last subscription = Subscription.last payment_schedule = PaymentSchedule.last # subscription assertions assert_equal user_subscriptions_count + 1, user.subscriptions.count assert_equal user, subscription.user assert_not_nil user.subscribed_plan, "user's subscribed plan was not found" assert_not_nil user.subscription, "user's subscription was not found" assert_equal plan.id, user.subscribed_plan.id, "user's plan does not match" # reservation assertions assert reservation.original_payment_schedule assert_equal payment_schedule.main_object.object, reservation # payment schedule assertions assert_not_nil payment_schedule.reference assert_equal 'card', payment_schedule.payment_method assert_equal 2, payment_schedule.payment_gateway_objects.count assert_not_nil payment_schedule.gateway_payment_mean assert_not_nil payment_schedule.wallet_transaction assert_equal payment_schedule.ordered_items.first.amount, payment_schedule.wallet_amount assert_equal Coupon.find_by(code: 'GIME3EUR').id, payment_schedule.coupon_id assert_equal 'test', payment_schedule.environment assert payment_schedule.check_footprint assert_equal user.invoicing_profile.id, payment_schedule.invoicing_profile_id assert_equal payment_schedule.invoicing_profile_id, payment_schedule.operator_profile_id # Check the answer result = json_response(response.body) assert_equal payment_schedule.id, result[:id], 'payment schedule id does not match' subscription = payment_schedule.payment_schedule_objects.find { |pso| pso.object_type == Subscription.name }.object assert_equal plan.id, subscription.plan_id, 'subscribed plan does not match' end test 'user reserves a space with success' do login_as(@user_without_subscription, scope: :user) space = Space.first availability = space.availabilities.first reservations_count = Reservation.count invoice_count = Invoice.count invoice_items_count = InvoiceItem.count users_credit_count = UsersCredit.count subscriptions_count = Subscription.count VCR.use_cassette('reservations_create_for_space_success') do post '/api/stripe/confirm_payment', params: { payment_method_id: stripe_payment_method, cart_items: { items: [ { reservation: { reservable_id: space.id, reservable_type: space.class.name, slots_reservations_attributes: [ { slot_id: availability.slots.first.id } ] } } ] } }.to_json, headers: default_headers end # general assertions assert_equal 201, response.status assert_equal reservations_count + 1, Reservation.count assert_equal invoice_count + 1, Invoice.count assert_equal invoice_items_count + 1, InvoiceItem.count assert_equal users_credit_count, UsersCredit.count assert_equal subscriptions_count, Subscription.count # subscription assertions assert_equal 0, @user_without_subscription.subscriptions.count assert_nil @user_without_subscription.subscribed_plan # reservation assertions reservation = Reservation.last assert reservation.original_invoice assert_equal 1, reservation.original_invoice.invoice_items.count # invoice_items assertions invoice_item = InvoiceItem.last assert_equal space.prices.find_by(group_id: @user_without_subscription.group_id, plan_id: nil).amount, invoice_item.amount assert invoice_item.check_footprint # invoice assertions item = InvoiceItem.find_by(object: reservation) invoice = item.invoice assert_invoice_pdf invoice assert_not_nil invoice.debug_footprint refute invoice.payment_gateway_object.blank? refute invoice.total.blank? assert invoice.check_footprint # notification assert_not_empty Notification.where(attached_object: reservation) end end