220 lines
7.7 KiB
Ruby
220 lines
7.7 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
# Redmine plugin for Document Management System "Features"
|
|
#
|
|
# 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.
|
|
|
|
# Upload helper
|
|
module DmsfUploadHelper
|
|
include Redmine::I18n
|
|
|
|
def self.commit_files_internal(committed_files, project, folder, controller = nil, container = nil, new_object: false)
|
|
failed_uploads = []
|
|
files = []
|
|
if committed_files
|
|
failed_uploads = []
|
|
committed_files.each_value do |committed_file|
|
|
name = committed_file[:name]
|
|
new_revision = DmsfFileRevision.new
|
|
file = DmsfFile.visible.find_file_by_name(project, folder, name)
|
|
unless file
|
|
link = DmsfLink.find_link_by_file_name(project, folder, name)
|
|
file = link.target_file if link
|
|
end
|
|
if file
|
|
if file.last_revision
|
|
last_revision = file.last_revision
|
|
new_revision.source_revision = last_revision
|
|
end
|
|
else
|
|
file = DmsfFile.new
|
|
file.project_id = project.id
|
|
file.name = name
|
|
file.dmsf_folder = folder
|
|
file.notification = RedmineDmsf.dmsf_default_notifications?
|
|
end
|
|
if file.locked_for_user?
|
|
failed_uploads.push file
|
|
next
|
|
end
|
|
|
|
new_revision.dmsf_file = file
|
|
new_revision.user = User.current
|
|
new_revision.name = name
|
|
new_revision.title = committed_file[:title]
|
|
new_revision.description = committed_file[:description]
|
|
new_revision.comment = committed_file[:comment]
|
|
new_revision.major_version = if committed_file[:version_major].present?
|
|
DmsfUploadHelper.db_version committed_file[:version_major]
|
|
else
|
|
1
|
|
end
|
|
new_revision.minor_version = if committed_file[:version_minor].present?
|
|
DmsfUploadHelper.db_version committed_file[:version_minor]
|
|
end
|
|
new_revision.patch_version = if committed_file[:version_patch].present?
|
|
DmsfUploadHelper.db_version committed_file[:version_patch]
|
|
end
|
|
new_revision.mime_type = committed_file[:mime_type]
|
|
new_revision.size = committed_file[:size]
|
|
new_revision.digest = committed_file[:digest]
|
|
# Custom fields
|
|
new_revision.copy_custom_field_values(committed_file[:custom_field_values])
|
|
# Need to save file first to generate id for it in case of creation.
|
|
# File id is needed to properly generate revision disk filename
|
|
unless new_revision.valid?
|
|
Rails.logger.error new_revision.errors.full_messages.to_sentence
|
|
failed_uploads.push new_revision
|
|
next
|
|
end
|
|
unless file.save
|
|
Rails.logger.error file.errors.full_messages.to_sentence
|
|
failed_uploads.push file
|
|
next
|
|
end
|
|
|
|
new_revision.disk_filename = new_revision.new_storage_filename
|
|
|
|
if new_revision.save
|
|
new_revision.assign_workflow committed_file[:dmsf_workflow_id]
|
|
begin
|
|
# If the path is not present we get it from the token
|
|
if committed_file[:token].present?
|
|
a = Attachment.find_by_token(committed_file[:token])
|
|
committed_file[:tempfile_path] = a.diskfile if a
|
|
end
|
|
FileUtils.mv committed_file[:tempfile_path], new_revision.disk_file(search_if_not_exists: false)
|
|
FileUtils.chmod 'u=wr,g=r', new_revision.disk_file(search_if_not_exists: false)
|
|
if defined?(EasyExtensions)
|
|
# We need to trigger Xapian indexing after the file is moved to its target destination
|
|
file.touch
|
|
end
|
|
file.last_revision = new_revision
|
|
files.push file
|
|
container.dmsf_file_added file if container && !new_object
|
|
Redmine::Hook.call_hook :dmsf_helper_upload_after_commit, { file: file }
|
|
rescue StandardError => e
|
|
Rails.logger.error e.message
|
|
controller.flash[:error] = e.message if controller
|
|
failed_uploads.push file
|
|
end
|
|
else
|
|
failed_uploads.push new_revision
|
|
end
|
|
# Approval workflow
|
|
next if committed_file[:workflow_id].blank?
|
|
|
|
wf = DmsfWorkflow.find_by(id: committed_file[:workflow_id])
|
|
next unless wf
|
|
|
|
# Assign the workflow
|
|
new_revision.set_workflow wf.id, 'assign'
|
|
new_revision.assign_workflow wf.id
|
|
# Start the workflow
|
|
new_revision.set_workflow wf.id, 'start'
|
|
if new_revision.save
|
|
wf.notify_users project, new_revision, controller
|
|
begin
|
|
file.lock!
|
|
rescue RedmineDmsf::Errors::DmsfLockError => e
|
|
Rails.logger.warn e.message
|
|
end
|
|
else
|
|
Rails.logger.error l(:error_workflow_assign)
|
|
end
|
|
end
|
|
# Notifications
|
|
begin
|
|
recipients = DmsfMailer.deliver_files_updated(project, files)
|
|
if RedmineDmsf.dmsf_display_notified_recipients? && recipients.any?
|
|
max_recipients = RedmineDmsf.dmsf_max_notification_receivers_info
|
|
to = recipients.collect { |user, _| user.name }.first(max_recipients).join(', ')
|
|
if to.present?
|
|
to << (recipients.count > max_recipients ? ',...' : '.')
|
|
controller.flash[:warning] = l(:warning_email_notifications, to: to) if controller
|
|
end
|
|
end
|
|
rescue StandardError => e
|
|
Rails.logger.error "Could not send email notifications: #{e.message}"
|
|
end
|
|
end
|
|
if failed_uploads.present? && controller
|
|
controller.flash[:warning] = l(:warning_some_files_were_not_committed,
|
|
files: failed_uploads.pluck(:name).join(', '))
|
|
end
|
|
[files, failed_uploads]
|
|
end
|
|
|
|
# 0..99, A..Z except O
|
|
def self.major_version_select_options
|
|
(0..99).to_a + ('A'..'Y').to_a - ['O']
|
|
end
|
|
|
|
# 0..999, ' '
|
|
def self.minor_version_select_options
|
|
(0..999).to_a + [' ']
|
|
end
|
|
|
|
class << self
|
|
alias patch_version_select_options minor_version_select_options
|
|
end
|
|
|
|
# 1 -> 2, -1 -> -2, A -> B
|
|
def self.increase_version(version)
|
|
return nil unless version
|
|
return 1 if (version == ' ') || ((-version) == ' '.ord)
|
|
|
|
if Integer(version)
|
|
if version >= 0
|
|
if (version + 1) < 1000
|
|
version + 1
|
|
else
|
|
version
|
|
end
|
|
elsif -(version - 1) < 'Z'.ord
|
|
version - 1
|
|
else
|
|
version
|
|
end
|
|
end
|
|
rescue StandardError
|
|
if (version.ord + 1) < 'Z'.ord
|
|
(version.ord + 1).chr
|
|
else
|
|
version
|
|
end
|
|
end
|
|
|
|
# 1 -> 1, -65 -> A
|
|
def self.gui_version(version)
|
|
return ' ' unless version
|
|
return version if version >= 0
|
|
|
|
(-version).chr
|
|
end
|
|
|
|
# 1 -> 1, A -> -65
|
|
def self.db_version(version)
|
|
return nil if version.nil? || version == ' '
|
|
|
|
version.to_i if Integer(version)
|
|
rescue StandardError
|
|
version = ' ' if version.blank?
|
|
-version.ord
|
|
end
|
|
end
|