scinote-web/app/models/table.rb

100 lines
3 KiB
Ruby
Raw Normal View History

2016-02-12 23:52:43 +08:00
class Table < ActiveRecord::Base
2016-07-21 19:11:15 +08:00
include SearchableModel
2017-01-17 00:11:08 +08:00
auto_strip_attributes :name, nullify: false
validates :name,
length: { maximum: Constants::NAME_MAX_LENGTH }
2016-02-12 23:52:43 +08:00
validates :contents,
presence: true,
length: { maximum: Constants::TABLE_JSON_MAX_SIZE_MB.megabytes }
2016-02-12 23:52:43 +08:00
belongs_to :created_by, foreign_key: 'created_by_id', class_name: 'User'
belongs_to :last_modified_by, foreign_key: 'last_modified_by_id', class_name: 'User'
has_one :step_table, inverse_of: :table
has_one :step, through: :step_table
has_one :result_table, inverse_of: :table
has_one :result, through: :result_table
has_many :report_elements, inverse_of: :table, dependent: :destroy
after_save :update_ts_index
2016-07-21 19:11:15 +08:00
#accepts_nested_attributes_for :table
def self.search(user, include_archived, query = nil, page = 1)
step_ids =
Step
.search(user, include_archived, nil, Constants::SEARCH_NO_LIMIT)
2016-07-21 19:11:15 +08:00
.joins(:step_tables)
.select("step_tables.id")
.distinct
result_ids =
Result
.search(user, include_archived, nil, Constants::SEARCH_NO_LIMIT)
2016-07-21 19:11:15 +08:00
.joins(:result_table)
.select("result_tables.id")
.distinct
2017-01-18 21:05:57 +08:00
if query
a_query = query.strip
.gsub('_', '\\_')
.gsub('%', '\\%')
.split(/\s+/)
.map { |t| '%' + t + '%' }
else
a_query = query
end
2016-07-21 19:11:15 +08:00
# Trim whitespace and replace it with OR character. Make prefixed
# wildcard search term and escape special characters.
# For example, search term 'demo project' is transformed to
# 'demo:*|project:*' which makes word inclusive search with postfix
# wildcard.
s_query = query.gsub(/[!()&|:]/, " ")
.strip
.split(/\s+/)
.map {|t| t + ":*" }
.join("|")
.gsub('\'', '"')
2017-01-18 21:05:57 +08:00
table_query =
Table
2016-07-21 19:11:15 +08:00
.distinct
.joins("LEFT OUTER JOIN step_tables ON step_tables.table_id = tables.id")
.joins("LEFT OUTER JOIN result_tables ON result_tables.table_id = tables.id")
.joins("LEFT OUTER JOIN results ON result_tables.result_id = results.id")
.where("step_tables.id IN (?) OR result_tables.id IN (?)", step_ids, result_ids)
2017-01-18 21:05:57 +08:00
.where(
'(tables.name ILIKE ANY (array[?])'\
'OR tables.data_vector @@ to_tsquery(?))',
a_query,
s_query
)
2016-07-21 19:11:15 +08:00
new_query = table_query
# Show all results if needed
if page == Constants::SEARCH_NO_LIMIT
2016-07-21 19:11:15 +08:00
new_query
else
new_query
.limit(Constants::SEARCH_LIMIT)
.offset((page - 1) * Constants::SEARCH_LIMIT)
2016-07-21 19:11:15 +08:00
end
end
2016-02-12 23:52:43 +08:00
def contents_utf_8
contents.present? ? contents.force_encoding(Encoding::UTF_8) : nil
end
def update_ts_index
if contents_changed?
sql = "UPDATE tables " +
"SET data_vector = " +
"to_tsvector(substring(encode(contents::bytea, 'escape'), 9)) " +
"WHERE id = " + Integer(id).to_s
Table.connection.execute(sql)
end
end
end