mirror of
https://github.com/funamitech/mastodon
synced 2024-12-11 21:29:17 +09:00
08652baab0
=~ made sense when we were passing it through to a regex, but we're no longer doing that: TagMatcher looks at individual tags and returns a value that *looks* like what you get out of #=~ but really isn't that meaningful. Probably a good idea to not subvert convention like this and instead use a name with guessable intent.
101 lines
2.3 KiB
Ruby
101 lines
2.3 KiB
Ruby
# frozen_string_literal: true
|
|
# == Schema Information
|
|
#
|
|
# Table name: glitch_keyword_mutes
|
|
#
|
|
# id :integer not null, primary key
|
|
# account_id :integer not null
|
|
# keyword :string not null
|
|
# whole_word :boolean default(TRUE), not null
|
|
# created_at :datetime not null
|
|
# updated_at :datetime not null
|
|
#
|
|
|
|
class Glitch::KeywordMute < ApplicationRecord
|
|
belongs_to :account, required: true
|
|
|
|
validates_presence_of :keyword
|
|
|
|
after_commit :invalidate_cached_matchers
|
|
|
|
def self.text_matcher_for(account_id)
|
|
TextMatcher.new(account_id)
|
|
end
|
|
|
|
def self.tag_matcher_for(account_id)
|
|
TagMatcher.new(account_id)
|
|
end
|
|
|
|
private
|
|
|
|
def invalidate_cached_matchers
|
|
Rails.cache.delete(TextMatcher.cache_key(account_id))
|
|
Rails.cache.delete(TagMatcher.cache_key(account_id))
|
|
end
|
|
|
|
class RegexpMatcher
|
|
attr_reader :account_id
|
|
attr_reader :regex
|
|
|
|
def initialize(account_id)
|
|
@account_id = account_id
|
|
regex_text = Rails.cache.fetch(self.class.cache_key(account_id)) { make_regex_text }
|
|
@regex = /#{regex_text}/
|
|
end
|
|
|
|
protected
|
|
|
|
def keywords
|
|
Glitch::KeywordMute.where(account_id: account_id).pluck(:whole_word, :keyword)
|
|
end
|
|
|
|
def boundary_regex_for_keyword(keyword)
|
|
sb = keyword =~ /\A[[:word:]]/ ? '\b' : ''
|
|
eb = keyword =~ /[[:word:]]\Z/ ? '\b' : ''
|
|
|
|
/(?mix:#{sb}#{Regexp.escape(keyword)}#{eb})/
|
|
end
|
|
end
|
|
|
|
class TextMatcher < RegexpMatcher
|
|
def self.cache_key(account_id)
|
|
format('keyword_mutes:regex:text:%s', account_id)
|
|
end
|
|
|
|
def matches?(str)
|
|
!!(regex =~ str)
|
|
end
|
|
|
|
private
|
|
|
|
def make_regex_text
|
|
kws = keywords.map! do |whole_word, keyword|
|
|
whole_word ? boundary_regex_for_keyword(keyword) : keyword
|
|
end
|
|
|
|
Regexp.union(kws).source
|
|
end
|
|
end
|
|
|
|
class TagMatcher < RegexpMatcher
|
|
def self.cache_key(account_id)
|
|
format('keyword_mutes:regex:tag:%s', account_id)
|
|
end
|
|
|
|
def matches?(tags)
|
|
tags.pluck(:name).any? { |n| regex =~ n }
|
|
end
|
|
|
|
private
|
|
|
|
def make_regex_text
|
|
kws = keywords.map! do |whole_word, keyword|
|
|
term = (Tag::HASHTAG_RE =~ keyword) ? $1 : keyword
|
|
whole_word ? boundary_regex_for_keyword(term) : term
|
|
end
|
|
|
|
Regexp.union(kws).source
|
|
end
|
|
end
|
|
end
|