162 lines
6.1 KiB
Ruby
162 lines
6.1 KiB
Ruby
# Redmine plugin for Document Management System "Features"
|
|
#
|
|
# Copyright (C) 2012 Daniel Munn <dan.munn@munnster.co.uk>
|
|
#
|
|
# 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.
|
|
|
|
require 'fileutils'
|
|
require 'uuidtools'
|
|
|
|
class Dmsf144 < ActiveRecord::Migration
|
|
|
|
|
|
class DmsfFileLock < ActiveRecord::Base
|
|
belongs_to :file, :class_name => 'DmsfFile', :foreign_key => 'dmsf_file_id'
|
|
belongs_to :user
|
|
end
|
|
|
|
def self.up
|
|
#Add our entity_type column (used with our entity type)
|
|
add_column :dmsf_file_locks, :entity_type, :integer, :null => true
|
|
|
|
#Add our lock relevent columns (ENUM) - null (till we upgrade data)
|
|
add_column :dmsf_file_locks, :lock_type_cd, :integer, :null => true
|
|
add_column :dmsf_file_locks, :lock_scope_cd, :integer, :null => true
|
|
add_column :dmsf_file_locks, :uuid, :string, :null => true, :limit => 36
|
|
|
|
#Add our expires_at column
|
|
add_column :dmsf_file_locks, :expires_at, :datetime, :null => true
|
|
|
|
do_not_delete = []
|
|
# - 2012-07-12: Better compatibility for postgres - query used 3 columns however
|
|
# only on appearing in group, find_each imposes a limit and incorrect
|
|
# ordering, so adapted that, we grab id's load a mock object, and reload
|
|
# data into it, which should enable us to run checks we need, not as
|
|
# efficient, however compatible across the board.
|
|
DmsfFileLock.reset_column_information
|
|
DmsfFileLock.select('MAX(id) id').
|
|
order('MAX(id) DESC').
|
|
group('dmsf_file_id').
|
|
each do |lock|
|
|
lock.reload
|
|
if (lock.locked)
|
|
do_not_delete << lock.id
|
|
end
|
|
end
|
|
|
|
#Generate new lock Id's for whats being persisted
|
|
do_not_delete.each {|l|
|
|
#Find the lock
|
|
next unless lock = DmsfFileLock.find(l)
|
|
lock.uuid = UUIDTools::UUID.random_create.to_s
|
|
lock.save!
|
|
}
|
|
|
|
say "Preserving #{do_not_delete.count} file lock(s) found in old schema"
|
|
|
|
DmsfFileLock.where(['id NOT IN (?)', do_not_delete]).delete_all
|
|
|
|
#We need to force our newly found
|
|
|
|
say 'Applying default lock scope / type - Exclusive / Write'
|
|
DmsfFileLock.update_all ['entity_type = ?, lock_type_cd = ?, lock_scope_cd = ?', 0, 0, 0]
|
|
|
|
#These are not null-allowed columns
|
|
change_column :dmsf_file_locks, :entity_type, :integer, :null => false
|
|
change_column :dmsf_file_locks, :lock_type_cd, :integer, :null => false
|
|
change_column :dmsf_file_locks, :lock_scope_cd, :integer, :null => false
|
|
|
|
#Data cleanup
|
|
rename_column :dmsf_file_locks, :dmsf_file_id, :entity_id
|
|
remove_column :dmsf_file_locks, :locked
|
|
rename_table :dmsf_file_locks, :dmsf_locks
|
|
|
|
#Not sure if this is the right place to do this, as its file manipulation, not database (strictly)
|
|
say "Completing one-time file migration ..."
|
|
begin
|
|
DmsfFileRevision.visible.each {|rev|
|
|
next if rev.project.nil?
|
|
existing = DmsfFile.storage_path.join rev.disk_filename
|
|
new_path = rev.disk_file(false)
|
|
begin
|
|
if File.exist?(existing)
|
|
if File.exist?(new_path)
|
|
rev.disk_filename = rev.new_storage_filename
|
|
new_path = rev.disk_file(false)
|
|
rev.save!
|
|
end
|
|
#Ensure the project path exists
|
|
project_directory = File.dirname(new_path)
|
|
Dir.mkdir(project_directory) unless File.directory? project_directory
|
|
FileUtils.mv(existing, new_path)
|
|
say "Migration: #{existing} -> #{new_path} succeeded"
|
|
end
|
|
rescue #Here we wrap around IO in the loop to prevent one failure ruining complete run.
|
|
say "Migration: #{existing} -> #{new_path} failed"
|
|
end
|
|
}
|
|
say 'Action was successful'
|
|
rescue Exception => e
|
|
say 'Action was not successful'
|
|
puts e.message
|
|
puts e.backtrace.inspect #See issue 86
|
|
#Nothing here, we just dont want a migration to break
|
|
end
|
|
end
|
|
|
|
def self.down
|
|
rename_table :dmsf_locks, :dmsf_file_locks
|
|
add_column :dmsf_file_locks, :locked, :boolean, :default => false, :null => false
|
|
|
|
#Data cleanup - delete all expired locks, or any folder locks
|
|
DmsfFileLock.reset_column_information
|
|
say 'Removing all expired and/or folder locks'
|
|
DmsfFileLock.where(['expires_at < ? OR entity_type = 1', Time.now]).delete_all
|
|
|
|
say 'Changing all records to be locked'
|
|
DmsfFileLock.update_all ['locked = ?', true]
|
|
|
|
rename_column :dmsf_file_locks, :entity_id, :dmsf_file_id
|
|
|
|
remove_column :dmsf_file_locks, :entity_type
|
|
remove_column :dmsf_file_locks, :lock_type_cd
|
|
remove_column :dmsf_file_locks, :lock_scope_cd
|
|
remove_column :dmsf_file_locks, :expires_at
|
|
remove_column :dmsf_file_locks, :uuid
|
|
|
|
#Not sure if this is the right place to do this, as its file manipulation, not database (stricly)
|
|
begin
|
|
say 'restoring old file-structure'
|
|
DmsfFileRevision.visible.each {|rev|
|
|
next if rev.project.nil?
|
|
project = rev.project.identifier.gsub(/[^\w\.\-]/,'_')
|
|
existing = DmsfFile.storage_path.join("p_#{project}/#{rev.disk_filename}")
|
|
new_path = DmsfFile.storage_path.join(rev.disk_filename)
|
|
if File.exist?(existing)
|
|
if File.exist?(new_path)
|
|
rev.disk_filename = rev.new_storage_filename
|
|
rev.save!
|
|
new_path = DmsfFile.storage_path.join(rev.disk_filename)
|
|
end
|
|
FileUtils.mv(existing, new_path)
|
|
end
|
|
}
|
|
rescue
|
|
#Nothing here, we just dont want a migration to break
|
|
end
|
|
end
|
|
|
|
end
|