redmine_dmsf/app/models/dmsf_file_revision.rb
2011-06-29 13:55:24 +00:00

207 lines
7.5 KiB
Ruby

# Redmine plugin for Document Management System "Features"
#
# Copyright (C) 2011 Vít Jonáš <vit.jonas@gmail.com>
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# of the License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
class DmsfFileRevision < ActiveRecord::Base
unloadable
belongs_to :file, :class_name => "DmsfFile", :foreign_key => "dmsf_file_id"
belongs_to :source_revision, :class_name => "DmsfFileRevision", :foreign_key => "source_dmsf_file_revision_id"
belongs_to :user
belongs_to :folder, :class_name => "DmsfFolder", :foreign_key => "dmsf_folder_id"
belongs_to :deleted_by_user, :class_name => "User", :foreign_key => "deleted_by_user_id"
acts_as_event :title => Proc.new {|o| "#{l(:label_dmsf_updated)}: #{o.file.dmsf_path_str}"},
:url => Proc.new {|o| {:controller => 'dmsf_files', :action => 'show', :id => o.file}},
:datetime => Proc.new {|o| o.updated_at },
:description => Proc.new {|o| o.comment },
:author => Proc.new {|o| o.user }
acts_as_activity_provider :type => "dmsf_files",
:timestamp => "#{DmsfFileRevision.table_name}.updated_at",
:author_key => "#{DmsfFileRevision.table_name}.user_id",
:permission => :view_dmsf_files,
:find_options => {:select => "#{DmsfFileRevision.table_name}.*",
:joins =>
"INNER JOIN #{DmsfFile.table_name} ON #{DmsfFileRevision.table_name}.dmsf_file_id = #{DmsfFile.table_name}.id " +
"INNER JOIN #{Project.table_name} ON #{DmsfFile.table_name}.project_id = #{Project.table_name}.id",
:conditions => ["#{DmsfFile.table_name}.deleted = :false", {:false => false}]
}
validates_presence_of :title
validates_presence_of :name
validates_format_of :name, :with => DmsfFolder.invalid_characters,
:message => l(:error_contains_invalid_character)
def self.remove_extension(filename)
filename[0, (filename.length - File.extname(filename).length)]
end
#TODO: check if better to move to dmsf_upload class
def self.filename_to_title(filename)
remove_extension(filename).gsub(/_+/, " ");
end
def delete(delete_all = false)
if self.file.locked_for_user?
errors.add_to_base(l(:error_file_is_locked))
return false
end
if !delete_all && self.file.revisions.length <= 1
errors.add_to_base(l(:error_at_least_one_revision_must_be_present))
return false
end
dependent = DmsfFileRevision.find(:all, :conditions =>
["source_dmsf_file_revision_id = :id and deleted = :deleted",
{:id => self.id, :deleted => false}])
dependent.each do |d|
d.source_revision = self.source_revision
d.save!
end
if Setting.plugin_redmine_dmsf["dmsf_really_delete_files"]
dependent = DmsfFileRevision.find(:all, :conditions =>
["disk_filename = :filename", {:filename => self.disk_filename}])
File.delete(self.disk_file) if dependent.length <= 1 && File.exist?(self.disk_file)
DmsfFileRevisionAccess.find(:all, :conditions => ["dmsf_file_revision_id = ?", self.id]).each {|a| a.destroy}
self.destroy
else
self.deleted = true
self.deleted_by_user = User.current
save
end
end
def self.access_grouped(revision_id)
sql = "select user_id, count(*) as count, min(created_at) as min, max(created_at) as max from #{DmsfFileRevisionAccess.table_name} where dmsf_file_revision_id = ? group by user_id"
self.connection.select_all(self.sanitize_sql_array([sql, revision_id]))
end
def access_grouped
DmsfFileRevision.access_grouped(self.id)
end
def version
"#{self.major_version}.#{self.minor_version}"
end
def project
self.file.project
end
def disk_file
"#{DmsfFile.storage_path}/#{self.disk_filename}"
end
def detect_content_type
content_type = self.mime_type
content_type = Redmine::MimeType.of(self.disk_filename) if content_type.blank?
content_type = "application/octet-stream" if content_type.blank?
content_type.to_s
end
#TODO: use standard clone method
def clone
new_revision = DmsfFileRevision.new
new_revision.file = self.file
new_revision.disk_filename = self.disk_filename
new_revision.size = self.size
new_revision.mime_type = self.mime_type
new_revision.title = self.title
new_revision.description = self.description
new_revision.workflow = self.workflow
new_revision.major_version = self.major_version
new_revision.minor_version = self.minor_version
new_revision.source_revision = self
new_revision.user = User.current
new_revision.name = self.name
new_revision.folder = self.folder
return new_revision
end
#TODO: validate if it isn't doubled or move it to view
def workflow_str
case workflow
when 1 then l(:title_waiting_for_approval)
when 2 then l(:title_approved)
else nil
end
end
def set_workflow(workflow)
if User.current.allowed_to?(:file_approval, self.file.project)
self.workflow = workflow
else
if self.source_revision.nil?
self.workflow = workflow == 2 ? 1 : workflow
else
if workflow == 2 || self.source_revision.workflow == 1 || self.source_revision.workflow == 2
self.workflow = 1
else
self.workflow = workflow
end
end
end
end
def increase_version(version_to_increase, new_content)
if new_content
self.minor_version = case version_to_increase
when 2 then 0
else self.minor_version + 1
end
else
self.minor_version = case version_to_increase
when 1 then self.minor_version + 1
when 2 then 0
else self.minor_version
end
end
self.major_version = case version_to_increase
when 2 then self.major_version + 1
else self.major_version
end
end
def display_title
#if self.title.length > 35
# return self.title[0, 30] + "..."
#else
return self.title
#end
end
def new_storage_filename
filename = DmsfHelper.sanitize_filename(self.name)
timestamp = DateTime.now.strftime("%y%m%d%H%M%S")
while File.exist?(File.join(DmsfFile.storage_path, "#{timestamp}_#{self.id}_#{filename}"))
timestamp.succ!
end
"#{timestamp}_#{file.id}_#{filename}"
end
def copy_file_content(open_file)
File.open(self.disk_file, "wb") do |f|
while (buffer = open_file.read(8192))
f.write(buffer)
end
end
end
end