2018-01-19 11:51:42 -05:00
|
|
|
require "db_helper"
|
2013-06-15 05:29:20 -04:00
|
|
|
require "digest/sha1"
|
2018-01-19 11:51:42 -05:00
|
|
|
require "base62"
|
2013-06-15 05:29:20 -04:00
|
|
|
|
2016-04-11 14:42:40 -04:00
|
|
|
################################################################################
|
|
|
|
# gather #
|
|
|
|
################################################################################
|
|
|
|
|
|
|
|
task "uploads:gather" => :environment do
|
2016-04-12 10:00:25 -04:00
|
|
|
ENV["RAILS_DB"] ? gather_uploads : gather_uploads_for_all_sites
|
|
|
|
end
|
|
|
|
|
|
|
|
def gather_uploads_for_all_sites
|
|
|
|
RailsMultisite::ConnectionManagement.each_connection { gather_uploads }
|
|
|
|
end
|
|
|
|
|
2016-04-13 10:33:00 -04:00
|
|
|
def file_exists?(path)
|
|
|
|
File.exists?(path) && File.size(path) > 0
|
|
|
|
rescue
|
|
|
|
false
|
|
|
|
end
|
|
|
|
|
2016-04-12 10:00:25 -04:00
|
|
|
def gather_uploads
|
2016-04-11 14:42:40 -04:00
|
|
|
public_directory = "#{Rails.root}/public"
|
|
|
|
current_db = RailsMultisite::ConnectionManagement.current_db
|
|
|
|
|
|
|
|
puts "", "Gathering uploads for '#{current_db}'...", ""
|
|
|
|
|
2016-04-11 15:17:33 -04:00
|
|
|
Upload.where("url ~ '^\/uploads\/'")
|
2017-07-27 21:20:09 -04:00
|
|
|
.where("url !~ '^\/uploads\/#{current_db}'")
|
|
|
|
.find_each do |upload|
|
2016-04-11 14:42:40 -04:00
|
|
|
begin
|
|
|
|
old_db = upload.url[/^\/uploads\/([^\/]+)\//, 1]
|
|
|
|
from = upload.url.dup
|
|
|
|
to = upload.url.sub("/uploads/#{old_db}/", "/uploads/#{current_db}/")
|
|
|
|
source = "#{public_directory}#{from}"
|
|
|
|
destination = "#{public_directory}#{to}"
|
|
|
|
|
2016-04-13 10:33:00 -04:00
|
|
|
# create destination directory & copy file unless it already exists
|
|
|
|
unless file_exists?(destination)
|
|
|
|
`mkdir -p '#{File.dirname(destination)}'`
|
|
|
|
`cp --link '#{source}' '#{destination}'`
|
|
|
|
end
|
|
|
|
|
2016-04-11 14:42:40 -04:00
|
|
|
# ensure file has been succesfuly copied over
|
2016-04-13 10:33:00 -04:00
|
|
|
raise unless file_exists?(destination)
|
|
|
|
|
2016-04-11 14:42:40 -04:00
|
|
|
# remap links in db
|
|
|
|
DbHelper.remap(from, to)
|
|
|
|
rescue
|
|
|
|
putc "!"
|
|
|
|
else
|
|
|
|
putc "."
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
puts "", "Done!"
|
|
|
|
|
|
|
|
end
|
|
|
|
|
2015-05-25 11:59:00 -04:00
|
|
|
################################################################################
|
|
|
|
# backfill_shas #
|
|
|
|
################################################################################
|
|
|
|
|
2013-06-15 05:29:20 -04:00
|
|
|
task "uploads:backfill_shas" => :environment do
|
|
|
|
RailsMultisite::ConnectionManagement.each_connection do |db|
|
2015-06-10 11:19:58 -04:00
|
|
|
puts "Backfilling #{db}..."
|
|
|
|
Upload.where(sha1: nil).find_each do |u|
|
|
|
|
begin
|
|
|
|
path = Discourse.store.path_for(u)
|
2016-09-02 02:50:13 -04:00
|
|
|
u.sha1 = Upload.generate_digest(path)
|
2015-06-10 11:19:58 -04:00
|
|
|
u.save!
|
2013-06-15 05:29:20 -04:00
|
|
|
putc "."
|
2016-08-23 03:05:37 -04:00
|
|
|
rescue => e
|
2016-08-28 22:30:10 -04:00
|
|
|
puts "Skipping #{u.original_filename} (#{u.url}) #{e.message}"
|
2013-06-15 05:29:20 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2015-06-10 11:19:58 -04:00
|
|
|
puts "", "Done"
|
2013-06-15 05:29:20 -04:00
|
|
|
end
|
2014-06-24 09:35:15 -04:00
|
|
|
|
2015-05-25 11:59:00 -04:00
|
|
|
################################################################################
|
|
|
|
# migrate_from_s3 #
|
|
|
|
################################################################################
|
|
|
|
|
2014-06-24 09:35:15 -04:00
|
|
|
task "uploads:migrate_from_s3" => :environment do
|
2016-07-25 06:12:10 -04:00
|
|
|
ENV["RAILS_DB"] ? migrate_from_s3 : migrate_all_from_s3
|
|
|
|
end
|
2014-06-24 09:35:15 -04:00
|
|
|
|
2016-07-25 06:12:10 -04:00
|
|
|
def guess_filename(url, raw)
|
|
|
|
begin
|
|
|
|
uri = URI.parse("http:#{url}")
|
|
|
|
f = uri.open("rb", read_timeout: 5, redirect: true, allow_redirections: :all)
|
|
|
|
filename = if f.meta && f.meta["content-disposition"]
|
|
|
|
f.meta["content-disposition"][/filename="([^"]+)"/, 1].presence
|
|
|
|
end
|
|
|
|
filename ||= raw[/<a class="attachment" href="(?:https?:)?#{Regexp.escape(url)}">([^<]+)<\/a>/, 1].presence
|
|
|
|
filename ||= File.basename(url)
|
|
|
|
filename
|
|
|
|
rescue
|
2017-07-27 21:20:09 -04:00
|
|
|
nil
|
2016-07-25 06:12:10 -04:00
|
|
|
ensure
|
|
|
|
f.try(:close!) rescue nil
|
|
|
|
end
|
|
|
|
end
|
2015-03-18 13:23:55 -04:00
|
|
|
|
2016-07-25 06:12:10 -04:00
|
|
|
def migrate_all_from_s3
|
|
|
|
RailsMultisite::ConnectionManagement.each_connection { migrate_from_s3 }
|
|
|
|
end
|
2015-03-18 13:23:55 -04:00
|
|
|
|
2016-07-25 06:12:10 -04:00
|
|
|
def migrate_from_s3
|
|
|
|
require "file_store/s3_store"
|
2014-06-24 09:35:15 -04:00
|
|
|
|
2016-07-25 06:12:10 -04:00
|
|
|
# make sure S3 is disabled
|
2018-01-21 19:42:09 -05:00
|
|
|
if SiteSetting.Upload.enable_s3_uploads
|
2016-07-25 06:12:10 -04:00
|
|
|
puts "You must disable S3 uploads before running that task."
|
|
|
|
return
|
|
|
|
end
|
2014-06-24 09:35:15 -04:00
|
|
|
|
2016-07-25 06:12:10 -04:00
|
|
|
db = RailsMultisite::ConnectionManagement.current_db
|
2014-06-24 09:35:15 -04:00
|
|
|
|
2016-07-25 06:12:10 -04:00
|
|
|
puts "Migrating uploads from S3 to local storage for '#{db}'..."
|
2018-11-13 06:49:26 -05:00
|
|
|
|
2018-11-13 06:27:38 -05:00
|
|
|
max_file_size = [SiteSetting.max_image_size_kb, SiteSetting.max_attachment_size_kb].max.kilobytes
|
2014-06-24 09:35:15 -04:00
|
|
|
|
2018-08-01 16:58:46 -04:00
|
|
|
Post
|
|
|
|
.where("user_id > 0")
|
|
|
|
.where("raw LIKE '%.s3%.amazonaws.com/%' OR raw LIKE '%(upload://%'")
|
|
|
|
.find_each do |post|
|
2018-01-23 12:48:06 -05:00
|
|
|
begin
|
|
|
|
updated = false
|
|
|
|
|
2018-01-24 07:22:07 -05:00
|
|
|
post.raw.gsub!(/(\/\/[\w.-]+amazonaws\.com\/(original|optimized)\/([a-z0-9]+\/)+\h{40}([\w.-]+)?)/i) do |url|
|
2016-07-25 06:12:10 -04:00
|
|
|
begin
|
|
|
|
if filename = guess_filename(url, post.raw)
|
2018-11-13 06:27:38 -05:00
|
|
|
file = FileHelper.download("http:#{url}", max_file_size: max_file_size, tmp_file_name: "from_s3", follow_redirect: true)
|
2018-01-24 07:22:07 -05:00
|
|
|
sha1 = Upload.generate_digest(file)
|
|
|
|
origin = nil
|
|
|
|
|
|
|
|
existing_upload = Upload.find_by(sha1: sha1)
|
|
|
|
if existing_upload&.url&.start_with?("//")
|
|
|
|
filename = existing_upload.original_filename
|
|
|
|
origin = existing_upload.origin
|
|
|
|
existing_upload.destroy
|
|
|
|
end
|
|
|
|
|
|
|
|
new_upload = UploadCreator.new(file, filename, origin: origin).create_for(post.user_id || -1)
|
2018-01-23 12:48:06 -05:00
|
|
|
if new_upload&.save
|
|
|
|
updated = true
|
2018-01-24 07:22:07 -05:00
|
|
|
url = new_upload.url
|
2016-07-25 06:12:10 -04:00
|
|
|
end
|
|
|
|
end
|
2018-01-23 12:48:06 -05:00
|
|
|
|
|
|
|
url
|
|
|
|
rescue
|
|
|
|
url
|
2016-07-25 06:12:10 -04:00
|
|
|
end
|
|
|
|
end
|
2018-01-23 12:48:06 -05:00
|
|
|
|
2018-08-01 16:58:46 -04:00
|
|
|
post.raw.gsub!(/(upload:\/\/[0-9a-zA-Z]+\.\w+)/) do |url|
|
|
|
|
begin
|
|
|
|
if sha1 = Upload.sha1_from_short_url(url)
|
|
|
|
if upload = Upload.find_by(sha1: sha1)
|
2018-08-02 04:04:41 -04:00
|
|
|
if upload.url.start_with?("//")
|
2018-11-13 06:27:38 -05:00
|
|
|
file = FileHelper.download("http:#{upload.url}", max_file_size: max_file_size, tmp_file_name: "from_s3", follow_redirect: true)
|
2018-08-02 04:04:41 -04:00
|
|
|
filename = upload.original_filename
|
|
|
|
origin = upload.origin
|
|
|
|
upload.destroy
|
|
|
|
|
|
|
|
new_upload = UploadCreator.new(file, filename, origin: origin).create_for(post.user_id || -1)
|
|
|
|
if new_upload&.save
|
|
|
|
updated = true
|
|
|
|
url = new_upload.url
|
|
|
|
end
|
2018-08-01 16:58:46 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
url
|
|
|
|
rescue
|
|
|
|
url
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2018-01-23 12:48:06 -05:00
|
|
|
if updated
|
|
|
|
post.save!
|
|
|
|
post.rebake!
|
|
|
|
putc "#"
|
|
|
|
else
|
|
|
|
putc "."
|
|
|
|
end
|
2018-01-24 07:22:07 -05:00
|
|
|
|
2018-01-23 12:48:06 -05:00
|
|
|
rescue
|
|
|
|
putc "X"
|
2014-06-24 09:35:15 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2016-07-25 06:12:10 -04:00
|
|
|
puts "Done!"
|
2014-06-24 09:35:15 -04:00
|
|
|
end
|
2014-09-29 12:31:53 -04:00
|
|
|
|
2015-05-25 11:59:00 -04:00
|
|
|
################################################################################
|
|
|
|
# migrate_to_s3 #
|
|
|
|
################################################################################
|
|
|
|
|
|
|
|
task "uploads:migrate_to_s3" => :environment do
|
|
|
|
require "file_store/s3_store"
|
|
|
|
require "file_store/local_store"
|
|
|
|
|
|
|
|
ENV["RAILS_DB"] ? migrate_to_s3 : migrate_to_s3_all_sites
|
|
|
|
end
|
|
|
|
|
|
|
|
def migrate_to_s3_all_sites
|
|
|
|
RailsMultisite::ConnectionManagement.each_connection { migrate_to_s3 }
|
|
|
|
end
|
|
|
|
|
|
|
|
def migrate_to_s3
|
|
|
|
# make sure s3 is enabled
|
2017-10-06 01:20:01 -04:00
|
|
|
if !SiteSetting.Upload.enable_s3_uploads
|
2015-05-25 11:59:00 -04:00
|
|
|
puts "You must enable s3 uploads before running that task"
|
|
|
|
return
|
|
|
|
end
|
|
|
|
|
|
|
|
db = RailsMultisite::ConnectionManagement.current_db
|
|
|
|
|
2017-10-06 01:20:01 -04:00
|
|
|
puts "Migrating uploads to S3 (#{SiteSetting.Upload.s3_upload_bucket}) for '#{db}'..."
|
2015-05-25 11:59:00 -04:00
|
|
|
|
|
|
|
# will throw an exception if the bucket is missing
|
|
|
|
s3 = FileStore::S3Store.new
|
|
|
|
local = FileStore::LocalStore.new
|
|
|
|
|
2018-11-07 23:37:36 -05:00
|
|
|
exclude_tables = %i{
|
|
|
|
incoming_emails
|
|
|
|
stylesheet_cache
|
|
|
|
search_logs
|
|
|
|
post_search_data
|
|
|
|
notifications
|
2018-11-08 03:37:19 -05:00
|
|
|
email_logs
|
2018-11-07 23:37:36 -05:00
|
|
|
}
|
|
|
|
|
2015-05-25 11:59:00 -04:00
|
|
|
# Migrate all uploads
|
2018-11-08 03:37:19 -05:00
|
|
|
file_uploads = Upload.where.not(sha1: nil).where("url NOT LIKE '#{s3.absolute_base_url}%'")
|
|
|
|
image_uploads = file_uploads.where("lower(extension) NOT IN (?)", FileHelper.supported_images.to_a)
|
|
|
|
|
|
|
|
[image_uploads, file_uploads].each do |uploads|
|
2018-11-08 03:42:12 -05:00
|
|
|
uploads.find_in_batches(batch_size: 100) do |batch|
|
2018-11-08 03:37:19 -05:00
|
|
|
batch.each do |upload|
|
|
|
|
now = Process.clock_gettime(Process::CLOCK_MONOTONIC)
|
|
|
|
# remove invalid uploads
|
|
|
|
if upload.url.blank?
|
|
|
|
upload.destroy!
|
|
|
|
next
|
|
|
|
end
|
|
|
|
# store the old url
|
|
|
|
from = upload.url
|
|
|
|
# retrieve the path to the local file
|
|
|
|
path = local.path_for(upload)
|
|
|
|
# make sure the file exists locally
|
|
|
|
if !path || !File.exists?(path)
|
|
|
|
puts "#{from} does not exist locally"
|
|
|
|
next
|
|
|
|
end
|
2015-05-25 11:59:00 -04:00
|
|
|
|
2018-11-08 03:37:19 -05:00
|
|
|
begin
|
|
|
|
file = File.open(path)
|
|
|
|
content_type = `file --mime-type -b #{path}`.strip
|
|
|
|
to = s3.store_upload(file, upload, content_type)
|
|
|
|
rescue => e
|
|
|
|
puts "Encountered an error while migrating #{upload.url}: #{e.class}: #{e.message}"
|
|
|
|
next
|
|
|
|
ensure
|
|
|
|
file&.close
|
|
|
|
end
|
2018-11-07 23:37:36 -05:00
|
|
|
|
2018-11-08 03:37:19 -05:00
|
|
|
# remap the URL
|
|
|
|
DbHelper.remap(from, to, exclude_tables: exclude_tables)
|
|
|
|
upload.optimized_images.destroy_all
|
|
|
|
puts "Migrating #{from} --> #{to} took #{Process.clock_gettime(Process::CLOCK_MONOTONIC) - now} seconds"
|
|
|
|
end
|
2015-05-25 11:59:00 -04:00
|
|
|
|
2018-11-08 03:37:19 -05:00
|
|
|
[
|
|
|
|
Discourse.asset_host,
|
|
|
|
Discourse.base_url_no_prefix
|
|
|
|
].each do |from|
|
|
|
|
now = Process.clock_gettime(Process::CLOCK_MONOTONIC)
|
|
|
|
from = "#{from}#{SiteSetting.Upload.s3_base_url}"
|
|
|
|
to = SiteSetting.s3_cdn_url
|
|
|
|
DbHelper.remap(from, to, exclude_tables: exclude_tables)
|
|
|
|
puts "Remapping #{from} --> #{to} took #{Process.clock_gettime(Process::CLOCK_MONOTONIC) - now} seconds"
|
|
|
|
end
|
|
|
|
end
|
2015-05-25 11:59:00 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
################################################################################
|
|
|
|
# clean_up #
|
|
|
|
################################################################################
|
|
|
|
|
2014-09-29 12:31:53 -04:00
|
|
|
task "uploads:clean_up" => :environment do
|
2016-09-02 02:50:13 -04:00
|
|
|
if ENV["RAILS_DB"]
|
|
|
|
clean_up_uploads
|
|
|
|
else
|
|
|
|
RailsMultisite::ConnectionManagement.each_connection { clean_up_uploads }
|
|
|
|
end
|
|
|
|
end
|
2014-09-29 12:31:53 -04:00
|
|
|
|
2016-09-02 02:50:13 -04:00
|
|
|
def clean_up_uploads
|
|
|
|
db = RailsMultisite::ConnectionManagement.current_db
|
2014-09-29 12:31:53 -04:00
|
|
|
|
2016-09-02 02:50:13 -04:00
|
|
|
puts "Cleaning up uploads and thumbnails for '#{db}'..."
|
2014-09-29 12:31:53 -04:00
|
|
|
|
2016-09-02 02:50:13 -04:00
|
|
|
if Discourse.store.external?
|
|
|
|
puts "This task only works for internal storages."
|
|
|
|
exit 1
|
|
|
|
end
|
2014-09-29 12:31:53 -04:00
|
|
|
|
2016-09-02 02:50:13 -04:00
|
|
|
puts <<~OUTPUT
|
|
|
|
This task will remove upload records and files permanently.
|
2014-09-29 12:31:53 -04:00
|
|
|
|
2016-09-02 02:50:13 -04:00
|
|
|
Would you like to take a full backup before the clean up? (Y/N)
|
|
|
|
OUTPUT
|
2014-09-29 12:31:53 -04:00
|
|
|
|
2016-09-02 02:50:13 -04:00
|
|
|
if STDIN.gets.chomp.downcase == 'y'
|
|
|
|
puts "Starting backup..."
|
|
|
|
backuper = BackupRestore::Backuper.new(Discourse.system_user.id)
|
|
|
|
backuper.run
|
|
|
|
exit 1 unless backuper.success
|
|
|
|
end
|
2014-09-29 12:31:53 -04:00
|
|
|
|
2016-09-02 02:50:13 -04:00
|
|
|
public_directory = Rails.root.join("public").to_s
|
2014-09-29 12:31:53 -04:00
|
|
|
|
2016-09-02 02:50:13 -04:00
|
|
|
##
|
|
|
|
## DATABASE vs FILE SYSTEM
|
|
|
|
##
|
2014-09-29 12:31:53 -04:00
|
|
|
|
2016-09-02 02:50:13 -04:00
|
|
|
# uploads & avatars
|
|
|
|
Upload.find_each do |upload|
|
|
|
|
path = File.join(public_directory, upload.url)
|
2014-09-29 12:31:53 -04:00
|
|
|
|
2016-09-02 02:50:13 -04:00
|
|
|
if !File.exists?(path)
|
|
|
|
upload.destroy!
|
|
|
|
putc "#"
|
|
|
|
else
|
|
|
|
putc "."
|
2014-09-29 12:31:53 -04:00
|
|
|
end
|
2016-09-02 02:50:13 -04:00
|
|
|
end
|
2014-09-29 12:31:53 -04:00
|
|
|
|
2016-09-02 02:50:13 -04:00
|
|
|
# optimized images
|
|
|
|
OptimizedImage.find_each do |optimized_image|
|
|
|
|
path = File.join(public_directory, optimized_image.url)
|
|
|
|
|
|
|
|
if !File.exists?(path)
|
|
|
|
optimized_image.destroy!
|
|
|
|
putc "#"
|
|
|
|
else
|
|
|
|
putc "."
|
2014-09-29 12:31:53 -04:00
|
|
|
end
|
2016-09-02 02:50:13 -04:00
|
|
|
end
|
2014-09-29 12:31:53 -04:00
|
|
|
|
2016-09-02 02:50:13 -04:00
|
|
|
##
|
|
|
|
## FILE SYSTEM vs DATABASE
|
|
|
|
##
|
2014-09-29 12:31:53 -04:00
|
|
|
|
2016-09-02 02:50:13 -04:00
|
|
|
uploads_directory = File.join(public_directory, 'uploads', db).to_s
|
|
|
|
|
|
|
|
# avatars (no avatar should be stored in that old directory)
|
|
|
|
FileUtils.rm_rf("#{uploads_directory}/avatars")
|
|
|
|
|
|
|
|
# uploads and optimized images
|
|
|
|
Dir.glob("#{uploads_directory}/**/*.*").each do |file_path|
|
|
|
|
sha1 = Upload.generate_digest(file_path)
|
|
|
|
url = file_path.split(public_directory, 2)[1]
|
|
|
|
|
|
|
|
if (Upload.where(sha1: sha1).empty? &&
|
|
|
|
Upload.where(url: url).empty?) &&
|
|
|
|
(OptimizedImage.where(sha1: sha1).empty? &&
|
|
|
|
OptimizedImage.where(url: url).empty?)
|
|
|
|
|
|
|
|
FileUtils.rm(file_path)
|
|
|
|
putc "#"
|
|
|
|
else
|
|
|
|
putc "."
|
|
|
|
end
|
2014-09-29 12:31:53 -04:00
|
|
|
end
|
|
|
|
|
2016-09-02 02:50:13 -04:00
|
|
|
puts "Removing empty directories..."
|
|
|
|
puts `find #{uploads_directory} -type d -empty -exec rmdir {} \\;`
|
|
|
|
|
|
|
|
puts "Done!"
|
2014-09-29 12:31:53 -04:00
|
|
|
end
|
2015-05-10 20:30:22 -04:00
|
|
|
|
2015-05-25 11:59:00 -04:00
|
|
|
################################################################################
|
|
|
|
# missing #
|
|
|
|
################################################################################
|
2015-05-10 20:30:22 -04:00
|
|
|
|
|
|
|
# list all missing uploads and optimized images
|
|
|
|
task "uploads:missing" => :environment do
|
2016-09-01 22:22:03 -04:00
|
|
|
if ENV["RAILS_DB"]
|
2018-03-12 08:08:53 -04:00
|
|
|
list_missing_uploads(skip_optimized: ENV['SKIP_OPTIMIZED'])
|
2016-09-01 22:22:03 -04:00
|
|
|
else
|
|
|
|
RailsMultisite::ConnectionManagement.each_connection do |db|
|
2018-03-12 08:08:53 -04:00
|
|
|
list_missing_uploads(skip_optimized: ENV['SKIP_OPTIMIZED'])
|
2015-05-10 20:30:22 -04:00
|
|
|
end
|
2016-09-01 22:22:03 -04:00
|
|
|
end
|
|
|
|
end
|
2015-05-10 20:30:22 -04:00
|
|
|
|
2018-03-12 08:08:53 -04:00
|
|
|
def list_missing_uploads(skip_optimized: false)
|
2018-11-26 14:24:51 -05:00
|
|
|
Discourse.store.list_missing_uploads(skip_optimized: skip_optimized)
|
2015-05-10 20:30:22 -04:00
|
|
|
end
|
2015-05-11 06:59:50 -04:00
|
|
|
|
2016-09-21 04:50:08 -04:00
|
|
|
################################################################################
|
|
|
|
# Recover from tombstone #
|
|
|
|
################################################################################
|
|
|
|
|
|
|
|
task "uploads:recover_from_tombstone" => :environment do
|
|
|
|
if ENV["RAILS_DB"]
|
|
|
|
recover_from_tombstone
|
|
|
|
else
|
|
|
|
RailsMultisite::ConnectionManagement.each_connection { recover_from_tombstone }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def recover_from_tombstone
|
|
|
|
if Discourse.store.external?
|
|
|
|
puts "This task only works for internal storages."
|
|
|
|
return
|
|
|
|
end
|
|
|
|
|
2016-09-21 05:47:27 -04:00
|
|
|
begin
|
2018-03-28 09:51:47 -04:00
|
|
|
previous_image_size = SiteSetting.max_image_size_kb
|
|
|
|
previous_attachment_size = SiteSetting.max_attachment_size_kb
|
|
|
|
previous_extensions = SiteSetting.authorized_extensions
|
|
|
|
|
|
|
|
SiteSetting.max_image_size_kb = 10 * 1024
|
|
|
|
SiteSetting.max_attachment_size_kb = 10 * 1024
|
|
|
|
SiteSetting.authorized_extensions = "*"
|
|
|
|
|
2016-09-22 00:28:36 -04:00
|
|
|
current_db = RailsMultisite::ConnectionManagement.current_db
|
2016-09-21 05:47:27 -04:00
|
|
|
public_path = Rails.root.join("public")
|
2016-09-22 00:28:36 -04:00
|
|
|
paths = Dir.glob(File.join(public_path, 'uploads', 'tombstone', current_db, '**', '*.*'))
|
2018-01-19 09:40:17 -05:00
|
|
|
max = paths.size
|
2016-09-21 05:47:27 -04:00
|
|
|
|
2016-09-21 22:39:39 -04:00
|
|
|
paths.each_with_index do |path, index|
|
2016-09-21 06:01:59 -04:00
|
|
|
filename = File.basename(path)
|
2016-09-21 22:39:39 -04:00
|
|
|
printf("%9d / %d (%5.1f%%)\n", (index + 1), max, (((index + 1).to_f / max.to_f) * 100).round(1))
|
2016-09-21 05:47:27 -04:00
|
|
|
|
2018-01-19 09:40:17 -05:00
|
|
|
Post.where("raw LIKE ?", "%#{filename}%").find_each do |post|
|
2016-09-21 06:01:59 -04:00
|
|
|
doc = Nokogiri::HTML::fragment(post.raw)
|
2016-09-21 22:39:39 -04:00
|
|
|
updated = false
|
2016-09-21 04:50:08 -04:00
|
|
|
|
2018-03-28 09:51:47 -04:00
|
|
|
image_urls = doc.css("img[src]").map { |img| img["src"] }
|
|
|
|
attachment_urls = doc.css("a.attachment[href]").map { |a| a["href"] }
|
2016-09-21 04:50:08 -04:00
|
|
|
|
2018-03-28 09:51:47 -04:00
|
|
|
(image_urls + attachment_urls).each do |url|
|
2018-01-19 09:40:17 -05:00
|
|
|
next if !url.start_with?("/uploads/")
|
|
|
|
next if Upload.exists?(url: url)
|
2016-09-21 04:50:08 -04:00
|
|
|
|
2018-01-19 09:40:17 -05:00
|
|
|
puts "Restoring #{path}..."
|
|
|
|
tombstone_path = File.join(public_path, 'uploads', 'tombstone', url.gsub(/^\/uploads\//, ""))
|
2016-09-21 06:01:59 -04:00
|
|
|
|
2018-01-19 09:40:17 -05:00
|
|
|
if File.exists?(tombstone_path)
|
|
|
|
File.open(tombstone_path) do |file|
|
|
|
|
new_upload = UploadCreator.new(file, File.basename(url)).create_for(Discourse::SYSTEM_USER_ID)
|
2016-09-21 06:01:59 -04:00
|
|
|
|
2018-01-19 09:40:17 -05:00
|
|
|
if new_upload.persisted?
|
|
|
|
puts "Restored into #{new_upload.url}"
|
|
|
|
DbHelper.remap(url, new_upload.url)
|
|
|
|
updated = true
|
|
|
|
else
|
|
|
|
puts "Failed to create upload for #{url}: #{new_upload.errors.full_messages}."
|
2016-09-21 05:47:27 -04:00
|
|
|
end
|
|
|
|
end
|
2018-01-19 09:40:17 -05:00
|
|
|
else
|
|
|
|
puts "Failed to find file (#{tombstone_path}) in tombstone."
|
2016-09-21 04:50:08 -04:00
|
|
|
end
|
|
|
|
end
|
2016-09-21 22:39:39 -04:00
|
|
|
|
|
|
|
post.rebake! if updated
|
2016-09-21 04:50:08 -04:00
|
|
|
end
|
2018-01-19 09:40:17 -05:00
|
|
|
|
|
|
|
sha1 = File.basename(filename, File.extname(filename))
|
|
|
|
short_url = "upload://#{Base62.encode(sha1.hex)}"
|
|
|
|
|
|
|
|
Post.where("raw LIKE ?", "%#{short_url}%").find_each do |post|
|
|
|
|
puts "Restoring #{path}..."
|
|
|
|
|
|
|
|
File.open(path) do |file|
|
|
|
|
new_upload = UploadCreator.new(file, filename).create_for(Discourse::SYSTEM_USER_ID)
|
|
|
|
|
|
|
|
if new_upload.persisted?
|
|
|
|
puts "Restored into #{new_upload.short_url}"
|
|
|
|
DbHelper.remap(short_url, new_upload.short_url) if short_url != new_upload.short_url
|
|
|
|
post.rebake!
|
|
|
|
else
|
|
|
|
puts "Failed to create upload for #{filename}: #{new_upload.errors.full_messages}."
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2016-09-21 04:50:08 -04:00
|
|
|
end
|
2016-09-21 05:47:27 -04:00
|
|
|
ensure
|
2018-03-28 09:51:47 -04:00
|
|
|
SiteSetting.max_image_size_kb = previous_image_size
|
|
|
|
SiteSetting.max_attachment_size_kb = previous_attachment_size
|
|
|
|
SiteSetting.authorized_extensions = previous_extensions
|
2016-09-21 04:50:08 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2015-05-25 11:59:00 -04:00
|
|
|
################################################################################
|
|
|
|
# regenerate_missing_optimized #
|
|
|
|
################################################################################
|
|
|
|
|
2015-05-11 06:59:50 -04:00
|
|
|
# regenerate missing optimized images
|
|
|
|
task "uploads:regenerate_missing_optimized" => :environment do
|
2016-09-02 01:06:31 -04:00
|
|
|
if ENV["RAILS_DB"]
|
|
|
|
regenerate_missing_optimized
|
|
|
|
else
|
|
|
|
RailsMultisite::ConnectionManagement.each_connection { regenerate_missing_optimized }
|
|
|
|
end
|
2015-05-11 10:19:16 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def regenerate_missing_optimized
|
2015-05-11 13:07:39 -04:00
|
|
|
db = RailsMultisite::ConnectionManagement.current_db
|
|
|
|
|
|
|
|
puts "Regenerating missing optimized images for '#{db}'..."
|
2015-05-11 06:59:50 -04:00
|
|
|
|
|
|
|
if Discourse.store.external?
|
|
|
|
puts "This task only works for internal storages."
|
|
|
|
return
|
|
|
|
end
|
|
|
|
|
|
|
|
public_directory = "#{Rails.root}/public"
|
|
|
|
missing_uploads = Set.new
|
|
|
|
|
2016-08-25 06:29:52 -04:00
|
|
|
avatar_upload_ids = UserAvatar.all.pluck(:custom_upload_id, :gravatar_upload_id).flatten.compact
|
2015-05-11 06:59:50 -04:00
|
|
|
|
2016-08-25 06:29:52 -04:00
|
|
|
default_scope = OptimizedImage.includes(:upload)
|
2015-05-11 13:07:39 -04:00
|
|
|
|
2016-08-25 06:29:52 -04:00
|
|
|
[
|
|
|
|
default_scope
|
|
|
|
.where("optimized_images.upload_id IN (?)", avatar_upload_ids),
|
2015-05-11 09:41:52 -04:00
|
|
|
|
2016-08-25 06:29:52 -04:00
|
|
|
default_scope
|
|
|
|
.where("optimized_images.upload_id NOT IN (?)", avatar_upload_ids)
|
|
|
|
.where("LENGTH(COALESCE(url, '')) > 0")
|
|
|
|
.where("width > 0 AND height > 0")
|
|
|
|
].each do |scope|
|
|
|
|
scope.find_each do |optimized_image|
|
|
|
|
upload = optimized_image.upload
|
2015-05-11 06:59:50 -04:00
|
|
|
|
2016-08-25 06:29:52 -04:00
|
|
|
next unless optimized_image.url =~ /^\/[^\/]/
|
|
|
|
next unless upload.url =~ /^\/[^\/]/
|
|
|
|
|
|
|
|
thumbnail = "#{public_directory}#{optimized_image.url}"
|
|
|
|
original = "#{public_directory}#{upload.url}"
|
|
|
|
|
|
|
|
if !File.exists?(thumbnail) || File.size(thumbnail) <= 0
|
|
|
|
# make sure the original image exists locally
|
|
|
|
if (!File.exists?(original) || File.size(original) <= 0) && upload.origin.present?
|
|
|
|
# try to fix it by redownloading it
|
|
|
|
begin
|
2017-05-24 13:42:52 -04:00
|
|
|
downloaded = FileHelper.download(
|
|
|
|
upload.origin,
|
|
|
|
max_file_size: SiteSetting.max_image_size_kb.kilobytes,
|
|
|
|
tmp_file_name: "discourse-missing",
|
|
|
|
follow_redirect: true
|
|
|
|
) rescue nil
|
2016-08-25 06:29:52 -04:00
|
|
|
if downloaded && downloaded.size > 0
|
|
|
|
FileUtils.mkdir_p(File.dirname(original))
|
|
|
|
File.open(original, "wb") { |f| f.write(downloaded.read) }
|
|
|
|
end
|
|
|
|
ensure
|
|
|
|
downloaded.try(:close!) if downloaded.respond_to?(:close!)
|
2015-05-11 13:07:39 -04:00
|
|
|
end
|
2015-05-11 11:03:48 -04:00
|
|
|
end
|
|
|
|
|
2016-08-25 06:29:52 -04:00
|
|
|
if File.exists?(original) && File.size(original) > 0
|
|
|
|
FileUtils.mkdir_p(File.dirname(thumbnail))
|
|
|
|
OptimizedImage.resize(original, thumbnail, optimized_image.width, optimized_image.height)
|
|
|
|
putc "#"
|
|
|
|
else
|
|
|
|
missing_uploads << original
|
|
|
|
putc "X"
|
|
|
|
end
|
2015-05-11 06:59:50 -04:00
|
|
|
else
|
2016-08-25 06:29:52 -04:00
|
|
|
putc "."
|
2015-05-11 06:59:50 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
puts "", "Done"
|
|
|
|
|
|
|
|
if missing_uploads.size > 0
|
|
|
|
puts "Missing uploads:"
|
|
|
|
missing_uploads.sort.each { |u| puts u }
|
|
|
|
end
|
|
|
|
end
|
2015-05-19 06:31:51 -04:00
|
|
|
|
2015-05-25 11:59:00 -04:00
|
|
|
################################################################################
|
2015-06-12 06:02:36 -04:00
|
|
|
# migrate_to_new_scheme #
|
2015-05-25 11:59:00 -04:00
|
|
|
################################################################################
|
|
|
|
|
2015-06-12 06:02:36 -04:00
|
|
|
task "uploads:start_migration" => :environment do
|
|
|
|
SiteSetting.migrate_to_new_scheme = true
|
|
|
|
puts "Migration started!"
|
2015-05-19 06:31:51 -04:00
|
|
|
end
|
|
|
|
|
2015-06-12 06:02:36 -04:00
|
|
|
task "uploads:stop_migration" => :environment do
|
|
|
|
SiteSetting.migrate_to_new_scheme = false
|
|
|
|
puts "Migration stoped!"
|
2015-05-19 06:31:51 -04:00
|
|
|
end
|
2016-09-01 03:19:14 -04:00
|
|
|
|
|
|
|
task "uploads:analyze", [:cache_path, :limit] => :environment do |_, args|
|
|
|
|
now = Time.zone.now
|
|
|
|
current_db = RailsMultisite::ConnectionManagement.current_db
|
|
|
|
|
|
|
|
puts "Analyzing uploads for '#{current_db}'... This may take awhile...\n"
|
|
|
|
cache_path = args[:cache_path]
|
|
|
|
|
|
|
|
current_db = RailsMultisite::ConnectionManagement.current_db
|
|
|
|
uploads_path = Rails.root.join('public', 'uploads', current_db)
|
|
|
|
|
|
|
|
path =
|
|
|
|
if cache_path
|
|
|
|
cache_path
|
|
|
|
else
|
|
|
|
path = "/tmp/#{current_db}-#{now.to_i}-paths.txt"
|
|
|
|
FileUtils.touch("/tmp/#{now.to_i}-paths.txt")
|
|
|
|
`find #{uploads_path} -type f -printf '%s %h/%f\n' > #{path}`
|
|
|
|
path
|
|
|
|
end
|
|
|
|
|
|
|
|
extensions = {}
|
|
|
|
paths_count = 0
|
|
|
|
|
|
|
|
File.readlines(path).each do |line|
|
|
|
|
size, file_path = line.split(" ", 2)
|
|
|
|
|
|
|
|
paths_count += 1
|
|
|
|
extension = File.extname(file_path).chomp.downcase
|
|
|
|
extensions[extension] ||= {}
|
|
|
|
extensions[extension]["count"] ||= 0
|
|
|
|
extensions[extension]["count"] += 1
|
|
|
|
extensions[extension]["size"] ||= 0
|
|
|
|
extensions[extension]["size"] += size.to_i
|
|
|
|
end
|
|
|
|
|
|
|
|
uploads_count = Upload.count
|
|
|
|
optimized_images_count = OptimizedImage.count
|
|
|
|
|
|
|
|
puts <<~REPORT
|
|
|
|
Report for '#{current_db}'
|
|
|
|
-----------#{'-' * current_db.length}
|
|
|
|
Number of `Upload` records in DB: #{uploads_count}
|
|
|
|
Number of `OptimizedImage` records in DB: #{optimized_images_count}
|
|
|
|
**Total DB records: #{uploads_count + optimized_images_count}**
|
|
|
|
|
|
|
|
Number of images in uploads folder: #{paths_count}
|
|
|
|
------------------------------------#{'-' * paths_count.to_s.length}
|
|
|
|
|
|
|
|
REPORT
|
|
|
|
|
|
|
|
helper = Class.new do
|
|
|
|
include ActionView::Helpers::NumberHelper
|
|
|
|
end
|
|
|
|
|
|
|
|
helper = helper.new
|
|
|
|
|
|
|
|
printf "%-15s | %-15s | %-15s\n", 'extname', 'total size', 'count'
|
|
|
|
puts "-" * 45
|
|
|
|
|
|
|
|
extensions.sort_by { |_, value| value['size'] }.reverse.each do |extname, value|
|
|
|
|
printf "%-15s | %-15s | %-15s\n", extname, helper.number_to_human_size(value['size']), value['count']
|
|
|
|
end
|
|
|
|
|
|
|
|
puts "\n"
|
|
|
|
|
|
|
|
limit = args[:limit] || 10
|
|
|
|
|
|
|
|
sql = <<~SQL
|
|
|
|
SELECT
|
|
|
|
users.username,
|
|
|
|
COUNT(uploads.user_id) AS num_of_uploads,
|
|
|
|
SUM(uploads.filesize) AS total_size_of_uploads,
|
|
|
|
COUNT(optimized_images.id) AS num_of_optimized_images
|
|
|
|
FROM users
|
|
|
|
INNER JOIN uploads ON users.id = uploads.user_id
|
|
|
|
INNER JOIN optimized_images ON uploads.id = optimized_images.upload_id
|
|
|
|
GROUP BY users.id
|
|
|
|
ORDER BY total_size_of_uploads DESC
|
|
|
|
LIMIT #{limit}
|
|
|
|
SQL
|
|
|
|
|
|
|
|
puts "Users using the most disk space"
|
|
|
|
puts "-------------------------------\n"
|
|
|
|
printf "%-25s | %-25s | %-25s | %-25s\n", 'username', 'total size of uploads', 'number of uploads', 'number of optimized images'
|
|
|
|
puts "-" * 110
|
|
|
|
|
2018-06-19 02:13:14 -04:00
|
|
|
DB.query_single(sql).each do |username, num_of_uploads, total_size_of_uploads, num_of_optimized_images|
|
2016-09-01 03:19:14 -04:00
|
|
|
printf "%-25s | %-25s | %-25s | %-25s\n", username, helper.number_to_human_size(total_size_of_uploads), num_of_uploads, num_of_optimized_images
|
|
|
|
end
|
|
|
|
|
|
|
|
puts "\n"
|
|
|
|
puts "List of file paths @ #{path}"
|
|
|
|
puts "Duration: #{Time.zone.now - now} seconds"
|
|
|
|
end
|
2018-08-08 01:14:52 -04:00
|
|
|
|
|
|
|
task "uploads:fix_incorrect_extensions" => :environment do
|
2018-08-08 03:39:00 -04:00
|
|
|
require_dependency "upload_fixer"
|
2018-08-09 21:28:05 -04:00
|
|
|
UploadFixer.fix_all_extensions
|
2018-08-08 01:14:52 -04:00
|
|
|
end
|
2018-09-05 04:54:15 -04:00
|
|
|
|
2018-09-12 04:51:53 -04:00
|
|
|
task "uploads:recover" => :environment do
|
|
|
|
require_dependency "upload_recovery"
|
|
|
|
|
2018-09-12 09:53:01 -04:00
|
|
|
dry_run = ENV["DRY_RUN"].present?
|
|
|
|
|
2018-09-05 04:54:15 -04:00
|
|
|
if ENV["RAILS_DB"]
|
2018-09-12 09:53:01 -04:00
|
|
|
UploadRecovery.new(dry_run: dry_run).recover
|
2018-09-05 04:54:15 -04:00
|
|
|
else
|
|
|
|
RailsMultisite::ConnectionManagement.each_connection do |db|
|
2018-09-12 09:53:01 -04:00
|
|
|
UploadRecovery.new(dry_run: dry_run).recover
|
2018-09-10 03:14:30 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|