app/models/invoice.rb
Potentially dangerous attribute available for mass assignment# frozen_string_literal: true # TODO: add uniqueness check on provider/buyer/period scope#Class `Invoice` has 56 methods (exceeds 20 allowed). Consider refactoring.
File `invoice.rb` has 426 lines of code (exceeds 250 allowed). Consider refactoring.
Invoice has at least 53 methods
Invoice assumes too much for instance variable '@period'class Invoice < ApplicationRecord %I[due_on period issued_on last_charging_retry].each do |attr| attribute attr, :date end MAX_CHARGE_RETRIES = 3 DECIMALS = 2 CHARGE_PRECISION = 2 enum creation_type: {manual: 'manual', background: 'background'} include AfterCommitQueue audited :allow_mass_assignment => true has_associated_audits class InvalidInvoiceStateException < RuntimeError; end # Default gap between issued_on and due_on dates ISSUE_AND_DUE_DEFAULT_DELAY = 2.days belongs_to :buyer_account, :class_name => 'Account' belongs_to :provider_account, :class_name => 'Account' delegate :s3_provider_prefix, to: :provider_account alias provider provider_account alias buyer buyer_account has_many :paid_line_items, -> { where(invoices: {state: 'paid'}).includes(:invoice).references(:invoice) }, class_name: 'LineItem' has_many :line_items, -> { oldest_first }, dependent: :destroy, inverse_of: :invoice has_many :payment_transactions, -> { oldest_first }, dependent: :nullify, inverse_of: :invoice has_many :payment_intents, dependent: :destroy, inverse_of: :invoice has_attached_file :pdf, url: ':url_root/:class/:id/:attachment/:style/:basename.:extension' do_not_validate_attachment_file_type :pdf attr_accessible :provider_account, :buyer_account, :friendly_id, :period validates :provider_account, :buyer_account, :friendly_id, presence: true validates :period, presence: { :message => 'Billing period format should be YYYY-MM' } validate :period_range_valid?, if: -> { period.present? && will_save_change_to_period? } validates :friendly_id, format: { with: /\A\d{4}(-\d{2})?-\d{8}\Z/, message: 'format should be YYYY-MM-XXXXXXXX or YYYY-XXXXXXXX', if: :friendly_id_changed? } validates :from_address_name, :from_address_line1, :from_address_line2, :from_address_city, :from_address_region, :from_address_state, :from_address_country, :from_address_zip, :from_address_phone, :to_address_name, :to_address_line1, :to_address_line2, :to_address_city, :to_address_region, :to_address_state, :to_address_country, :to_address_zip, :to_address_phone, length: {maximum: 255} default_scope -> { order('invoices.created_at DESC') } # 'conditions' is a simple convenience method defined here ... see below scope :before, ->(month) { where('period < ?', month.beginning_of_month.to_date ) } scope :due, ->(time) { where(:due_on => time.to_date) } scope :due_on_or_before, ->(date) { where('due_on <= ?', date ) } scope :finalized_before, ->(date) { where("state='finalized' AND finalized_at <= ?", date) } # The month should be a YYYY-MM formatted string. scope :by_month, ->(month) { where(:period => ::Month.parse_month(month)) } scope :by_year, ->(year) { where.has { sift(:year, period) == year } } scope :by_month_number, ->(month) { where.has { sift(:month_number, period) == month } } # Can use * as wildcard in friendly id scope :by_number, ->(number) { number = number.dup if number.tr!('*', '%') where('friendly_id LIKE ?', number) else where(:friendly_id => number) end } scope :without_ids, ->(invoice) { where('id <> ?', invoice.id) } scope :by_state, ->(state) { where(:state => state.to_s) } scope :by_buyer, ->(buyer) { where(:buyer_account_id => buyer.id) } scope :by_buyer_query, ->(query) { where(:buyer_account_id => Account.buyers.search_ids(query)) } scope :by_provider, ->(provider) { where(:provider_account_id => provider.id) } # the invoice has to be due and at least 3 days later than the last # automatic charging date to be automatically chargeable scope :chargeable, ->(now) { where.has do ((state == 'unpaid') | (state == 'pending')) & (due_on <= now) & ((last_charging_retry == nil) | (last_charging_retry <= (now - 3.days))) end } scope :opened, -> { where(:state => 'open') } scope :finalized, -> { where(:state => 'finalized') } scope :not_cancelled, -> { where("#{Invoice.table_name}.state <> 'cancelled'") } scope :not_frozen, -> { where("#{Invoice.table_name}.state = 'open' OR #{Invoice.table_name}.state = 'finalized'") } scope :visible_for_buyer, -> { where(state: ["pending", "unpaid", "paid", "failed"]) } scope :by_creation_type, ->(creation_type) { where(creation_type: Invoice.creation_types[creation_type]) } scope :with_normalized_friendly_id, ->(numbering_period, month) { case numbering_period when 'monthly' by_year(month.begin.year).by_month_number(month.begin.month).where.has { func(:length, friendly_id) == 16 } when 'yearly', nil by_year(month.begin.year).where.has { func(:length, friendly_id) == 13 } else raise "unknown numbering period: #{numbering_period}.inspect" end .reordering { func(:substr, friendly_id, -8).desc } } after_create :log_entry_created include ThreeScale::Search::Scopes self.allowed_sort_columns = %w{ friendly_id accounts.org_name period state } self.sort_columns_joins = {'accounts.org_name' => :buyer_account} self.allowed_search_scopes = [:number, :month, :month_number, :year, :state, :buyer_query] composed_of :from_address, :mapping => ThreeScale::Address.mapping('from_address'), :class_name => 'ThreeScale::Address' composed_of :to_address, :mapping => ThreeScale::Address.mapping('to_address'), :class_name => 'ThreeScale::Address' state_machine :initial => :open do state :open state :finalized state :pending state :unpaid state :paid state :failed state :cancelled state :open, :finalized do def from provider.address_for_invoice end def to buyer.address_for_invoice end def currency provider.try!(:currency) end def fiscal_code (buyer && (buyer.fiscal_code || '')) || '' end def vat_code (buyer && (buyer.vat_code || '')) || '' end end state all - [ :open, :finalized ] do def from self.from_address end def to self.to_address end def currency self[:currency] end def fiscal_code self[:fiscal_code] || '' end def vat_code self[:vat_code] || '' end end before_transition :to => :finalized do |invoice| invoice.finalized_at = Time.now.utc end # Mental note: # # TL;DR - Data of the invoices closed before the Rails3 deploy are # not necessarily correct (in sync with the PDFs). # # Why? Because BEFORE having the freezing implemented, addresses # and vat_stuff (tm) was changing dynamically. Now was all # frozen at the moment of migration so they should be 'more correct' # although not perfect. Invoices issued AFTER the Rails3 deploy # can be considered reliable. # before_transition :to => :pending do |invoice| raise "Cannot issue invoice (#{invoice.id}) with a deleted buyer" unless invoice.buyer invoice.vat_rate = invoice.buyer.vat_rate invoice.vat_code = invoice.buyer.vat_code || '' invoice.fiscal_code = invoice.buyer.fiscal_code || '' invoice.currency = invoice.provider.currency # freezes all the available information invoice.from_address = invoice.provider.address_for_invoice invoice.to_address = invoice.buyer.address_for_invoice invoice.issued_on = Time.now.utc.to_date invoice.due_on = invoice.issued_on + ISSUE_AND_DUE_DEFAULT_DELAY invoice.generate_pdf! end before_transition :to => :paid do |invoice| invoice.paid_at = Time.now.utc end master_invoice = ->(invoice) { invoice.provider.master? } after_transition to: :paid, do: :notify_buyer_about_payment after_transition to: :paid, if: master_invoice do |invoice, _| account = invoice.buyer_account bought_plan = account.bought_plan ThreeScale::Analytics.track_account(account, 'Charged Invoice', { plan: bought_plan.name, period: invoice.period.to_s, revenue: invoice.cost.to_f }) end after_transition if: master_invoice do |invoice, _| account = invoice.buyer_account ThreeScale::Analytics::Salesforce.new(account).update_invoice_status(invoice) end event :finalize do transition :open => :finalized end event :issue do transition [ :open, :finalized ] => :pending end event :mark_as_unpaid do transition :pending => :unpaid end event :pay do transition [ :unpaid, :pending, :failed ] => :paid end event :fail do transition :unpaid => :failed end event :cancel do transition [ :open, :finalized, :pending, :unpaid, :failed ] => :cancelled end end # this scope has to be defined after the state_machine definition, or # demons are released (create invoice crashes) scope :unresolved, -> { where(:state => ['open', 'finalized', 'pending', 'unpaid']) } private :issue! # ---- Instance methods ---- def log_entry_created LogEntry.log( :info, "Invoice created for #{buyer_account.org_name} for period #{period}", provider_account, buyer_account) end # TODO: move to decorator def name self.period.begin.strftime('%B, %Y') end def cinstance buyer_account.bought_cinstances.provided_by(provider_account).first end def to_xml(options = {}) markup = Finance::Builder::XmlMarkup.new(options)Invoice#to_xml refers to 'markup' more than self (maybe move it to another class?) markup.invoice!(self) markup.to_xml end def issued?Invoice#issued? performs a nil-check !issued_on.nil? end Invoice has missing safe method 'generate_pdf!' def generate_pdf! data = Pdf::Finance::InvoiceReportData.new(self) self.pdf = Pdf::Finance::InvoiceGenerator.new(data).generate_as_attachment save(:validate => false) end def period unless @period attr = self[:period] @period = attr ? ::Month.new(attr) : nil end @period end def reload(*) @period = nil super end # vat_rate is updated by a buyer#after_save callback so the code # is the same both before and after issuing (see Account#update_vat_rates) # before_create :set_vat_rate def set_vat_rate self.vat_rate = buyer.vat_rate end # TODO: - move to submodule of Month and add 'has_month :period' or # similar helper def period=(value) # TODO: duck type it if ::Month === value @period = value elsif Range === value @period = ::Month.new(value.begin.beginning_of_month) elsif String === value @period = ::Month.parse_month(value) else raise ArgumentError.new("Expected Month or Range instance, got #{value.class}") end self[:period] = @period.try!(:begin) end # REFACTOR: remove - use period.begin instead def period_start period.begin end # REFACTOR: remove - use period.end instead delegate :end, to: :period, prefix: true def vat_amount (BigDecimal((vat_rate || 0).to_s) / 100 * exact_cost_without_vat).to_has_money(currency) end def charge_cost_vat_amount vat_amount.round(CHARGE_PRECISION) end def exact_cost_without_vat line_items.sum(:cost).to_has_money(currency) end def charge_cost_without_vat exact_cost_without_vat.round(CHARGE_PRECISION) end def charge_cost_with_vat charge_cost_without_vat + charge_cost_vat_amount end def exact_cost_with_vat (charge_cost_without_vat + charge_cost_vat_amount).to_has_money(currency) end def charge_cost charge_cost_with_vat.to_has_money(currency) end def next_transition_from_state(state)Invoice#next_transition_from_state has the variable name 't' state_transitions.find {|t| t.to == state.to_s } end # @deprecatedInvoice#cost has boolean parameter 'vat_included' def cost(vat_included: true, rounding: CHARGE_PRECISION)Invoice#cost is controlled by argument 'vat_included' sum = vat_included ? exact_cost_with_vat : exact_cost_without_vat Invoice#cost is controlled by argument 'rounding' if rounding sum = sum.round(CHARGE_PRECISION) end sum.to_has_money(currency) end # REFACTOR: remove this method and replace it by open? def current? period.same_month?(Time.now.utc.to_date) && !buyer_account.try!(:destroyed?) end def editable? open? || finalized? end def check_editable_line_items raise InvalidInvoiceStateException, state unless editable? end Invoice has missing safe method 'issue_and_pay_if_free!' def issue_and_pay_if_free! issue! pay! if self.cost == 0 end # Enhances the AASM allowed_event? method so that it # can also govern other methods. # # REFACTOR: read more about loopback transitions in # http://www.pluginaweek.org/ def transition_allowed?(event) allowed = case event when :charge [ :pending, :failed, :unpaid ].include?(self.state.to_sym) when :generate_pdf true else state_events.include?(event) end end # REFACTOR: charging should not happen here # When charging is successful, the invoice is marked as paid (method # #pay! is called) #Method `charge!` has a Cognitive Complexity of 32 (exceeds 5 allowed). Consider refactoring.
Method `charge!` has 43 lines of code (exceeds 25 allowed). Consider refactoring.
Invoice#charge! has boolean parameter 'automatic'
Invoice#charge! has approx 25 statements
Invoice has missing safe method 'charge!' def charge!(automatic = true) ensure_payable_state! unless chargeable? logger.info "Not charging invoice #{id} (buyer #{buyer_account_id}), reason: #{reason_cannot_charge}" cancel! unless positive? return end if buyer_account.charge!(cost, :invoice => self)Invoice#charge! calls 'provider.billing_strategy' 2 times provider.billing_strategy&.info("Invoice #{id} (buyer #{buyer_account_id}) for period #{period} was charged, marking as paid", buyer) pay! else logger.info("Invoice #{id} (buyer #{buyer_account_id}) was not charged") false end rescue Finance::Payment::CreditCardError, ActiveMerchant::ActiveMerchantError provider.billing_strategy&.error("Error when charging invoice #{id} (buyer #{buyer_account_id})", buyer) Invoice#charge! is controlled by argument 'automatic' if automatic self.charging_retries_count += 1 self.last_charging_retry = Time.now.utc.to_date # REFACTOR: Move the logic to InvoiceMessenger if charging_retries_count < MAX_CHARGE_RETRIES if unpaid? logger.info("Invoice #{id} (buyer #{buyer_account_id}) remains unpaid after #{charging_retries_count} attempts, will be retried") save! else logger.info("Marking invoice #{id} (buyer #{buyer_account_id}) as unpaid, will be retried") mark_as_unpaid! end InvoiceMessenger.unsuccessfully_charged_for_buyer(self).deliver # do not send email if provider's using new notification systemInvoice#charge! calls 'provider_account.provider_can_use?(:new_notification_system)' 2 times unless provider_account.provider_can_use?(:new_notification_system) InvoiceMessenger.unsuccessfully_charged_for_provider(self).deliver end event = Invoices::UnsuccessfullyChargedInvoiceProviderEvent.create(self)Invoice#charge! calls 'Rails.application.config' 2 times
Invoice#charge! calls 'Rails.application' 2 times
Invoice#charge! calls 'Rails.application.config.event_store.publish_event(event)' 2 times
Invoice#charge! calls 'Rails.application.config.event_store' 2 times Rails.application.config.event_store.publish_event(event) else logger.info("Marking invoice #{id} (buyer #{buyer_account_id}) as failed (too many retries)") fail! # TODO: Decouple the notification to observer and delete the IF InvoiceMessenger.unsuccessfully_charged_for_buyer_final(self).deliver # do not send email if provider's using new notification system unless provider_account.provider_can_use?(:new_notification_system) InvoiceMessenger.unsuccessfully_charged_for_provider_final(self).deliver end event = Invoices::UnsuccessfullyChargedInvoiceFinalProviderEvent.create(self) Rails.application.config.event_store.publish_event(event) end end end Invoice has missing safe method 'ensure_payable_state!' def ensure_payable_state! return if state_events.include?(:pay) logger.info("Invoice #{id} (buyer #{buyer_account_id}) was not charged because the state events don't include :pay") raise InvalidInvoiceStateException.new("Invoice #{id} is not in chargeable state!") end delegate :positive?, :negative?, :zero?, to: :cost delegate :present?, :payment_gateway_configured?, to: :provider, prefix: true delegate :paying_monthly?, to: :buyer_account, prefix: true def not_paid? !paid? end CONDITIONS_TO_CHARGE = %i[not_paid provider_present provider_payment_gateway_configured positive buyer_account_paying_monthly].freeze def reason_cannot_charge reason = CONDITIONS_TO_CHARGE.find { |condition| !method("#{condition}?").call } I18n.t(reason, scope: %i[invoices reasons_cannot_charge]) if reason end def chargeable? !reason_cannot_charge end def latest_pending_payment_intent payment_intents.pending.latest.first end def self.opened_by_buyer(buyer) opened.by_provider(buyer.provider_account) .where(['invoices.buyer_account_id = ?', buyer.id ]) .reorder('period DESC, created_at DESC').first end # TODO: investigate this ... should not be happening on-demand # def self.find_by_provider_account!(provider_account) find_by_provider_account_id(provider_account.to_param) end # TODO: scheduled for removal - see bill_for method on contract def used? true end # TODO: Remove both lines from Invoice def mark_as_used # noop end def should_bill? buyer_account.billing_monthly? end def friendly_id_already_used? self.provider_account.buyer_invoices.by_number(friendly_id).without_ids(self).exists? end def counter provider_account.buyer_invoice_counters.find_or_create_by(invoice_prefix: id_prefix) end def id_prefix return if friendly_id.blank? friendly_id.sub(/(-[^-]+?)$/, '') end def id_sufix return if friendly_id.blank? friendly_id.sub(/(.+)-/, '') end after_commit :set_friendly_id, on: :create after_commit :update_counter, on: :update def set_friendly_id return unless persisted? self.friendly_id = InvoiceFriendlyIdService.call(self) end private :set_friendly_id def update_counter return unless saved_change_to_friendly_id? counter.update_count(id_sufix.to_i) end def buyer_field_label(name) (buyer_account || provider_account.buyer_accounts.build).field_label(name) end # Returns years which have invoice, scoped by provider def self.years_by_provider(provider_id) connection.select_values( selecting { sift(:year, period).as('year') } .where.has { provider_account_id == provider_id } .distinct.reorder('year DESC').to_sql ).map(&:to_i) end protected def notify_buyer_about_payment run_after_commit do InvoiceMessenger.successfully_charged(self).deliver end end private # Allowed range for the invoice period is: # - from: creation date of the provider # - to: 12 months from now def period_range_valid? return if self[:period]&.between?(provider_account.created_at.to_date.beginning_of_month, Time.zone.now + 12.months) errors.add(:period, :invalid_range) endend