scinote-web/app/models/asset.rb

362 lines
10 KiB
Ruby
Raw Normal View History

2016-02-12 23:52:43 +08:00
class Asset < ActiveRecord::Base
include SearchableModel
include DatabaseHelper
2016-07-21 19:11:15 +08:00
include Encryptor
2016-02-12 23:52:43 +08:00
require 'tempfile'
# Paperclip validation
2016-12-08 18:26:13 +08:00
has_attached_file :file,
styles: { medium: [Constants::MEDIUM_PIC_FORMAT, :jpg] }
2016-02-12 23:52:43 +08:00
2016-09-16 21:41:31 +08:00
validates_attachment :file,
presence: true,
2016-10-07 00:36:55 +08:00
size: {
less_than: Constants::FILE_MAX_SIZE_MB.megabytes
}
2016-02-12 23:52:43 +08:00
validates :estimated_size, presence: true
validates :file_present, inclusion: { in: [true, false] }
# Should be checked for any security leaks
do_not_validate_attachment_file_type :file
before_file_post_process :is_image?
2016-12-08 18:26:13 +08:00
process_in_background :file
2016-02-12 23:52:43 +08:00
# Asset validation
# This could cause some problems if you create empty asset and want to
# assign it to result
validate :step_or_result
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_asset,
inverse_of: :asset,
dependent: :destroy
has_one :step, through: :step_asset,
dependent: :nullify
has_one :result_asset,
inverse_of: :asset,
dependent: :destroy
has_one :result, through: :result_asset,
dependent: :nullify
has_many :report_elements, inverse_of: :asset, dependent: :destroy
has_one :asset_text_datum, inverse_of: :asset, dependent: :destroy
# Specific file errors propagate to "file" error hash key,
# so use just these errors
after_validation :filter_paperclip_errors
# Needed because Paperclip validatates on creation
after_initialize :filter_paperclip_errors, if: :new_record?
2016-07-21 19:11:15 +08:00
attr_accessor :file_content, :file_info, :preview_cached
2016-02-12 23:52:43 +08:00
def file_empty(name, size)
file_ext = name.split(".").last
self.file_file_name = name
self.file_content_type = Rack::Mime.mime_type(".#{file_ext}")
self.file_file_size = size
self.file_updated_at = DateTime.now
self.file_present = false
end
def self.new_empty(name, size)
asset = self.new
asset.file_empty name, size
asset
end
def self.search(
user,
include_archived,
query = nil,
page = 1
)
step_ids =
Step
.search(user, include_archived, nil, Constants::SEARCH_NO_LIMIT)
2016-02-12 23:52:43 +08:00
.joins(:step_assets)
.select("step_assets.id")
.distinct
result_ids =
Result
.search(user, include_archived, nil, Constants::SEARCH_NO_LIMIT)
2016-02-12 23:52:43 +08:00
.joins(:result_asset)
.select("result_assets.id")
.distinct
2016-07-21 19:11:15 +08:00
if query
a_query = query.strip
.gsub("_","\\_")
.gsub("%","\\%")
.split(/\s+/)
.map {|t| "%" + t + "%" }
else
a_query = query
end
# 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('\'', '"')
ids = Asset
.select(:id)
2016-02-12 23:52:43 +08:00
.distinct
.joins("LEFT OUTER JOIN step_assets ON step_assets.asset_id = assets.id")
.joins("LEFT OUTER JOIN result_assets ON result_assets.asset_id = assets.id")
2016-07-21 19:11:15 +08:00
.joins("LEFT JOIN asset_text_data ON assets.id = asset_text_data.asset_id")
.where("(step_assets.id IN (?) OR result_assets.id IN (?))", step_ids, result_ids)
2016-02-12 23:52:43 +08:00
.where(
2016-07-21 19:11:15 +08:00
"(assets.file_file_name ILIKE ANY (array[?]) " +
"OR asset_text_data.data_vector @@ to_tsquery(?))",
a_query,
s_query
2016-02-12 23:52:43 +08:00
)
# Show all results if needed
if page != Constants::SEARCH_NO_LIMIT
2016-07-21 19:11:15 +08:00
ids = ids
2016-10-07 00:36:55 +08:00
.limit(Constants::SEARCH_LIMIT)
.offset((page - 1) * Constants::SEARCH_LIMIT)
2016-02-12 23:52:43 +08:00
end
2016-07-21 19:11:15 +08:00
Asset
.joins("LEFT JOIN asset_text_data ON assets.id = asset_text_data.asset_id")
.select("assets.*")
.select("ts_headline(data, to_tsquery('" + s_query + "'),
'StartSel=<mark>, StopSel=</mark>') headline")
.where("assets.id IN (?)", ids)
2016-02-12 23:52:43 +08:00
end
def is_image?
%r{^image/#{Regexp.union(Constants::WHITELISTED_IMAGE_TYPES)}} ===
file.content_type
2016-02-12 23:52:43 +08:00
end
2016-07-21 19:11:15 +08:00
def text?
Constants::TEXT_EXTRACT_FILE_TYPES.any? do |v|
file_content_type.start_with? v
end
end
2016-02-12 23:52:43 +08:00
# TODO: get the current_user
# before_save do
# if current_user
# self.created_by ||= current_user
# self.last_modified_by = current_user if self.changed?
# end
# end
def is_stored_on_s3?
file.options[:storage].to_sym == :s3
end
def post_process_file(org = nil)
# Update self.empty
self.update(file_present: true)
# Extract asset text if it's of correct type
if text?
2016-02-12 23:52:43 +08:00
Rails.logger.info "Asset #{id}: Creating extract text job"
# The extract_asset_text also includes
# estimated size calculation
delay(queue: :assets).extract_asset_text(org)
else
# Update asset's estimated size immediately
update_estimated_size(org)
end
end
def extract_asset_text(org = nil)
if file.blank?
return
end
begin
file_path = file.path
if file.is_stored_on_s3?
fa = file.fetch
file_path = fa.path
end
2016-07-21 19:11:15 +08:00
if (!Yomu.class_eval('@@server_pid'))
Yomu.server(:text,nil)
sleep(5)
end
2016-02-12 23:52:43 +08:00
y = Yomu.new file_path
2016-07-21 19:11:15 +08:00
2016-02-12 23:52:43 +08:00
text_data = y.text
if asset_text_datum.present?
# Update existing text datum if it exists
asset_text_datum.update(data: text_data)
else
# Create new text datum
AssetTextDatum.create(data: text_data, asset: self)
end
Rails.logger.info "Asset #{id}: Asset file successfully extracted"
# Finally, update asset's estimated size to include
# the data vector
update_estimated_size(org)
rescue Exception => e
Rails.logger.fatal "Asset #{id}: Error extracting contents from asset file #{file.path}: " + e.message
ensure
File.delete file_path if fa
end
end
def destroy
report_elements.destroy_all
asset_text_datum.destroy if asset_text_datum.present?
# Nullify needed to force paperclip file deletion
self.file = nil
save
2016-08-18 02:49:20 +08:00
delete
end
2016-02-12 23:52:43 +08:00
# If organization is provided, its space_taken
# is updated as well
def update_estimated_size(org = nil)
if file_file_size.blank?
return
end
es = file_file_size
if asset_text_datum.present? and asset_text_datum.persisted? then
asset_text_datum.reload
es += get_octet_length_record(asset_text_datum, :data)
es += get_octet_length_record(asset_text_datum, :data_vector)
end
2016-10-07 00:36:55 +08:00
es *= Constants::ASSET_ESTIMATED_SIZE_FACTOR
2016-02-12 23:52:43 +08:00
update(estimated_size: es)
Rails.logger.info "Asset #{id}: Estimated size successfully calculated"
# Finally, update organization's space
if org.present?
org.take_space(es)
org.save
end
end
def url(style = :original, timeout: Constants::URL_SHORT_EXPIRE_TIME)
if file.is_stored_on_s3?
presigned_url(style, timeout: timeout)
else
file.url(style)
end
end
# When using S3 file upload, we can limit file accessibility with url signing
def presigned_url(style = :original,
download: false,
timeout: Constants::URL_SHORT_EXPIRE_TIME)
2016-02-12 23:52:43 +08:00
if file.is_stored_on_s3?
2016-08-18 02:49:20 +08:00
if download
download_arg = 'attachment; filename=' + URI.escape(file_file_name)
else
download_arg = nil
end
2016-02-12 23:52:43 +08:00
signer = Aws::S3::Presigner.new(client: S3_BUCKET.client)
signer.presigned_url(:get_object,
bucket: S3_BUCKET.name,
key: file.path(style)[1..-1],
expires_in: timeout,
2016-02-12 23:52:43 +08:00
# this response header forces object download
2016-08-18 02:49:20 +08:00
response_content_disposition: download_arg)
2016-02-12 23:52:43 +08:00
end
end
def open
if file.is_stored_on_s3?
Kernel.open(presigned_url, "rb")
else
File.open(file.path, "rb")
end
end
2016-07-21 19:11:15 +08:00
# Preserving attachments (on client-side) between failed validations (only usable for small/few files!)
# Needs to be called before save method and view needs to have :file_content and :file_info hidden field
# If file is an image, it can be viewed on front-end using @preview_cached with image_tag tag
def preserve(file_data)
if file_data[:file_content].present?
restore_cached(file_data[:file_content], file_data[:file_info])
end
cache
end
2016-02-12 23:52:43 +08:00
private
def filter_paperclip_errors
2016-08-18 02:49:20 +08:00
if errors.size > 1
temp_errors = errors[:file]
errors.clear
errors.set(:file, temp_errors)
end
end
2016-02-12 23:52:43 +08:00
def file_changed?
previous_changes.present? and
(
(
previous_changes.key? "file_file_name" and
previous_changes["file_file_name"].first !=
previous_changes["file_file_name"].last
) or (
previous_changes.key? "file_file_size" and
previous_changes["file_file_size"].first !=
previous_changes["file_file_size"].last
)
)
end
def step_or_result
# We must allow both step and result to be blank because of GUI
# (even though it's not really a "valid" asset)
2016-02-12 23:52:43 +08:00
if step.present? && result.present?
errors.add(:base, "Asset can only be result or step, not both.")
end
end
2016-07-21 19:11:15 +08:00
def cache
fetched_file = file.fetch
file_content = fetched_file.read
@file_content = encrypt(file_content)
@file_info = %Q[{"content_type" : "#{file.content_type}", "original_filename" : "#{file.original_filename}"}]
@file_info = encrypt(@file_info)
if !(file.content_type =~ /^image/).nil?
@preview_cached = "data:image/png;base64," + Base64.encode64(file_content)
end
end
def restore_cached(file_content, file_info)
decoded_data = decrypt(file_content)
decoded_data_info = decrypt(file_info)
decoded_data_info = JSON.parse decoded_data_info
data = StringIO.new(decoded_data)
data.class_eval do
attr_accessor :content_type, :original_filename
end
data.content_type = decoded_data_info['content_type']
data.original_filename = File.basename(decoded_data_info['original_filename'])
self.file = data
end
2016-02-12 23:52:43 +08:00
end