require 'effective_addresses' require 'effective_orders/engine' require 'effective_orders/version' module EffectiveOrders PENDING = 'pending'.freeze # New orders are created in a pending state CONFIRMED = 'confirmed'.freeze # Once the order has passed checkout step 1 DEFERRED = 'deferred'.freeze # Deferred providers. Cheque or Phone was selected. PURCHASED = 'purchased'.freeze # Purchased by provider DECLINED = 'declined'.freeze # Declined by provider STATES = { PENDING => PENDING, CONFIRMED => CONFIRMED, DEFERRED => DEFERRED, PURCHASED => PURCHASED, DECLINED => DECLINED } # Subscription statuses (as per stripe) ACTIVE = 'active'.freeze PAST_DUE = 'past_due'.freeze TRIALING = 'trialing'.freeze CANCELED = 'canceled'.freeze STATUSES = { ACTIVE => ACTIVE, PAST_DUE => PAST_DUE, CANCELED => CANCELED, TRIALING => TRIALING } # The following are all valid config keys mattr_accessor :orders_table_name mattr_accessor :order_items_table_name mattr_accessor :carts_table_name mattr_accessor :cart_items_table_name mattr_accessor :customers_table_name mattr_accessor :subscriptions_table_name mattr_accessor :products_table_name mattr_accessor :authorization_method mattr_accessor :layout mattr_accessor :mailer mattr_accessor :orders_collection_scope mattr_accessor :order_tax_rate_method mattr_accessor :obfuscate_order_ids mattr_accessor :billing_address mattr_accessor :shipping_address mattr_accessor :use_address_full_name mattr_accessor :collect_note mattr_accessor :collect_note_required mattr_accessor :collect_note_message mattr_accessor :terms_and_conditions mattr_accessor :terms_and_conditions_label mattr_accessor :minimum_charge # Features mattr_accessor :free_enabled mattr_accessor :mark_as_paid_enabled mattr_accessor :pretend_enabled mattr_accessor :pretend_message # Payment processors. false or Hash mattr_accessor :cheque mattr_accessor :moneris mattr_accessor :paypal mattr_accessor :phone mattr_accessor :refund mattr_accessor :stripe mattr_accessor :subscriptions # Stripe subscriptions mattr_accessor :trial # Trial mode def self.setup yield self end def self.authorized?(controller, action, resource) @_exceptions ||= [Effective::AccessDenied, (CanCan::AccessDenied if defined?(CanCan)), (Pundit::NotAuthorizedError if defined?(Pundit))].compact return !!authorization_method unless authorization_method.respond_to?(:call) controller = controller.controller if controller.respond_to?(:controller) begin !!(controller || self).instance_exec((controller || self), action, resource, &authorization_method) rescue *@_exceptions false end end def self.authorize!(controller, action, resource) raise Effective::AccessDenied.new('Access Denied', action, resource) unless authorized?(controller, action, resource) end def self.permitted_params [ :cc, :note, :terms_and_conditions, :confirmed_checkout, billing_address: EffectiveAddresses.permitted_params, shipping_address: EffectiveAddresses.permitted_params, subscripter: [:stripe_plan_id, :stripe_token] ] end def self.cheque? cheque.kind_of?(Hash) end def self.free? free_enabled == true end def self.deferred? deferred_providers.present? end def self.mark_as_paid? mark_as_paid_enabled == true end def self.moneris? moneris.kind_of?(Hash) end def self.paypal? paypal.kind_of?(Hash) end def self.phone? phone.kind_of?(Hash) end def self.pretend? pretend_enabled == true end def self.refund? refund.kind_of?(Hash) end def self.stripe? stripe.kind_of?(Hash) end def self.subscriptions? subscriptions.kind_of?(Hash) end def self.trial? trial.kind_of?(Hash) end def self.single_payment_processor? [moneris?, paypal?, stripe?].select { |enabled| enabled }.length == 1 end # The Effective::Order.payment_provider value must be in this collection def self.payment_providers [ ('cheque' if cheque?), ('credit card' if mark_as_paid?), ('free' if free?), ('moneris' if moneris?), ('paypal' if paypal?), ('phone' if phone?), ('pretend' if pretend?), ('refund' if refund?), ('stripe' if stripe?), ('other' if mark_as_paid?), 'none' ].compact end def self.deferred_providers [('cheque' if cheque?), ('phone' if phone?)].compact end def self.can_skip_checkout_step1? return false if require_billing_address return false if require_shipping_address return false if collect_note return false if terms_and_conditions true end def self.stripe_plans return [] unless (stripe? && subscriptions?) @stripe_plans ||= ( Rails.logger.info '[STRIPE] index plans' plans = begin Stripe::Plan.respond_to?(:all) ? Stripe::Plan.all : Stripe::Plan.list rescue => e raise e if Rails.env.production? Rails.logger.info "[STRIPE ERROR]: #{e.message}" Rails.logger.info "[STRIPE ERROR]: effective_orders continuing with empty stripe plans. This would fail loudly in Rails.env.production." [] end plans = plans.map do |plan| description = ("$#{'%0.2f' % (plan.amount / 100.0)}" + ' ' + plan.currency.upcase + '/' + plan.interval.to_s) { id: plan.id, product_id: plan.product, name: plan.nickname || description, description: description, amount: plan.amount, currency: plan.currency, interval: plan.interval, interval_count: plan.interval_count, trial_period_days: plan.trial_period_days } end.sort do |x, y| val ||= (x[:interval] <=> y[:interval]) val = nil if val == 0 val ||= (x[:amount] <=> y[:amount]) val = nil if val == 0 val ||= (x[:name] <=> y[:name]) val = nil if val == 0 val || (x[:id] <=> y[:id]) end # Calculate savings for any yearly per user plans, based on their matching monthly plans plans.select { |plan| plan[:interval] == 'year' }.each do |yearly| monthly_name = yearly[:name].downcase.gsub('year', 'month') monthly = plans.find { |plan| plan[:interval] == 'month' && plan[:name].downcase == monthly_name } next unless monthly savings = (monthly[:amount].to_i * 12) - yearly[:amount].to_i next unless savings > 0 yearly[:savings] = savings end plans ) end def self.stripe_plans_collection stripe_plans.map { |plan| [plan[:name], plan[:id]] } end class SoldOutException < Exception; end class AlreadyPurchasedException < Exception; end end