class User < ActiveRecord::Base include Verification devise :database_authenticatable, :registerable, :confirmable, :recoverable, :rememberable, :trackable, :validatable, :omniauthable, :async, :password_expirable, :secure_validatable acts_as_voter acts_as_paranoid column: :hidden_at include ActsAsParanoidAliases has_one :administrator has_one :moderator has_one :valuator has_one :manager has_one :organization has_one :lock has_many :flags has_many :identities, dependent: :destroy has_many :debates, -> { with_hidden }, foreign_key: :author_id has_many :proposals, -> { with_hidden }, foreign_key: :author_id has_many :comments, -> { with_hidden } has_many :spending_proposals, foreign_key: :author_id has_many :failed_census_calls has_many :notifications has_many :direct_messages_sent, class_name: 'DirectMessage', foreign_key: :sender_id has_many :direct_messages_received, class_name: 'DirectMessage', foreign_key: :receiver_id belongs_to :geozone validates :username, presence: true, if: :username_required? validates :username, uniqueness: { scope: :registering_with_oauth }, if: :username_required? validates :document_number, uniqueness: { scope: :document_type }, allow_nil: true validate :validate_username_length validates :official_level, inclusion: {in: 0..5} validates :terms_of_service, acceptance: { allow_nil: false }, on: :create validates :locale, inclusion: {in: I18n.available_locales.map(&:to_s), allow_nil: true} validates_associated :organization, message: false accepts_nested_attributes_for :organization, update_only: true attr_accessor :skip_password_validation attr_accessor :use_redeemable_code scope :administrators, -> { joins(:administrators) } scope :moderators, -> { joins(:moderator) } scope :organizations, -> { joins(:organization) } scope :officials, -> { where("official_level > 0") } scope :for_render, -> { includes(:organization) } scope :by_document, -> (document_type, document_number) { where(document_type: document_type, document_number: document_number) } scope :email_digest, -> { where(email_digest: true) } scope :active, -> { where(erased_at: nil) } before_validation :clean_document_number # Get the existing user by email if the provider gives us a verified email. def self.first_or_initialize_for_oauth(auth) oauth_email = auth.info.email oauth_email_confirmed = oauth_email.present? && (auth.info.verified || auth.info.verified_email) oauth_user = User.find_by(email: oauth_email) if oauth_email_confirmed oauth_user || User.new( username: auth.info.name || auth.uid, email: oauth_email, oauth_email: oauth_email, password: Devise.friendly_token[0,20], terms_of_service: '1', confirmed_at: oauth_email_confirmed ? DateTime.current : nil ) end def name organization? ? organization.name : username end def debate_votes(debates) voted = votes.for_debates(debates) voted.each_with_object({}) { |v, h| h[v.votable_id] = v.value } end def proposal_votes(proposals) voted = votes.for_proposals(proposals) voted.each_with_object({}) { |v, h| h[v.votable_id] = v.value } end def spending_proposal_votes(spending_proposals) voted = votes.for_spending_proposals(spending_proposals) voted.each_with_object({}) { |v, h| h[v.votable_id] = v.value } end def comment_flags(comments) comment_flags = flags.for_comments(comments) comment_flags.each_with_object({}){ |f, h| h[f.flaggable_id] = true } end def administrator? administrator.present? end def moderator? moderator.present? end def valuator? valuator.present? end def manager? manager.present? end def organization? organization.present? end def verified_organization? organization && organization.verified? end def official? official_level && official_level > 0 end def add_official_position!(position, level) return if position.blank? || level.blank? update official_position: position, official_level: level.to_i end def remove_official_position! update official_position: nil, official_level: 0 end def has_official_email? domain = Setting['email_domain_for_officials'] !email.blank? && ( (email.end_with? "@#{domain}") || (email.end_with? ".#{domain}") ) end def display_official_position_badge? return true if official_level > 1 official_position_badge? && official_level == 1 end def block debates_ids = Debate.where(author_id: id).pluck(:id) comments_ids = Comment.where(user_id: id).pluck(:id) proposal_ids = Proposal.where(author_id: id).pluck(:id) self.hide Debate.hide_all debates_ids Comment.hide_all comments_ids Proposal.hide_all proposal_ids end def erase(erase_reason = nil) self.update( erased_at: Time.current, erase_reason: erase_reason, username: nil, email: nil, unconfirmed_email: nil, phone_number: nil, encrypted_password: "", confirmation_token: nil, reset_password_token: nil, email_verification_token: nil, confirmed_phone: nil, unconfirmed_phone: nil ) self.identities.destroy_all end def erased? erased_at.present? end def locked? Lock.find_or_create_by(user: self).locked? end def self.search(term) term.present? ? where("email = ? OR username ILIKE ?", term, "%#{term}%") : none end def self.username_max_length @@username_max_length ||= self.columns.find { |c| c.name == 'username' }.limit || 60 end def show_welcome_screen? sign_in_count == 1 && unverified? && !organization && !administrator? end def password_required? return false if skip_password_validation super end def username_required? !organization? && !erased? end def email_required? !erased? end def locale self[:locale] ||= I18n.default_locale.to_s end def confirmation_required? super && !registering_with_oauth end def send_oauth_confirmation_instructions if oauth_email != email self.update(confirmed_at: nil) self.send_confirmation_instructions end self.update(oauth_email: nil) if oauth_email.present? end def name_and_email "#{name} (#{email})" end def save_requiring_finish_signup begin self.registering_with_oauth = true self.save(validate: false) # Devise puts unique constraints for the email the db, so we must detect & handle that rescue ActiveRecord::RecordNotUnique self.email = nil self.save(validate: false) end true end def ability @ability ||= Ability.new(self) end delegate :can?, :cannot?, to: :ability private def clean_document_number self.document_number = self.document_number.gsub(/[^a-z0-9]+/i, "").upcase unless self.document_number.blank? end def validate_username_length validator = ActiveModel::Validations::LengthValidator.new( attributes: :username, maximum: User.username_max_length) validator.validate(self) end end