1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798 |
- # frozen_string_literal: true
- # == Schema Information
- #
- # Table name: email_domain_blocks
- #
- # id :bigint(8) not null, primary key
- # domain :string default(""), not null
- # created_at :datetime not null
- # updated_at :datetime not null
- # parent_id :bigint(8)
- # allow_with_approval :boolean default(FALSE), not null
- #
- class EmailDomainBlock < ApplicationRecord
- self.ignored_columns += %w(
- ips
- last_refresh_at
- )
- include DomainNormalizable
- include Paginable
- with_options class_name: 'EmailDomainBlock' do
- belongs_to :parent, optional: true
- has_many :children, foreign_key: :parent_id, inverse_of: :parent, dependent: :destroy
- end
- validates :domain, presence: true, uniqueness: true, domain: true
- scope :parents, -> { where(parent_id: nil) }
- # Used for adding multiple blocks at once
- attr_accessor :other_domains
- def to_log_human_identifier
- domain
- end
- def history
- @history ||= Trends::History.new('email_domain_blocks', id)
- end
- class Matcher
- def initialize(domain_or_domains, attempt_ip: nil)
- @uris = extract_uris(domain_or_domains)
- @attempt_ip = attempt_ip
- end
- def match?(...)
- blocking?(...) || invalid_uri?
- end
- private
- def invalid_uri?
- @uris.any?(&:nil?)
- end
- def blocking?(allow_with_approval: false)
- blocks = EmailDomainBlock.where(domain: domains_with_variants, allow_with_approval: allow_with_approval).by_domain_length
- blocks.each { |block| block.history.add(@attempt_ip) } if @attempt_ip.present?
- blocks.any?
- end
- def domains_with_variants
- @uris.flat_map do |uri|
- next if uri.nil?
- segments = uri.normalized_host.split('.')
- segments.map.with_index { |_, i| segments[i..].join('.') }
- end
- end
- def extract_uris(domain_or_domains)
- Array(domain_or_domains).map do |str|
- domain = if str.include?('@')
- str.split('@', 2).last
- else
- str
- end
- Addressable::URI.new.tap { |u| u.host = domain.strip } if domain.present?
- rescue Addressable::URI::InvalidURIError, IDN::Idna::IdnaError
- nil
- end
- end
- end
- def self.block?(domain_or_domains, attempt_ip: nil)
- Matcher.new(domain_or_domains, attempt_ip: attempt_ip).match?
- end
- def self.requires_approval?(domain_or_domains, attempt_ip: nil)
- Matcher.new(domain_or_domains, attempt_ip: attempt_ip).match?(allow_with_approval: true)
- end
- end
|