546 lines
18 KiB
Ruby
546 lines
18 KiB
Ruby
# Redmine plugin for Document Management System "Features"
|
|
#
|
|
# Copyright (C) 2011 Vít Jonáš <vit.jonas@gmail.com>
|
|
# Copyright (C) 2012 Daniel Munn <dan.munn@munnster.co.uk>
|
|
# Copyright (C) 2011-14 Karel Pičman <karel.picman@kontron.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 DmsfController < ApplicationController
|
|
unloadable
|
|
|
|
class ZipMaxFilesError < StandardError; end
|
|
class EmailMaxFileSize < StandardError; end
|
|
class FileNotFound < StandardError; end
|
|
|
|
before_filter :find_project
|
|
before_filter :authorize
|
|
before_filter :find_folder, :except => [:new, :create, :edit_root, :save_root]
|
|
before_filter :find_parent, :only => [:new, :create]
|
|
|
|
helper :all
|
|
|
|
def show
|
|
@folder_manipulation_allowed = User.current.allowed_to?(:folder_manipulation, @project)
|
|
@file_manipulation_allowed = User.current.allowed_to?(:file_manipulation, @project)
|
|
@file_delete_allowed = User.current.allowed_to?(:file_delete, @project)
|
|
@force_file_unlock_allowed = User.current.allowed_to?(:force_file_unlock, @project)
|
|
|
|
@workflows_available = DmsfWorkflow.where(['project_id = ? OR project_id IS NULL', @project.id]).count > 0
|
|
|
|
unless @folder
|
|
if params[:custom_field_id].present? && params[:custom_value].present?
|
|
@subfolders = []
|
|
DmsfFolder.where(:project_id => @project.id).visible.each do |f|
|
|
f.custom_field_values.each do |v|
|
|
if v.custom_field_id == params[:custom_field_id].to_i
|
|
if v.custom_field.compare_values?(v.value, params[:custom_value])
|
|
@subfolders << f
|
|
break
|
|
end
|
|
end
|
|
end
|
|
end
|
|
@files = []
|
|
DmsfFile.where(:project_id => @project.id).visible.each do |f|
|
|
r = f.last_revision
|
|
if r
|
|
r.custom_field_values.each do |v|
|
|
if v.custom_field_id == params[:custom_field_id].to_i
|
|
if v.custom_field.compare_values?(v.value, params[:custom_value])
|
|
@files << f
|
|
break
|
|
end
|
|
end
|
|
end
|
|
end
|
|
end
|
|
@dir_links = []
|
|
DmsfLink.where(:project_id => @project.id, :target_type => DmsfFolder.model_name).visible.each do |l|
|
|
l.target_folder.custom_field_values.each do |v|
|
|
if v.custom_field_id == params[:custom_field_id].to_i
|
|
if v.custom_field.compare_values?(v.value, params[:custom_value])
|
|
@dir_links << l
|
|
break
|
|
end
|
|
end
|
|
end
|
|
end
|
|
@file_links = []
|
|
DmsfLink.where(:project_id => @project.id, :target_type => DmsfFile.model_name).visible.each do |l|
|
|
r = l.target_file.last_revision
|
|
if r
|
|
r.custom_field_values.each do |v|
|
|
if v.custom_field_id == params[:custom_field_id].to_i
|
|
if v.custom_field.compare_values?(v.value, params[:custom_value])
|
|
@file_links << l
|
|
break
|
|
end
|
|
end
|
|
end
|
|
end
|
|
end
|
|
else
|
|
@subfolders = @project.dmsf_folders.visible
|
|
@files = @project.dmsf_files.visible
|
|
@dir_links = @project.folder_links.visible
|
|
@file_links = @project.file_links.visible
|
|
end
|
|
@locked_for_user = false
|
|
else
|
|
@subfolders = @folder.subfolders.visible
|
|
@files = @folder.files.visible
|
|
@dir_links = @folder.folder_links.visible
|
|
@file_links = @folder.file_links.visible
|
|
@locked_for_user = @folder.locked_for_user?
|
|
end
|
|
|
|
@ajax_upload_size = Setting.plugin_redmine_dmsf['dmsf_max_ajax_upload_filesize'].present? ? Setting.plugin_redmine_dmsf['dmsf_max_ajax_upload_filesize'] : 100
|
|
end
|
|
|
|
def trash
|
|
@folder_manipulation_allowed = User.current.allowed_to? :folder_manipulation, @project
|
|
@file_manipulation_allowed = User.current.allowed_to? :file_manipulation, @project
|
|
@file_delete_allowed = User.current.allowed_to? :file_delete, @project
|
|
@subfolders = DmsfFolder.deleted.where(:project_id => @project.id)
|
|
@files = DmsfFile.deleted.where(:project_id => @project.id)
|
|
@dir_links = DmsfLink.deleted.where(:project_id => @project.id, :target_type => DmsfFolder.model_name)
|
|
@file_links = DmsfLink.deleted.where(:project_id => @project.id, :target_type => DmsfFile.model_name)
|
|
end
|
|
|
|
def download_email_entries
|
|
send_file(
|
|
params[:path],
|
|
:filename => 'Documents.zip',
|
|
:type => 'application/zip',
|
|
:disposition => 'attachment')
|
|
rescue Exception => e
|
|
flash[:error] = e.message
|
|
end
|
|
|
|
def entries_operation
|
|
# Download/Email
|
|
selected_folders = params[:subfolders] || []
|
|
selected_files = params[:files] || []
|
|
selected_dir_links = params[:dir_links] || []
|
|
selected_file_links = params[:file_links] || []
|
|
|
|
if selected_folders.blank? && selected_files.blank? &&
|
|
selected_dir_links.blank? && selected_file_links.blank?
|
|
flash[:warning] = l(:warning_no_entries_selected)
|
|
redirect_to :back
|
|
return
|
|
end
|
|
|
|
if selected_dir_links.present?
|
|
selected_dir_links.each do |id|
|
|
link = DmsfLink.find_by_id id
|
|
selected_folders << link.target_id if link && !selected_folders.include?(link.target_id.to_s)
|
|
end
|
|
end
|
|
|
|
if selected_file_links.present?
|
|
selected_file_links.each do |id|
|
|
link = DmsfLink.find_by_id id
|
|
selected_files << link.target_id if link && !selected_files.include?(link.target_id.to_s)
|
|
end
|
|
end
|
|
|
|
if params[:email_entries].present?
|
|
email_entries(selected_folders, selected_files)
|
|
elsif params[:restore_entries].present?
|
|
restore_entries(selected_folders, selected_files, selected_dir_links, selected_file_links)
|
|
redirect_to :back
|
|
elsif params[:delete_entries].present?
|
|
delete_entries(selected_folders, selected_files, selected_dir_links, selected_file_links, false)
|
|
redirect_to :back
|
|
elsif params[:destroy_entries].present?
|
|
delete_entries(selected_folders, selected_files, selected_dir_links, selected_file_links, true)
|
|
redirect_to :back
|
|
else
|
|
download_entries(selected_folders, selected_files)
|
|
end
|
|
rescue ZipMaxFilesError
|
|
flash[:error] = l(:error_max_files_exceeded, :number => Setting.plugin_redmine_dmsf['dmsf_max_file_download'])
|
|
redirect_to :back
|
|
rescue EmailMaxFileSize
|
|
flash[:error] = l(:error_max_email_filesize_exceeded, :number => Setting.plugin_redmine_dmsf['dmsf_max_email_filesize'])
|
|
redirect_to :back
|
|
rescue FileNotFound
|
|
render_404
|
|
rescue DmsfAccessError
|
|
render_403
|
|
end
|
|
|
|
def tag_changed
|
|
# Tag filter
|
|
if params[:dmsf_folder] && params[:dmsf_folder][:custom_field_values].present?
|
|
redirect_to dmsf_folder_path(
|
|
:id => @project,
|
|
:custom_field_id => params[:dmsf_folder][:custom_field_values].first[0],
|
|
:custom_value => params[:dmsf_folder][:custom_field_values].first[1])
|
|
else
|
|
redirect_to :back
|
|
end
|
|
end
|
|
|
|
def entries_email
|
|
@email_params = params[:email]
|
|
if @email_params[:to].strip.blank?
|
|
flash.now[:error] = l(:error_email_to_must_be_entered)
|
|
render :action => 'email_entries'
|
|
return
|
|
end
|
|
DmsfMailer.send_documents(@project, User.current, @email_params).deliver
|
|
File.delete(@email_params['zipped_content'])
|
|
flash[:notice] = l(:notice_email_sent, @email_params['to'])
|
|
|
|
redirect_to dmsf_folder_path(:id => @project, :folder_id => @folder)
|
|
end
|
|
|
|
def new
|
|
@folder = DmsfFolder.new
|
|
@pathfolder = @parent
|
|
render :action => 'edit'
|
|
end
|
|
|
|
def create
|
|
@folder = DmsfFolder.new(params[:dmsf_folder])
|
|
@folder.project = @project
|
|
@folder.user = User.current
|
|
if @folder.save
|
|
flash[:notice] = l(:notice_folder_created)
|
|
redirect_to({:controller => 'dmsf', :action => 'show', :id => @project, :folder_id => @folder})
|
|
else
|
|
@pathfolder = @parent
|
|
render :action => 'edit'
|
|
end
|
|
end
|
|
|
|
def edit
|
|
@parent = @folder.folder
|
|
@pathfolder = copy_folder(@folder)
|
|
end
|
|
|
|
def save
|
|
unless params[:dmsf_folder]
|
|
redirect_to :controller => 'dmsf', :action => 'show', :id => @project, :folder_id => @folder
|
|
return
|
|
end
|
|
@pathfolder = copy_folder(@folder)
|
|
@folder.attributes = params[:dmsf_folder]
|
|
if @folder.save
|
|
flash[:notice] = l(:notice_folder_details_were_saved)
|
|
redirect_to :controller => 'dmsf', :action => 'show', :id => @project, :folder_id => @folder
|
|
else
|
|
render :action => 'edit'
|
|
end
|
|
end
|
|
|
|
def delete
|
|
commit = params[:commit] == 'yes'
|
|
if @folder.delete(commit)
|
|
flash[:notice] = l(:notice_folder_deleted)
|
|
else
|
|
flash[:error] = @folder.errors[:base][0]
|
|
end
|
|
if commit
|
|
redirect_to :back
|
|
else
|
|
redirect_to dmsf_folder_path(:id => @project, :folder_id => @folder.folder)
|
|
end
|
|
end
|
|
|
|
def restore
|
|
if @folder.restore
|
|
flash[:notice] = l(:notice_dmsf_folder_restored)
|
|
else
|
|
flash[:error] = @folder.errors[:base][0]
|
|
end
|
|
redirect_to :back
|
|
end
|
|
|
|
def edit_root
|
|
end
|
|
|
|
def save_root
|
|
@project.dmsf_description = params[:project][:dmsf_description]
|
|
@project.save!
|
|
flash[:notice] = l(:notice_folder_details_were_saved)
|
|
redirect_to :controller => 'dmsf', :action => 'show', :id => @project
|
|
end
|
|
|
|
def notify_activate
|
|
if((@folder && @folder.notification) || (@folder.nil? && @project.dmsf_notification))
|
|
flash[:warning] = l(:warning_folder_notifications_already_activated)
|
|
else
|
|
if @folder
|
|
@folder.notify_activate
|
|
else
|
|
@project.dmsf_notification = true
|
|
@project.save
|
|
end
|
|
flash[:notice] = l(:notice_folder_notifications_activated)
|
|
end
|
|
redirect_to :back
|
|
end
|
|
|
|
def notify_deactivate
|
|
if((@folder && !@folder.notification) || (@folder.nil? && !@project.dmsf_notification))
|
|
flash[:warning] = l(:warning_folder_notifications_already_deactivated)
|
|
else
|
|
if @folder
|
|
@folder.notify_deactivate
|
|
else
|
|
@project.dmsf_notification = false
|
|
@project.save
|
|
end
|
|
flash[:notice] = l(:notice_folder_notifications_deactivated)
|
|
end
|
|
redirect_to :back
|
|
end
|
|
|
|
def lock
|
|
if @folder.nil?
|
|
flash[:warning] = l(:warning_foler_unlockable)
|
|
elsif @folder.locked?
|
|
flash[:warning] = l(:warning_folder_already_locked)
|
|
else
|
|
@folder.lock!
|
|
flash[:notice] = l(:notice_folder_locked)
|
|
end
|
|
redirect_to :back
|
|
end
|
|
|
|
def unlock
|
|
if @folder.nil?
|
|
flash[:warning] = l(:warning_foler_unlockable)
|
|
elsif !@folder.locked?
|
|
flash[:warning] = l(:warning_folder_not_locked)
|
|
else
|
|
if @folder.locks[0].user == User.current || User.current.allowed_to?(:force_file_unlock, @project)
|
|
@folder.unlock!
|
|
flash[:notice] = l(:notice_folder_unlocked)
|
|
else
|
|
flash[:error] = l(:error_only_user_that_locked_folder_can_unlock_it)
|
|
end
|
|
end
|
|
redirect_to :back
|
|
end
|
|
|
|
private
|
|
|
|
def log_activity(file, action)
|
|
Rails.logger.info "#{Time.now.strftime('%Y-%m-%d %H:%M:%S')} #{User.current.login}@#{request.remote_ip}/#{request.env['HTTP_X_FORWARDED_FOR']}: #{action} dmsf://#{file.project.identifier}/#{file.id}"
|
|
end
|
|
|
|
def email_entries(selected_folders, selected_files)
|
|
begin
|
|
zip = DmsfZip.new
|
|
zip_entries(zip, selected_folders, selected_files)
|
|
|
|
zipped_content = "#{DmsfHelper.temp_dir}/#{DmsfHelper.temp_filename('dmsf_email_sent_documents.zip')}";
|
|
|
|
File.open(zipped_content, 'wb') do |f|
|
|
zip_file = File.open(zip.finish, 'rb')
|
|
while (buffer = zip_file.read(8192))
|
|
f.write(buffer)
|
|
end
|
|
end
|
|
|
|
max_filesize = Setting.plugin_redmine_dmsf['dmsf_max_email_filesize'].to_f
|
|
if max_filesize > 0 && File.size(zipped_content) > max_filesize * 1048576
|
|
raise EmailMaxFileSize
|
|
end
|
|
|
|
zip.files.each do |f|
|
|
log_activity(f, 'emailing zip')
|
|
audit = DmsfFileRevisionAccess.new(:user_id => User.current.id, :dmsf_file_revision_id => f.last_revision.id,
|
|
:action => DmsfFileRevisionAccess::EmailAction)
|
|
audit.save!
|
|
end
|
|
|
|
@email_params = {
|
|
:zipped_content => zipped_content,
|
|
:folders => selected_folders,
|
|
:files => selected_files
|
|
}
|
|
render :action => 'email_entries'
|
|
rescue Exception => e
|
|
flash[:error] = e.message
|
|
ensure
|
|
zip.close if zip
|
|
end
|
|
end
|
|
|
|
def download_entries(selected_folders, selected_files)
|
|
begin
|
|
zip = DmsfZip.new
|
|
zip_entries(zip, selected_folders, selected_files)
|
|
|
|
zip.files.each do |f|
|
|
log_activity(f, 'download zip')
|
|
audit = DmsfFileRevisionAccess.new(:user_id => User.current.id, :dmsf_file_revision_id => f.last_revision.id,
|
|
:action => DmsfFileRevisionAccess::DownloadAction)
|
|
audit.save!
|
|
end
|
|
|
|
send_file(zip.finish,
|
|
:filename => filename_for_content_disposition("#{@project.name}-#{DateTime.now.strftime('%y%m%d%H%M%S')}.zip"),
|
|
:type => 'application/zip',
|
|
:disposition => 'attachment')
|
|
rescue Exception => e
|
|
flash[:error] = e.message
|
|
ensure
|
|
zip.close if zip
|
|
end
|
|
end
|
|
|
|
def zip_entries(zip, selected_folders, selected_files)
|
|
if selected_folders && selected_folders.is_a?(Array)
|
|
selected_folders.each do |selected_folder_id|
|
|
folder = DmsfFolder.visible.find_by_id selected_folder_id
|
|
if folder
|
|
zip.add_folder(folder, (@folder.dmsf_path_str if @folder))
|
|
else
|
|
raise FileNotFound
|
|
end
|
|
end
|
|
end
|
|
if selected_files && selected_files.is_a?(Array)
|
|
selected_files.each do |selected_file_id|
|
|
file = DmsfFile.visible.find_by_id selected_file_id
|
|
if file && file.last_revision && File.exists?(file.last_revision.disk_file)
|
|
zip.add_file(file, (@folder.dmsf_path_str if @folder)) if file
|
|
else
|
|
raise FileNotFound
|
|
end
|
|
end
|
|
end
|
|
max_files = Setting.plugin_redmine_dmsf['dmsf_max_file_download'].to_i
|
|
if max_files > 0 && zip.files.length > max_files
|
|
raise ZipMaxFilesError, zip.files.length
|
|
end
|
|
zip
|
|
end
|
|
|
|
def restore_entries(selected_folders, selected_files, selected_dir_links, selected_file_links)
|
|
# Folders
|
|
selected_folders.each do |id|
|
|
folder = DmsfFolder.find_by_id id
|
|
if folder
|
|
unless folder.restore
|
|
flash[:error] = folder.errors[:base][0]
|
|
end
|
|
else
|
|
raise FileNotFound
|
|
end
|
|
end
|
|
# Files
|
|
selected_files.each do |id|
|
|
file = DmsfFile.find_by_id id
|
|
if file
|
|
unless file.restore
|
|
flash[:error] = file.errors[:base][0]
|
|
end
|
|
else
|
|
raise FileNotFound
|
|
end
|
|
end
|
|
# Links
|
|
(selected_dir_links + selected_file_links).each do |id|
|
|
link = DmsfLink.find_by_id id
|
|
if link
|
|
unless link.restore
|
|
flash[:error] = link.errors[:base][0]
|
|
end
|
|
else
|
|
raise FileNotFound
|
|
end
|
|
end
|
|
end
|
|
|
|
def delete_entries(selected_folders, selected_files, selected_dir_links, selected_file_links, commit)
|
|
# Folders
|
|
selected_folders.each do |id|
|
|
folder = DmsfFolder.find_by_id id
|
|
if folder
|
|
unless folder.delete commit
|
|
flash[:error] = folder.errors[:base][0]
|
|
end
|
|
else
|
|
raise FileNotFound
|
|
end
|
|
end
|
|
# Files
|
|
deleted_files = []
|
|
not_deleted_files = []
|
|
selected_files.each do |id|
|
|
file = DmsfFile.find_by_id id
|
|
if file
|
|
if file.delete(commit)
|
|
deleted_files << file unless commit
|
|
else
|
|
not_deleted_files << file
|
|
end
|
|
else
|
|
raise FileNotFound
|
|
end
|
|
end
|
|
# Activities
|
|
if !deleted_files.empty?
|
|
deleted_files.each do |f|
|
|
log_activity(f, 'deleted')
|
|
end
|
|
begin
|
|
DmsfMailer.get_notify_users(User.current, deleted_files).each do |u|
|
|
DmsfMailer.files_deleted(u, @project, deleted_files).deliver
|
|
end
|
|
rescue Exception => e
|
|
Rails.logger.error "Could not send email notifications: #{e.message}"
|
|
end
|
|
end
|
|
unless not_deleted_files.empty?
|
|
flash[:warning] = l(:warning_some_entries_were_not_deleted, :entries => not_deleted_files.map{|e| e.title}.join(', '))
|
|
end
|
|
# Links
|
|
(selected_dir_links + selected_file_links).each do |id|
|
|
link = DmsfLink.find_by_id id
|
|
link.delete commit if link
|
|
end
|
|
if flash[:error].blank? && flash[:warning].blank?
|
|
flash[:notice] = l(:notice_entries_deleted)
|
|
end
|
|
end
|
|
|
|
def find_folder
|
|
@folder = DmsfFolder.find params[:folder_id] if params[:folder_id].present?
|
|
rescue DmsfAccessError
|
|
render_403
|
|
end
|
|
|
|
def find_parent
|
|
@parent = DmsfFolder.visible.find params[:parent_id] if params[:parent_id].present?
|
|
rescue DmsfAccessError
|
|
render_403
|
|
end
|
|
|
|
def copy_folder(folder)
|
|
copy = folder.clone
|
|
copy.id = folder.id
|
|
copy
|
|
end
|
|
|
|
end |