houdini/app/models/nonprofit.rb
2021-04-02 13:51:02 -05:00

289 lines
9.8 KiB
Ruby
Executable file

# frozen_string_literal: true
# License: AGPL-3.0-or-later WITH WTO-AP-3.0-or-later
# Full license explanation at https://github.com/houdiniproject/houdini/blob/master/LICENSE
class Nonprofit < ApplicationRecord
attr_accessor :register_np_only, :user_id, :user
Categories = ['Public Benefit', 'Human Services', 'Education', 'Civic Duty', 'Human Rights', 'Animals', 'Environment', 'Health', 'Arts, Culture, Humanities', 'International', 'Children', 'Religion', 'LGBTQ', "Women's Rights", 'Disaster Relief', 'Veterans'].freeze
include Image::AttachmentExtensions
include Model::Jbuilder
# :name, # str
# :stripe_account_id, # str
# :summary, # text: paragraph-sized organization summary
# :tagline, # str
# :email, # str: public organization contact email
# :phone, # str: public org contact phone
# :main_image, # str: url of featured image - first image in profile carousel
# :second_image, # str: url of 2nd image in carousel
# :third_image, # str: url of 3rd image in carousel
# :background_image, # str: url of large profile background
# :remove_background_image, #bool carrierwave
# :logo, # str: small logo image url for searching
# :zip_code, # int
# :website, # str: their own website url
# :categories, # text [str]: see the constant Categories
# :achievements, # text [str]: highlights about this org
# :full_description, # text
# :state_code, # str: two-letter state code (eg. CA)
# :statement, # str: bank statement for donations towards the nonprofit
# :city, # str
# :slug, # str
# :city_slug, #str
# :state_code_slug, #str
# :ein, # str: employee identification number
# :published, # boolean; whether to display this profile
# :vetted, # bool: Whether a super admin (one of CommitChange's employees) have approved this org
# :verification_status, # str (either 'pending', 'unverified', 'escalated', 'verified' -- whether the org has submitted the identity verification form and it has been approved)
# :timezone, # str
# :address, # text
# :thank_you_note, # text
# :referrer, # str
# :no_anon, # bool: whether to allow anonymous donations
# :roles_attributes,
# :brand_font, #string (lowercase key eg. 'helvetica')
# :brand_color, #string (hex color value)
# :hide_activity_feed, # bool
# :tracking_script,
# :facebook, #string (url)
# :twitter, #string (url)
# :youtube, #string (url)
# :instagram, #string (url)
# :blog, #string (url)
# :card_failure_message_top, # text
# :card_failure_message_bottom, # text
# :autocomplete_supporter_address # boolean
has_many :payouts
has_many :charges
has_many :refunds, through: :charges
has_many :donations
has_many :recurring_donations
has_many :payments
has_many :supporters, dependent: :destroy
has_many :supporter_notes, through: :supporters
has_many :profiles, through: :donations
has_many :campaigns, dependent: :destroy
has_many :events, dependent: :destroy
has_many :object_event_hook_configs, dependent: :destroy do
def for_type(type)
self.select do |config|
config.object_event_types.include? type
end
end
end
has_many :tickets, through: :events
has_many :roles, as: :host, dependent: :destroy
has_many :users, through: :roles
has_many :admins, -> { where('name = ?', :nonprofit_admin) }, through: :roles, class_name: 'User', autosave: true, source: :user do
def build_admin(**kwargs)
build(kwargs.merge({name: :nonprofit_admin}))
end
end
has_many :tag_masters, dependent: :destroy
has_many :custom_field_masters, dependent: :destroy
has_many :activities, as: :host, dependent: :destroy
has_many :imports
has_many :email_settings
has_many :cards, as: :holder
has_one :bank_account, -> { where('COALESCE(deleted, false) = false') },
dependent: :destroy
has_one :billing_subscription, dependent: :destroy
has_one :billing_plan, through: :billing_subscription
has_one :miscellaneous_np_info
validates_associated :admins, on: :create
validates :name, presence: true
validates :city, presence: true
validates :state_code, presence: true
validates_format_of :email, with: Email::Regex, allow_nil: true
validates_presence_of :slug
validates_uniqueness_of :slug, scope: %i[city_slug state_code_slug]
validates :website, url: { schemes: ['http', 'https'], public_suffix: true }, allow_nil: true
validates_presence_of :user_id, on: :create, unless: -> {register_np_only}
validate :user_is_valid, on: :create, unless: -> {register_np_only}
validate :user_registerable_as_admin, on: :create, unless: -> {register_np_only}
scope :vetted, -> { where(vetted: true) }
scope :identity_verified, -> { where(verification_status: 'verified') }
scope :published, -> { where(published: true) }
has_one_attached :main_image
has_one_attached :second_image
has_one_attached :third_image
has_one_attached :background_image
has_one_attached :logo
# way too wordy
has_one_attached_with_sizes(:logo, {small: 30, normal: 100, large: 180})
has_one_attached_with_sizes(:background_image, {normal: [1000,600]})
has_one_attached_with_sizes(:main_image, {nonprofit_carousel: [590, 338], thumb: [188, 120], thumb_explore: [100, 100]})
has_one_attached_with_sizes(:second_image, {nonprofit_carousel: [590, 338], thumb: [188, 120], thumb_explore: [100, 100]})
has_one_attached_with_sizes(:third_image, {nonprofit_carousel: [590, 338], thumb: [188, 120], thumb_explore: [100, 100]})
has_one_attached_with_default(:logo, Houdini.defaults.image.profile,
filename: "logo_#{SecureRandom.uuid}#{Pathname.new(Houdini.defaults.image.profile).extname}")
has_one_attached_with_default(:background_image, Houdini.defaults.image.nonprofit,
filename: "background_image_#{SecureRandom.uuid}#{Pathname.new(Houdini.defaults.image.nonprofit).extname}")
has_one_attached_with_default(:main_image, Houdini.defaults.image.profile,
filename: "main_image_#{SecureRandom.uuid}#{Pathname.new(Houdini.defaults.image.profile).extname}")
has_one_attached_with_default(:second_image, Houdini.defaults.image.profile,
filename: "second_image_#{SecureRandom.uuid}#{Pathname.new(Houdini.defaults.image.profile).extname}")
has_one_attached_with_default(:third_image, Houdini.defaults.image.profile,
filename: "third_image_#{SecureRandom.uuid}#{Pathname.new(Houdini.defaults.image.profile).extname}")
serialize :achievements, Array
serialize :categories, Array
before_validation(on: :create) do
set_slugs
set_user
add_billing_subscription
self
end
after_create :build_admin_role, unless: -> {register_np_only}
# Register (create) a nonprofit with an initial admin
def self.register(user, params)
np = create ConstructNonprofit.construct(user, params)
role = Role.create(user: user, name: 'nonprofit_admin', host: np) if np.valid?
np
end
def nonprofit_personnel_emails
roles.nonprofit_personnel.joins(:user).pluck('users.email')
end
def total_recurring
recurring_donations.active.sum(:amount)
end
def donation_history_monthly
donation_history_monthly = []
donations.order('created_at')
.group_by { |d| d.created_at.beginning_of_month }
.each { |_, ds| donation_history_monthly.push(ds.map(&:amount).sum) }
donation_history_monthly
end
def as_json(options = {})
h = super(options)
h[:url] = url
h
end
def url
"/#{state_code_slug}/#{city_slug}/#{slug}"
end
def set_slugs
self.slug = Format::Url.convert_to_slug name unless slug
self.city_slug = Format::Url.convert_to_slug city unless city_slug
unless state_code_slug
self.state_code_slug = Format::Url.convert_to_slug state_code
end
if Nonprofit.where(slug: slug, city_slug: city_slug, state_code_slug: state_code_slug).any?
correct_nonunique_slug
end
self
end
def correct_nonunique_slug
begin
slug = SlugNonprofitNamingAlgorithm.new(self.state_code_slug, self.city_slug).create_copy_name(self.slug)
self.slug = slug
rescue UnableToCreateNameCopyError
errors.add(:slug, "could not be created.")
end
end
def set_user
if (user_id && User.where(id: user_id).any?)
@user = User.find(user_id)
end
self
end
def full_address
Format::Address.full_address(address, city, state_code)
end
def total_raised
QueryPayments.get_payout_totals(QueryPayments.ids_for_payout(id))['net_amount']
end
def can_make_payouts
vetted &&
verification_status == 'verified' &&
bank_account &&
!bank_account.pending_verification
end
def active_cards
cards.where('COALESCE(cards.inactive, FALSE) = FALSE')
end
# @param [Card] card the new active_card
def active_card=(card)
unless card.class == Card
raise ArgumentError, 'Pass a card to active_card or else'
end
Card.transaction do
active_cards.update_all inactive: true
end
cards << card
end
def active_card
active_cards.first
end
def create_active_card(card_data)
if card_data[:inactive]
raise ArgumentError, 'This method is for creating active cards only'
end
active_cards.update_all inactive: true
cards.create(card_data)
end
def currency_symbol
Houdini.intl.all_currencies[currency.downcase.to_sym][:symbol]
end
def to_builder(*expand)
init_builder(*expand) do |json|
json.(self, :name)
end
end
private
def build_admin_role
role = user.roles.build(host: self, name: 'nonprofit_admin')
role.save!
end
def add_billing_subscription
billing_plan = BillingPlan.find(Houdini.default_bp)
b_sub = build_billing_subscription(billing_plan: billing_plan, status: 'active')
end
def user_registerable_as_admin
if user && user.roles.nonprofit_admins.any?
errors.add(:user_id, "cannot already be an admin for a nonprofit.")
end
end
def user_is_valid
(user && user.is_a?(User)) || errors.add(:user_id, "is not a valid user")
end
end