Search code examples
ruby-on-railsrubyruby-on-rails-3billingrecurring-billing

Applying payments to invoices in rails billing system


I've already written most of the functional parts to a custom billing system for my business (Wireless ISP,) but have stumped myself with a bit of it.

Here's the basic overview: It's a recurring billing system for my customers that automatically generates and sends invoices to each customer every month. However, I need to allow check/cash payments since I deal with local customers and also need to allow pre-payment so I can't just use something like Stripe's recurring billing. Essentially the payments are not directly associated with invoices to allow for these type of payments.

models/invoice.rb

class Invoice < ActiveRecord::Base
  belongs_to :customer
  has_many :line_items
  has_many :invoice_payments
  has_many :payments, through: :invoice_payments

models/payment.rb

class Payment < ActiveRecord::Base
  belongs_to :customer

models/customer.rb

class Customer < ActiveRecord::Base
  has_many :subscriptions, dependent: :destroy
  has_many :services, through: :subscriptions
  has_many :invoices, :order => 'id DESC'
  has_many :payments

I need a method that will automatically associate all unapplied payments to all unpaid invoices. Right now I'm putting that as def apply_unapplied_payments on the customer model, but will likely abstract it out later in to its own module in lib/.

This is what I've done so far in models/customer.rb

def apply_unapplied_payments
  unpaid_invoices = invoices.find_by_status("unpaid")
  unapplied_payments = payments.where("invoice_id is null")
  unpaid_invoices.each do |invoice|
    # find the oldest unapplied payment
    # apply payment to this invoice
    # if payment > invoice, keep whatever amount is left and move to next invoice
    # apply leftover amount to invoice, if invoice is still unpaid, move to next unapplied payment
  end
  customer.update_balance
end

Any suggestions for what to fill in the pseudocode with? I'm open to any level of refactoring, so please let me know if you can think of a better way to handle this!


Solution

  • Here's what I'd do (may require some helper methods in payment and invoice classes):

    def apply_unapplied_payments
      unpaid_invoices = invoices.find_by_status("unpaid")
      unapplied_payments = payments.where("invoice_id is null")
      invoice = unpaid_invoices.shift
      payment = unapplied_payments.shift
      while invoice && invoice.remaining_balance > 0 && payment && payment.remaining_credit > 0
          apply_payment_to_invoice(invoice, payment)
          invoice = unpaid_invoices.shift    if invoice.remaining_balance == 0
          payment = unapplied_payments.shift if payment.remaining_credit  == 0
      end
    end