class ApplicationRecord < ActiveRecord::Base self.abstract_class = true include Danbooru::Paginator::ActiveRecordExtension concerning :SearchMethods do class_methods do def attribute_matches(attribute, value) return all if value.nil? column = column_for_attribute(attribute) case column.sql_type_metadata.type when :boolean boolean_attribute_matches(attribute, value) when :integer, :datetime numeric_attribute_matches(attribute, value) else raise ArgumentError, "unhandled attribute type" end end def boolean_attribute_matches(attribute, value) if value.to_s.truthy? value = true elsif value.to_s.falsy? value = false else raise ArgumentError, "value must be truthy or falsy" end where(attribute => value) end # range: "5", ">5", "<5", ">=5", "<=5", "5..10", "5,6,7" def numeric_attribute_matches(attribute, range) 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 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(order_clause.join(', ')) end def search(params = {}) params ||= {} q = all q = q.attribute_matches(:id, params[:id]) q = q.attribute_matches(:created_at, params[:created_at]) if attribute_names.include?("created_at") q = q.attribute_matches(:updated_at, params[:updated_at]) if attribute_names.include?("updated_at") q end end end module ApiMethods extend ActiveSupport::Concern def as_json(options = {}) options ||= {} options[:except] ||= [] options[:except] += hidden_attributes options[:methods] ||= [] options[:methods] += method_attributes super(options) end def to_xml(options = {}, &block) options ||= {} options[:except] ||= [] options[:except] += hidden_attributes options[:methods] ||= [] options[:methods] += method_attributes super(options, &block) end def serializable_hash(*args) hash = super(*args) hash.transform_keys { |key| key.delete("?") } end protected def hidden_attributes [:uploader_ip_addr, :updater_ip_addr, :creator_ip_addr, :ip_addr] end def method_attributes [] end end concerning :ActiveRecordExtensions do class_methods do def without_timeout connection.execute("SET STATEMENT_TIMEOUT = 0") unless Rails.env == "test" yield ensure connection.execute("SET STATEMENT_TIMEOUT = #{CurrentUser.user.try(:statement_timeout) || 3_000}") unless Rails.env == "test" end def with_timeout(n, default_value = nil, new_relic_params = {}) connection.execute("SET STATEMENT_TIMEOUT = #{n}") unless Rails.env == "test" yield rescue ::ActiveRecord::StatementInvalid => x if Rails.env.production? NewRelic::Agent.notice_error(x, :custom_params => new_relic_params.merge(:user_id => CurrentUser.id, :user_ip_addr => CurrentUser.ip_addr)) end return default_value ensure connection.execute("SET STATEMENT_TIMEOUT = #{CurrentUser.user.try(:statement_timeout) || 3_000}") unless Rails.env == "test" end end %w(execute select_value select_values select_all).each do |method_name| define_method("#{method_name}_sql") do |sql, *params| self.class.connection.__send__(method_name, self.class.send(:sanitize_sql_array, [sql, *params])) end self.class.__send__(:define_method, "#{method_name}_sql") do |sql, *params| connection.__send__(method_name, send(:sanitize_sql_array, [sql, *params])) end end end concerning :PostgresExtensions do class_methods do def columns(*params) super.reject {|x| x.sql_type == "tsvector"} end def test_connection limit(1).select(:id) return true rescue PG::Error return false end end end concerning :UserMethods do class_methods do def belongs_to_creator(options = {}) class_eval do belongs_to :creator, options.merge(class_name: "User") before_validation(on: :create) do |rec| if rec.creator_id.nil? rec.creator_id = CurrentUser.id rec.creator_ip_addr = CurrentUser.ip_addr if rec.respond_to?(:creator_ip_addr=) rec.ip_addr = CurrentUser.ip_addr if rec.respond_to?(:ip_addr=) end end define_method :creator_name do User.id_to_name(creator_id) end end end def belongs_to_updater(options = {}) class_eval do belongs_to :updater, options.merge(class_name: "User") before_validation do |rec| rec.updater_id = CurrentUser.id rec.updater_ip_addr = CurrentUser.ip_addr if rec.respond_to?(:updater_ip_addr=) end define_method :updater_name do User.id_to_name(updater_id) end end end end end def warnings @warnings ||= ActiveModel::Errors.new(self) end include ApiMethods end