discourse/app/controllers/admin/backups_controller.rb

161 lines
5.0 KiB
Ruby
Raw Normal View History

require "backup_restore/backup_restore"
2014-02-12 23:33:21 -05:00
class Admin::BackupsController < Admin::AdminController
2014-05-27 16:14:37 -04:00
skip_before_filter :check_xhr, only: [:index, :show, :logs, :check_backup_chunk, :upload_backup_chunk]
2014-02-12 23:33:21 -05:00
def index
respond_to do |format|
format.html do
store_preloaded("backups", MultiJson.dump(serialize_data(Backup.all, BackupSerializer)))
store_preloaded("operations_status", MultiJson.dump(BackupRestore.operations_status))
store_preloaded("logs", MultiJson.dump(BackupRestore.logs))
2014-02-12 23:33:21 -05:00
render "default/empty"
end
format.json do
render_serialized(Backup.all, BackupSerializer)
end
end
end
def status
render_json_dump(BackupRestore.operations_status)
end
def create
2014-08-20 12:48:56 -04:00
opts = {
publish_to_message_bus: true,
with_uploads: params.fetch(:with_uploads) == "true",
client_id: params[:client_id],
2014-08-20 12:48:56 -04:00
}
BackupRestore.backup!(current_user.id, opts)
2014-02-12 23:33:21 -05:00
rescue BackupRestore::OperationRunningError
render json: failed_json.merge(message: I18n.t("backup.operation_already_running"))
else
2016-02-27 12:38:24 -05:00
StaffActionLogger.new(current_user).log_backup_operation
2014-02-12 23:33:21 -05:00
render json: success_json
end
def cancel
BackupRestore.cancel!
rescue BackupRestore::OperationRunningError
render json: failed_json.merge(message: I18n.t("backup.operation_already_running"))
else
render json: success_json
end
# download
def show
filename = params.fetch(:id)
if backup = Backup[filename]
2014-09-22 19:25:53 -04:00
headers['Content-Length'] = File.size(backup.path)
2014-02-12 23:33:21 -05:00
send_file backup.path
else
render nothing: true, status: 404
end
end
def destroy
backup = Backup[params.fetch(:id)]
if backup
backup.remove
render nothing: true
else
render nothing: true, status: 404
end
2014-02-12 23:33:21 -05:00
end
def logs
store_preloaded("operations_status", MultiJson.dump(BackupRestore.operations_status))
store_preloaded("logs", MultiJson.dump(BackupRestore.logs))
2014-02-12 23:33:21 -05:00
render "default/empty"
end
def restore
opts = {
filename: params.fetch(:id),
client_id: params.fetch(:client_id),
publish_to_message_bus: true,
}
SiteSetting.set_and_log(:disable_emails, true, current_user)
BackupRestore.restore!(current_user.id, opts)
2014-02-12 23:33:21 -05:00
rescue BackupRestore::OperationRunningError
render json: failed_json.merge(message: I18n.t("backup.operation_already_running"))
else
render json: success_json
end
def rollback
BackupRestore.rollback!
rescue BackupRestore::OperationRunningError
render json: failed_json.merge(message: I18n.t("backup.operation_already_running"))
else
render json: success_json
end
def readonly
enable = params.fetch(:enable).to_s == "true"
readonly_mode_key = Discourse::USER_READONLY_MODE_KEY
if enable
Discourse.enable_readonly_mode(readonly_mode_key)
else
Discourse.disable_readonly_mode(readonly_mode_key)
end
StaffActionLogger.new(current_user).log_change_readonly_mode(enable)
2014-02-12 23:33:21 -05:00
render nothing: true
end
2014-05-27 16:14:37 -04:00
def check_backup_chunk
2014-02-21 19:41:01 -05:00
identifier = params.fetch(:resumableIdentifier)
filename = params.fetch(:resumableFilename)
chunk_number = params.fetch(:resumableChunkNumber)
current_chunk_size = params.fetch(:resumableCurrentChunkSize).to_i
# path to chunk file
chunk = Backup.chunk_path(identifier, filename, chunk_number)
2014-05-27 16:14:37 -04:00
# check chunk upload status
status = HandleChunkUpload.check_chunk(chunk, current_chunk_size: current_chunk_size)
2014-02-21 19:41:01 -05:00
render nothing: true, status: status
end
2014-05-27 16:14:37 -04:00
def upload_backup_chunk
filename = params.fetch(:resumableFilename)
total_size = params.fetch(:resumableTotalSize).to_i
return render status: 415, text: I18n.t("backup.backup_file_should_be_tar_gz") unless /\.(tar\.gz|t?gz)$/i =~ filename
return render status: 415, text: I18n.t("backup.not_enough_space_on_disk") unless has_enough_space_on_disk?(total_size)
2016-09-16 19:07:46 -04:00
return render status: 415, text: I18n.t("backup.invalid_filename") unless !!(/^[a-zA-Z0-9\._\-]+$/ =~ filename)
2014-02-21 19:41:01 -05:00
file = params.fetch(:file)
identifier = params.fetch(:resumableIdentifier)
chunk_number = params.fetch(:resumableChunkNumber).to_i
chunk_size = params.fetch(:resumableChunkSize).to_i
current_chunk_size = params.fetch(:resumableCurrentChunkSize).to_i
# path to chunk file
chunk = Backup.chunk_path(identifier, filename, chunk_number)
2014-05-27 16:14:37 -04:00
# upload chunk
HandleChunkUpload.upload_chunk(chunk, file: file)
2014-02-21 19:41:01 -05:00
uploaded_file_size = chunk_number * chunk_size
# when all chunks are uploaded
if uploaded_file_size + current_chunk_size >= total_size
# merge all the chunks in a background thread
Jobs.enqueue_in(5.seconds, :backup_chunks_merger, filename: filename, identifier: identifier, chunks: chunk_number)
2014-02-21 19:41:01 -05:00
end
render nothing: true
end
private
def has_enough_space_on_disk?(size)
`df -Pk #{Rails.root}/public/backups | awk 'NR==2 {print $4 * 1024;}'`.to_i > size
end
2014-02-12 23:33:21 -05:00
end