models: factor out concerns to app/logical/concerns.
This commit is contained in:
36
app/logical/concerns/has_bit_flags.rb
Normal file
36
app/logical/concerns/has_bit_flags.rb
Normal file
@@ -0,0 +1,36 @@
|
||||
module HasBitFlags
|
||||
extend ActiveSupport::Concern
|
||||
|
||||
# NOTE: the ordering of attributes has to be fixed#
|
||||
# new attributes should be appended to the end.
|
||||
def has_bit_flags(attributes, field: :bit_flags)
|
||||
attributes.each.with_index do |attribute, i|
|
||||
bit_flag = 1 << i
|
||||
|
||||
define_method(attribute) do
|
||||
send(field) & bit_flag > 0
|
||||
end
|
||||
|
||||
define_method("#{attribute}?") do
|
||||
send(field) & bit_flag > 0
|
||||
end
|
||||
|
||||
define_method("#{attribute}=") do |val|
|
||||
if val.to_s =~ /t|1|y/
|
||||
send("#{field}=", send(field) | bit_flag)
|
||||
else
|
||||
send("#{field}=", send(field) & ~bit_flag)
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
# bit_prefs_match
|
||||
define_singleton_method("#{field}_match") do |flag, value|
|
||||
value = value ? 1 : 0
|
||||
bits = attributes.length
|
||||
bit_index = bits - attributes.index(flag.to_s) - 1
|
||||
|
||||
where(sanitize_sql(["get_bit(#{field}::bit(?), ?) = ?", bits, bit_index, value]))
|
||||
end
|
||||
end
|
||||
end
|
||||
291
app/logical/concerns/searchable.rb
Normal file
291
app/logical/concerns/searchable.rb
Normal file
@@ -0,0 +1,291 @@
|
||||
module Searchable
|
||||
extend ActiveSupport::Concern
|
||||
|
||||
def where_like(attr, value)
|
||||
where("#{qualified_column_for(attr)} LIKE ? ESCAPE E'\\\\'", value.to_escaped_for_sql_like)
|
||||
end
|
||||
|
||||
def where_not_like(attr, value)
|
||||
where.not("#{qualified_column_for(attr)} LIKE ? ESCAPE E'\\\\'", value.to_escaped_for_sql_like)
|
||||
end
|
||||
|
||||
def where_ilike(attr, value)
|
||||
where("#{qualified_column_for(attr)} ILIKE ? ESCAPE E'\\\\'", value.mb_chars.to_escaped_for_sql_like)
|
||||
end
|
||||
|
||||
def where_not_ilike(attr, value)
|
||||
where.not("#{qualified_column_for(attr)} ILIKE ? ESCAPE E'\\\\'", value.mb_chars.to_escaped_for_sql_like)
|
||||
end
|
||||
|
||||
def where_iequals(attr, value)
|
||||
where_ilike(attr, value.gsub(/\\/, '\\\\').gsub(/\*/, '\*'))
|
||||
end
|
||||
|
||||
# https://www.postgresql.org/docs/current/static/functions-matching.html#FUNCTIONS-POSIX-REGEXP
|
||||
# "(?e)" means force use of ERE syntax; see sections 9.7.3.1 and 9.7.3.4.
|
||||
def where_regex(attr, value)
|
||||
where("#{qualified_column_for(attr)} ~ ?", "(?e)" + value)
|
||||
end
|
||||
|
||||
def where_not_regex(attr, value)
|
||||
where.not("#{qualified_column_for(attr)} ~ ?", "(?e)" + value)
|
||||
end
|
||||
|
||||
def where_inet_matches(attr, value)
|
||||
if value.match?(/[, ]/)
|
||||
ips = value.split(/[, ]+/).map { |ip| IPAddress.parse(ip).to_string }
|
||||
where("#{qualified_column_for(attr)} = ANY(ARRAY[?]::inet[])", ips)
|
||||
else
|
||||
ip = IPAddress.parse(value)
|
||||
where("#{qualified_column_for(attr)} <<= ?", ip.to_string)
|
||||
end
|
||||
end
|
||||
|
||||
def where_array_includes_any(attr, values)
|
||||
where("#{qualified_column_for(attr)} && ARRAY[?]", values)
|
||||
end
|
||||
|
||||
def where_array_includes_all(attr, values)
|
||||
where("#{qualified_column_for(attr)} @> ARRAY[?]", values)
|
||||
end
|
||||
|
||||
def where_array_includes_any_lower(attr, values)
|
||||
where("lower(#{qualified_column_for(attr)}::text)::text[] && ARRAY[?]", values.map(&:downcase))
|
||||
end
|
||||
|
||||
def where_array_includes_all_lower(attr, values)
|
||||
where("lower(#{qualified_column_for(attr)}::text)::text[] @> ARRAY[?]", values.map(&:downcase))
|
||||
end
|
||||
|
||||
def where_text_includes_lower(attr, values)
|
||||
where("lower(#{qualified_column_for(attr)}) IN (?)", values.map(&:downcase))
|
||||
end
|
||||
|
||||
def where_array_count(attr, value)
|
||||
relation = all
|
||||
qualified_column = "cardinality(#{qualified_column_for(attr)})"
|
||||
parsed_range = Tag.parse_helper(value, :integer)
|
||||
|
||||
PostQueryBuilder.new(nil).add_range_relation(parsed_range, qualified_column, relation)
|
||||
end
|
||||
|
||||
def search_boolean_attribute(attribute, params)
|
||||
return all unless params.key?(attribute)
|
||||
|
||||
value = params[attribute].to_s
|
||||
if value.truthy?
|
||||
where(attribute => true)
|
||||
elsif value.falsy?
|
||||
where(attribute => false)
|
||||
else
|
||||
raise ArgumentError, "value must be truthy or falsy"
|
||||
end
|
||||
end
|
||||
|
||||
def search_inet_attribute(attr, params)
|
||||
if params[attr].present?
|
||||
where_inet_matches(attr, params[attr])
|
||||
else
|
||||
all
|
||||
end
|
||||
end
|
||||
|
||||
# range: "5", ">5", "<5", ">=5", "<=5", "5..10", "5,6,7"
|
||||
def numeric_attribute_matches(attribute, range)
|
||||
return all unless range.present?
|
||||
|
||||
column = column_for_attribute(attribute)
|
||||
qualified_column = "#{table_name}.#{column.name}"
|
||||
parsed_range = Tag.parse_helper(range, column.type)
|
||||
|
||||
PostQueryBuilder.new(nil).add_range_relation(parsed_range, qualified_column, self)
|
||||
end
|
||||
|
||||
def text_attribute_matches(attribute, value, index_column: nil, ts_config: "english")
|
||||
return all unless value.present?
|
||||
|
||||
column = column_for_attribute(attribute)
|
||||
qualified_column = "#{table_name}.#{column.name}"
|
||||
|
||||
if value =~ /\*/
|
||||
where("lower(#{qualified_column}) LIKE :value ESCAPE E'\\\\'", value: value.mb_chars.downcase.to_escaped_for_sql_like)
|
||||
elsif index_column.present?
|
||||
where("#{table_name}.#{index_column} @@ plainto_tsquery(:ts_config, :value)", ts_config: ts_config, value: value)
|
||||
else
|
||||
where("to_tsvector(:ts_config, #{qualified_column}) @@ plainto_tsquery(:ts_config, :value)", ts_config: ts_config, value: value)
|
||||
end
|
||||
end
|
||||
|
||||
def search_attributes(params, *attributes)
|
||||
attributes.reduce(all) do |relation, attribute|
|
||||
relation.search_attribute(attribute, params)
|
||||
end
|
||||
end
|
||||
|
||||
def search_attribute(name, params)
|
||||
column = column_for_attribute(name)
|
||||
type = column.type || reflect_on_association(name)&.class_name
|
||||
|
||||
if column.try(:array?)
|
||||
return search_array_attribute(name, type, params)
|
||||
end
|
||||
|
||||
case type
|
||||
when "User"
|
||||
search_user_attribute(name, params)
|
||||
when "Post"
|
||||
search_post_id_attribute(params)
|
||||
when :string, :text
|
||||
search_text_attribute(name, params)
|
||||
when :boolean
|
||||
search_boolean_attribute(name, params)
|
||||
when :integer, :datetime
|
||||
numeric_attribute_matches(name, params[name])
|
||||
when :inet
|
||||
search_inet_attribute(name, params)
|
||||
else
|
||||
raise NotImplementedError, "unhandled attribute type"
|
||||
end
|
||||
end
|
||||
|
||||
def search_text_attribute(attr, params, **options)
|
||||
if params[attr].present?
|
||||
where(attr => params[attr])
|
||||
elsif params[:"#{attr}_eq"].present?
|
||||
where(attr => params[:"#{attr}_eq"])
|
||||
elsif params[:"#{attr}_not_eq"].present?
|
||||
where.not(attr => params[:"#{attr}_not_eq"])
|
||||
elsif params[:"#{attr}_like"].present?
|
||||
where_like(attr, params[:"#{attr}_like"])
|
||||
elsif params[:"#{attr}_ilike"].present?
|
||||
where_ilike(attr, params[:"#{attr}_ilike"])
|
||||
elsif params[:"#{attr}_not_like"].present?
|
||||
where_not_like(attr, params[:"#{attr}_not_like"])
|
||||
elsif params[:"#{attr}_not_ilike"].present?
|
||||
where_not_ilike(attr, params[:"#{attr}_not_ilike"])
|
||||
elsif params[:"#{attr}_regex"].present?
|
||||
where_regex(attr, params[:"#{attr}_regex"])
|
||||
elsif params[:"#{attr}_not_regex"].present?
|
||||
where_not_regex(attr, params[:"#{attr}_not_regex"])
|
||||
elsif params[:"#{attr}_array"].present?
|
||||
where(attr => params[:"#{attr}_array"])
|
||||
elsif params[:"#{attr}_comma"].present?
|
||||
where(attr => params[:"#{attr}_comma"].split(','))
|
||||
elsif params[:"#{attr}_space"].present?
|
||||
where(attr => params[:"#{attr}_space"].split(' '))
|
||||
elsif params[:"#{attr}_lower_array"].present?
|
||||
where_text_includes_lower(attr, params[:"#{attr}_lower_array"])
|
||||
elsif params[:"#{attr}_lower_comma"].present?
|
||||
where_text_includes_lower(attr, params[:"#{attr}_lower_comma"].split(','))
|
||||
elsif params[:"#{attr}_lower_space"].present?
|
||||
where_text_includes_lower(attr, params[:"#{attr}_lower_space"].split(' '))
|
||||
else
|
||||
all
|
||||
end
|
||||
end
|
||||
|
||||
def search_user_attribute(attr, params)
|
||||
if params["#{attr}_id"]
|
||||
search_attribute("#{attr}_id", params)
|
||||
elsif params["#{attr}_name"]
|
||||
where(attr => User.search(name_matches: params["#{attr}_name"]).reorder(nil))
|
||||
elsif params[attr]
|
||||
where(attr => User.search(params[attr]).reorder(nil))
|
||||
else
|
||||
all
|
||||
end
|
||||
end
|
||||
|
||||
def search_post_id_attribute(params)
|
||||
relation = all
|
||||
|
||||
if params[:post_id].present?
|
||||
relation = relation.search_attribute(:post_id, params)
|
||||
end
|
||||
|
||||
if params[:post_tags_match].present?
|
||||
relation = relation.where(post_id: Post.tag_match(params[:post_tags_match]).reorder(nil))
|
||||
end
|
||||
|
||||
relation
|
||||
end
|
||||
|
||||
def search_array_attribute(name, type, params)
|
||||
relation = all
|
||||
|
||||
if params[:"#{name}_include_any"]
|
||||
items = params[:"#{name}_include_any"].to_s.scan(/[^[:space:]]+/)
|
||||
items = items.map(&:to_i) if type == :integer
|
||||
|
||||
relation = relation.where_array_includes_any(name, items)
|
||||
elsif params[:"#{name}_include_all"]
|
||||
items = params[:"#{name}_include_all"].to_s.scan(/[^[:space:]]+/)
|
||||
items = items.map(&:to_i) if type == :integer
|
||||
|
||||
relation = relation.where_array_includes_all(name, items)
|
||||
elsif params[:"#{name}_include_any_array"]
|
||||
relation = relation.where_array_includes_any(name, params[:"#{name}_include_any_array"])
|
||||
elsif params[:"#{name}_include_all_array"]
|
||||
relation = relation.where_array_includes_all(name, params[:"#{name}_include_all_array"])
|
||||
elsif params[:"#{name}_include_any_lower"]
|
||||
items = params[:"#{name}_include_any_lower"].to_s.scan(/[^[:space:]]+/)
|
||||
items = items.map(&:to_i) if type == :integer
|
||||
|
||||
relation = relation.where_array_includes_any_lower(name, items)
|
||||
elsif params[:"#{name}_include_all_lower"]
|
||||
items = params[:"#{name}_include_all_lower"].to_s.scan(/[^[:space:]]+/)
|
||||
items = items.map(&:to_i) if type == :integer
|
||||
|
||||
relation = relation.where_array_includes_all_lower(name, items)
|
||||
elsif params[:"#{name}_include_any_lower_array"]
|
||||
relation = relation.where_array_includes_any_lower(name, params[:"#{name}_include_any_lower_array"])
|
||||
elsif params[:"#{name}_include_all_lower_array"]
|
||||
relation = relation.where_array_includes_all_lower(name, params[:"#{name}_include_all_lower_array"])
|
||||
end
|
||||
|
||||
if params[:"#{name.to_s.singularize}_count"]
|
||||
relation = relation.where_array_count(name, params[:"#{name.to_s.singularize}_count"])
|
||||
end
|
||||
|
||||
relation
|
||||
end
|
||||
|
||||
def apply_default_order(params)
|
||||
if params[:order] == "custom"
|
||||
parse_ids = Tag.parse_helper(params[:id])
|
||||
if parse_ids[0] == :in
|
||||
return find_ordered(parse_ids[1])
|
||||
end
|
||||
end
|
||||
return default_order
|
||||
end
|
||||
|
||||
def default_order
|
||||
order(id: :desc)
|
||||
end
|
||||
|
||||
def find_ordered(ids)
|
||||
order_clause = []
|
||||
ids.each do |id|
|
||||
order_clause << sanitize_sql_array(["ID=? DESC", id])
|
||||
end
|
||||
where(id: ids).order(Arel.sql(order_clause.join(', ')))
|
||||
end
|
||||
|
||||
def search(params = {})
|
||||
params ||= {}
|
||||
|
||||
default_attributes = (attribute_names.map(&:to_sym) & %i[id created_at updated_at])
|
||||
search_attributes(params, *default_attributes)
|
||||
end
|
||||
|
||||
private
|
||||
|
||||
def qualified_column_for(attr)
|
||||
if attr.is_a?(Symbol)
|
||||
"#{table_name}.#{column_for_attribute(attr).name}"
|
||||
else
|
||||
attr.to_s
|
||||
end
|
||||
end
|
||||
end
|
||||
@@ -1,40 +0,0 @@
|
||||
module Danbooru
|
||||
module HasBitFlags
|
||||
extend ActiveSupport::Concern
|
||||
|
||||
module ClassMethods
|
||||
# NOTE: the ordering of attributes has to be fixed#
|
||||
# new attributes should be appended to the end.
|
||||
def has_bit_flags(attributes, field: :bit_flags)
|
||||
attributes.each.with_index do |attribute, i|
|
||||
bit_flag = 1 << i
|
||||
|
||||
define_method(attribute) do
|
||||
send(field) & bit_flag > 0
|
||||
end
|
||||
|
||||
define_method("#{attribute}?") do
|
||||
send(field) & bit_flag > 0
|
||||
end
|
||||
|
||||
define_method("#{attribute}=") do |val|
|
||||
if val.to_s =~ /t|1|y/
|
||||
send("#{field}=", send(field) | bit_flag)
|
||||
else
|
||||
send("#{field}=", send(field) & ~bit_flag)
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
# bit_prefs_match
|
||||
define_singleton_method("#{field}_match") do |flag, value|
|
||||
value = value ? 1 : 0
|
||||
bits = attributes.length
|
||||
bit_index = bits - attributes.index(flag.to_s) - 1
|
||||
|
||||
where(sanitize_sql(["get_bit(#{field}::bit(?), ?) = ?", bits, bit_index, value]))
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
@@ -16,296 +16,6 @@ class ApplicationRecord < ActiveRecord::Base
|
||||
end
|
||||
end
|
||||
|
||||
concerning :SearchMethods do
|
||||
class_methods do
|
||||
def qualified_column_for(attr)
|
||||
if attr.is_a?(Symbol)
|
||||
"#{table_name}.#{column_for_attribute(attr).name}"
|
||||
else
|
||||
attr.to_s
|
||||
end
|
||||
end
|
||||
|
||||
def where_like(attr, value)
|
||||
where("#{qualified_column_for(attr)} LIKE ? ESCAPE E'\\\\'", value.to_escaped_for_sql_like)
|
||||
end
|
||||
|
||||
def where_not_like(attr, value)
|
||||
where.not("#{qualified_column_for(attr)} LIKE ? ESCAPE E'\\\\'", value.to_escaped_for_sql_like)
|
||||
end
|
||||
|
||||
def where_ilike(attr, value)
|
||||
where("#{qualified_column_for(attr)} ILIKE ? ESCAPE E'\\\\'", value.mb_chars.to_escaped_for_sql_like)
|
||||
end
|
||||
|
||||
def where_not_ilike(attr, value)
|
||||
where.not("#{qualified_column_for(attr)} ILIKE ? ESCAPE E'\\\\'", value.mb_chars.to_escaped_for_sql_like)
|
||||
end
|
||||
|
||||
def where_iequals(attr, value)
|
||||
where_ilike(attr, value.gsub(/\\/, '\\\\').gsub(/\*/, '\*'))
|
||||
end
|
||||
|
||||
# https://www.postgresql.org/docs/current/static/functions-matching.html#FUNCTIONS-POSIX-REGEXP
|
||||
# "(?e)" means force use of ERE syntax; see sections 9.7.3.1 and 9.7.3.4.
|
||||
def where_regex(attr, value)
|
||||
where("#{qualified_column_for(attr)} ~ ?", "(?e)" + value)
|
||||
end
|
||||
|
||||
def where_not_regex(attr, value)
|
||||
where.not("#{qualified_column_for(attr)} ~ ?", "(?e)" + value)
|
||||
end
|
||||
|
||||
def where_inet_matches(attr, value)
|
||||
if value.match?(/[, ]/)
|
||||
ips = value.split(/[, ]+/).map { |ip| IPAddress.parse(ip).to_string }
|
||||
where("#{qualified_column_for(attr)} = ANY(ARRAY[?]::inet[])", ips)
|
||||
else
|
||||
ip = IPAddress.parse(value)
|
||||
where("#{qualified_column_for(attr)} <<= ?", ip.to_string)
|
||||
end
|
||||
end
|
||||
|
||||
def where_array_includes_any(attr, values)
|
||||
where("#{qualified_column_for(attr)} && ARRAY[?]", values)
|
||||
end
|
||||
|
||||
def where_array_includes_all(attr, values)
|
||||
where("#{qualified_column_for(attr)} @> ARRAY[?]", values)
|
||||
end
|
||||
|
||||
def where_array_includes_any_lower(attr, values)
|
||||
where("lower(#{qualified_column_for(attr)}::text)::text[] && ARRAY[?]", values.map(&:downcase))
|
||||
end
|
||||
|
||||
def where_array_includes_all_lower(attr, values)
|
||||
where("lower(#{qualified_column_for(attr)}::text)::text[] @> ARRAY[?]", values.map(&:downcase))
|
||||
end
|
||||
|
||||
def where_text_includes_lower(attr, values)
|
||||
where("lower(#{qualified_column_for(attr)}) IN (?)", values.map(&:downcase))
|
||||
end
|
||||
|
||||
def where_array_count(attr, value)
|
||||
relation = all
|
||||
qualified_column = "cardinality(#{qualified_column_for(attr)})"
|
||||
parsed_range = Tag.parse_helper(value, :integer)
|
||||
|
||||
PostQueryBuilder.new(nil).add_range_relation(parsed_range, qualified_column, relation)
|
||||
end
|
||||
|
||||
def search_boolean_attribute(attribute, params)
|
||||
return all unless params.key?(attribute)
|
||||
|
||||
value = params[attribute].to_s
|
||||
if value.truthy?
|
||||
where(attribute => true)
|
||||
elsif value.falsy?
|
||||
where(attribute => false)
|
||||
else
|
||||
raise ArgumentError, "value must be truthy or falsy"
|
||||
end
|
||||
end
|
||||
|
||||
def search_inet_attribute(attr, params)
|
||||
if params[attr].present?
|
||||
where_inet_matches(attr, params[attr])
|
||||
else
|
||||
all
|
||||
end
|
||||
end
|
||||
|
||||
# range: "5", ">5", "<5", ">=5", "<=5", "5..10", "5,6,7"
|
||||
def numeric_attribute_matches(attribute, range)
|
||||
return all unless range.present?
|
||||
|
||||
column = column_for_attribute(attribute)
|
||||
qualified_column = "#{table_name}.#{column.name}"
|
||||
parsed_range = Tag.parse_helper(range, column.type)
|
||||
|
||||
PostQueryBuilder.new(nil).add_range_relation(parsed_range, qualified_column, self)
|
||||
end
|
||||
|
||||
def text_attribute_matches(attribute, value, index_column: nil, ts_config: "english")
|
||||
return all unless value.present?
|
||||
|
||||
column = column_for_attribute(attribute)
|
||||
qualified_column = "#{table_name}.#{column.name}"
|
||||
|
||||
if value =~ /\*/
|
||||
where("lower(#{qualified_column}) LIKE :value ESCAPE E'\\\\'", value: value.mb_chars.downcase.to_escaped_for_sql_like)
|
||||
elsif index_column.present?
|
||||
where("#{table_name}.#{index_column} @@ plainto_tsquery(:ts_config, :value)", ts_config: ts_config, value: value)
|
||||
else
|
||||
where("to_tsvector(:ts_config, #{qualified_column}) @@ plainto_tsquery(:ts_config, :value)", ts_config: ts_config, value: value)
|
||||
end
|
||||
end
|
||||
|
||||
def search_attributes(params, *attributes)
|
||||
attributes.reduce(all) do |relation, attribute|
|
||||
relation.search_attribute(attribute, params)
|
||||
end
|
||||
end
|
||||
|
||||
def search_attribute(name, params)
|
||||
column = column_for_attribute(name)
|
||||
type = column.type || reflect_on_association(name)&.class_name
|
||||
|
||||
if column.try(:array?)
|
||||
return search_array_attribute(name, type, params)
|
||||
end
|
||||
|
||||
case type
|
||||
when "User"
|
||||
search_user_attribute(name, params)
|
||||
when "Post"
|
||||
search_post_id_attribute(params)
|
||||
when :string, :text
|
||||
search_text_attribute(name, params)
|
||||
when :boolean
|
||||
search_boolean_attribute(name, params)
|
||||
when :integer, :datetime
|
||||
numeric_attribute_matches(name, params[name])
|
||||
when :inet
|
||||
search_inet_attribute(name, params)
|
||||
else
|
||||
raise NotImplementedError, "unhandled attribute type"
|
||||
end
|
||||
end
|
||||
|
||||
def search_text_attribute(attr, params, **options)
|
||||
if params[attr].present?
|
||||
where(attr => params[attr])
|
||||
elsif params[:"#{attr}_eq"].present?
|
||||
where(attr => params[:"#{attr}_eq"])
|
||||
elsif params[:"#{attr}_not_eq"].present?
|
||||
where.not(attr => params[:"#{attr}_not_eq"])
|
||||
elsif params[:"#{attr}_like"].present?
|
||||
where_like(attr, params[:"#{attr}_like"])
|
||||
elsif params[:"#{attr}_ilike"].present?
|
||||
where_ilike(attr, params[:"#{attr}_ilike"])
|
||||
elsif params[:"#{attr}_not_like"].present?
|
||||
where_not_like(attr, params[:"#{attr}_not_like"])
|
||||
elsif params[:"#{attr}_not_ilike"].present?
|
||||
where_not_ilike(attr, params[:"#{attr}_not_ilike"])
|
||||
elsif params[:"#{attr}_regex"].present?
|
||||
where_regex(attr, params[:"#{attr}_regex"])
|
||||
elsif params[:"#{attr}_not_regex"].present?
|
||||
where_not_regex(attr, params[:"#{attr}_not_regex"])
|
||||
elsif params[:"#{attr}_array"].present?
|
||||
where(attr => params[:"#{attr}_array"])
|
||||
elsif params[:"#{attr}_comma"].present?
|
||||
where(attr => params[:"#{attr}_comma"].split(','))
|
||||
elsif params[:"#{attr}_space"].present?
|
||||
where(attr => params[:"#{attr}_space"].split(' '))
|
||||
elsif params[:"#{attr}_lower_array"].present?
|
||||
where_text_includes_lower(attr, params[:"#{attr}_lower_array"])
|
||||
elsif params[:"#{attr}_lower_comma"].present?
|
||||
where_text_includes_lower(attr, params[:"#{attr}_lower_comma"].split(','))
|
||||
elsif params[:"#{attr}_lower_space"].present?
|
||||
where_text_includes_lower(attr, params[:"#{attr}_lower_space"].split(' '))
|
||||
else
|
||||
all
|
||||
end
|
||||
end
|
||||
|
||||
def search_user_attribute(attr, params)
|
||||
if params["#{attr}_id"]
|
||||
search_attribute("#{attr}_id", params)
|
||||
elsif params["#{attr}_name"]
|
||||
where(attr => User.search(name_matches: params["#{attr}_name"]).reorder(nil))
|
||||
elsif params[attr]
|
||||
where(attr => User.search(params[attr]).reorder(nil))
|
||||
else
|
||||
all
|
||||
end
|
||||
end
|
||||
|
||||
def search_post_id_attribute(params)
|
||||
relation = all
|
||||
|
||||
if params[:post_id].present?
|
||||
relation = relation.search_attribute(:post_id, params)
|
||||
end
|
||||
|
||||
if params[:post_tags_match].present?
|
||||
relation = relation.where(post_id: Post.tag_match(params[:post_tags_match]).reorder(nil))
|
||||
end
|
||||
|
||||
relation
|
||||
end
|
||||
|
||||
def search_array_attribute(name, type, params)
|
||||
relation = all
|
||||
|
||||
if params[:"#{name}_include_any"]
|
||||
items = params[:"#{name}_include_any"].to_s.scan(/[^[:space:]]+/)
|
||||
items = items.map(&:to_i) if type == :integer
|
||||
|
||||
relation = relation.where_array_includes_any(name, items)
|
||||
elsif params[:"#{name}_include_all"]
|
||||
items = params[:"#{name}_include_all"].to_s.scan(/[^[:space:]]+/)
|
||||
items = items.map(&:to_i) if type == :integer
|
||||
|
||||
relation = relation.where_array_includes_all(name, items)
|
||||
elsif params[:"#{name}_include_any_array"]
|
||||
relation = relation.where_array_includes_any(name, params[:"#{name}_include_any_array"])
|
||||
elsif params[:"#{name}_include_all_array"]
|
||||
relation = relation.where_array_includes_all(name, params[:"#{name}_include_all_array"])
|
||||
elsif params[:"#{name}_include_any_lower"]
|
||||
items = params[:"#{name}_include_any_lower"].to_s.scan(/[^[:space:]]+/)
|
||||
items = items.map(&:to_i) if type == :integer
|
||||
|
||||
relation = relation.where_array_includes_any_lower(name, items)
|
||||
elsif params[:"#{name}_include_all_lower"]
|
||||
items = params[:"#{name}_include_all_lower"].to_s.scan(/[^[:space:]]+/)
|
||||
items = items.map(&:to_i) if type == :integer
|
||||
|
||||
relation = relation.where_array_includes_all_lower(name, items)
|
||||
elsif params[:"#{name}_include_any_lower_array"]
|
||||
relation = relation.where_array_includes_any_lower(name, params[:"#{name}_include_any_lower_array"])
|
||||
elsif params[:"#{name}_include_all_lower_array"]
|
||||
relation = relation.where_array_includes_all_lower(name, params[:"#{name}_include_all_lower_array"])
|
||||
end
|
||||
|
||||
if params[:"#{name.to_s.singularize}_count"]
|
||||
relation = relation.where_array_count(name, params[:"#{name.to_s.singularize}_count"])
|
||||
end
|
||||
|
||||
relation
|
||||
end
|
||||
|
||||
def apply_default_order(params)
|
||||
if params[:order] == "custom"
|
||||
parse_ids = Tag.parse_helper(params[:id])
|
||||
if parse_ids[0] == :in
|
||||
return find_ordered(parse_ids[1])
|
||||
end
|
||||
end
|
||||
return default_order
|
||||
end
|
||||
|
||||
def default_order
|
||||
order(id: :desc)
|
||||
end
|
||||
|
||||
def find_ordered(ids)
|
||||
order_clause = []
|
||||
ids.each do |id|
|
||||
order_clause << sanitize_sql_array(["ID=? DESC", id])
|
||||
end
|
||||
where(id: ids).order(Arel.sql(order_clause.join(', ')))
|
||||
end
|
||||
|
||||
def search(params = {})
|
||||
params ||= {}
|
||||
|
||||
default_attributes = (attribute_names.map(&:to_sym) & %i[id created_at updated_at])
|
||||
search_attributes(params, *default_attributes)
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
module ApiMethods
|
||||
extend ActiveSupport::Concern
|
||||
|
||||
@@ -467,4 +177,7 @@ class ApplicationRecord < ActiveRecord::Base
|
||||
end
|
||||
|
||||
include ApiMethods
|
||||
include Mentionable
|
||||
extend HasBitFlags
|
||||
extend Searchable
|
||||
end
|
||||
|
||||
@@ -1,6 +1,4 @@
|
||||
class Comment < ApplicationRecord
|
||||
include Mentionable
|
||||
|
||||
validate :validate_creator_is_not_limited, :on => :create
|
||||
validates_presence_of :body, :message => "has no content"
|
||||
belongs_to :post
|
||||
|
||||
@@ -1,6 +1,4 @@
|
||||
class ForumPost < ApplicationRecord
|
||||
include Mentionable
|
||||
|
||||
attr_readonly :topic_id
|
||||
belongs_to :creator, class_name: "User"
|
||||
belongs_to_updater
|
||||
|
||||
@@ -1,5 +1,3 @@
|
||||
require 'danbooru/has_bit_flags'
|
||||
|
||||
class Post < ApplicationRecord
|
||||
class ApprovalError < StandardError; end
|
||||
class DisapprovalError < StandardError; end
|
||||
@@ -1694,13 +1692,8 @@ class Post < ApplicationRecord
|
||||
include PixivMethods
|
||||
include IqdbMethods
|
||||
include ValidationMethods
|
||||
include Danbooru::HasBitFlags
|
||||
|
||||
BOOLEAN_ATTRIBUTES = %w(
|
||||
has_embedded_notes
|
||||
has_cropped
|
||||
)
|
||||
has_bit_flags BOOLEAN_ATTRIBUTES
|
||||
has_bit_flags ["has_embedded_notes", "has_cropped"]
|
||||
|
||||
def safeblocked?
|
||||
CurrentUser.safe_mode? && (rating != "s" || Danbooru.config.safe_mode_restricted_tags.any? { |tag| tag.in?(tag_array) })
|
||||
|
||||
@@ -1,5 +1,4 @@
|
||||
require 'digest/sha1'
|
||||
require 'danbooru/has_bit_flags'
|
||||
|
||||
class User < ApplicationRecord
|
||||
class Error < StandardError; end
|
||||
@@ -67,7 +66,6 @@ class User < ApplicationRecord
|
||||
no_feedback
|
||||
)
|
||||
|
||||
include Danbooru::HasBitFlags
|
||||
has_bit_flags BOOLEAN_ATTRIBUTES, :field => "bit_prefs"
|
||||
|
||||
attr_accessor :password, :old_password
|
||||
|
||||
@@ -29,7 +29,7 @@ module Danbooru
|
||||
config.filter_parameters += [:password, :password_confirmation, :password_hash, :api_key]
|
||||
# config.assets.enabled = true
|
||||
# config.assets.version = '1.0'
|
||||
config.autoload_paths += %W(#{config.root}/app/presenters #{config.root}/app/logical #{config.root}/app/mailers)
|
||||
config.autoload_paths += %W(#{config.root}/app/presenters #{config.root}/app/logical/concerns #{config.root}/app/logical #{config.root}/app/mailers)
|
||||
config.plugins = [:all]
|
||||
config.time_zone = 'Eastern Time (US & Canada)'
|
||||
|
||||
|
||||
Reference in New Issue
Block a user