2013-06-19 15:20:30 +08:00
|
|
|
desc 'Update each post with latest markdown'
|
|
|
|
task 'posts:rebake' => :environment do
|
2014-04-04 23:10:47 +08:00
|
|
|
ENV['RAILS_DB'] ? rebake_posts : rebake_posts_all_sites
|
2013-06-19 15:20:30 +08:00
|
|
|
end
|
|
|
|
|
2017-09-12 15:10:18 +08:00
|
|
|
task 'posts:rebake_uncooked_posts' => :environment do
|
|
|
|
uncooked = Post.where(baked_version: nil)
|
|
|
|
|
|
|
|
rebaked = 0
|
|
|
|
total = uncooked.count
|
|
|
|
|
|
|
|
uncooked.find_each do |post|
|
|
|
|
rebake_post(post)
|
|
|
|
print_status(rebaked += 1, total)
|
|
|
|
end
|
|
|
|
|
|
|
|
puts "", "#{rebaked} posts done!", ""
|
|
|
|
end
|
|
|
|
|
2013-06-19 15:20:30 +08:00
|
|
|
desc 'Update each post with latest markdown and refresh oneboxes'
|
|
|
|
task 'posts:refresh_oneboxes' => :environment do
|
2014-04-04 23:10:47 +08:00
|
|
|
ENV['RAILS_DB'] ? rebake_posts(invalidate_oneboxes: true) : rebake_posts_all_sites(invalidate_oneboxes: true)
|
2013-06-19 15:20:30 +08:00
|
|
|
end
|
|
|
|
|
2015-09-24 04:44:53 +08:00
|
|
|
desc 'Rebake all posts with a quote using a letter_avatar'
|
|
|
|
task 'posts:fix_letter_avatars' => :environment do
|
|
|
|
return unless SiteSetting.external_system_avatars_enabled
|
|
|
|
|
|
|
|
search = Post.where("user_id <> -1")
|
2017-07-28 09:20:09 +08:00
|
|
|
.where("raw LIKE '%/letter\_avatar/%' OR cooked LIKE '%/letter\_avatar/%'")
|
2015-09-24 04:44:53 +08:00
|
|
|
|
|
|
|
rebaked = 0
|
|
|
|
total = search.count
|
|
|
|
|
2016-08-25 16:10:27 +08:00
|
|
|
search.find_each do |post|
|
2015-09-24 04:44:53 +08:00
|
|
|
rebake_post(post)
|
|
|
|
print_status(rebaked += 1, total)
|
|
|
|
end
|
|
|
|
|
|
|
|
puts "", "#{rebaked} posts done!", ""
|
|
|
|
end
|
|
|
|
|
2016-09-22 00:14:53 +08:00
|
|
|
desc 'Rebake all posts matching string/regex and optionally delay the loop'
|
2017-07-28 09:20:09 +08:00
|
|
|
task 'posts:rebake_match', [:pattern, :type, :delay] => [:environment] do |_, args|
|
2017-10-04 08:47:53 +08:00
|
|
|
args.with_defaults(type: 'string')
|
2016-08-17 01:18:28 +08:00
|
|
|
pattern = args[:pattern]
|
2017-10-04 08:47:53 +08:00
|
|
|
type = args[:type]&.downcase
|
|
|
|
delay = args[:delay]&.to_i
|
|
|
|
|
2016-08-17 01:18:28 +08:00
|
|
|
if !pattern
|
2016-09-22 00:14:53 +08:00
|
|
|
puts "ERROR: Expecting rake posts:rebake_match[pattern,type,delay]"
|
|
|
|
exit 1
|
|
|
|
elsif delay && delay < 1
|
|
|
|
puts "ERROR: delay parameter should be an integer and greater than 0"
|
2016-08-17 01:18:28 +08:00
|
|
|
exit 1
|
2017-10-04 08:47:53 +08:00
|
|
|
elsif type != 'string' && type != 'regex'
|
2016-08-17 01:18:28 +08:00
|
|
|
puts "ERROR: Expecting rake posts:rebake_match[pattern,type] where type is string or regex"
|
|
|
|
exit 1
|
|
|
|
end
|
|
|
|
|
2017-10-25 02:49:00 +08:00
|
|
|
search = Post.raw_match(pattern, type)
|
|
|
|
|
2016-08-17 01:18:28 +08:00
|
|
|
rebaked = 0
|
|
|
|
total = search.count
|
|
|
|
|
2017-10-25 02:49:00 +08:00
|
|
|
search.find_each do |post|
|
2016-08-17 01:18:28 +08:00
|
|
|
rebake_post(post)
|
|
|
|
print_status(rebaked += 1, total)
|
2016-09-22 00:14:53 +08:00
|
|
|
sleep(delay) if delay
|
2016-08-17 01:18:28 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
puts "", "#{rebaked} posts done!", ""
|
|
|
|
end
|
|
|
|
|
2014-04-04 23:10:47 +08:00
|
|
|
def rebake_posts_all_sites(opts = {})
|
2013-02-06 03:16:51 +08:00
|
|
|
RailsMultisite::ConnectionManagement.each_connection do |db|
|
2014-04-04 23:10:47 +08:00
|
|
|
rebake_posts(opts)
|
|
|
|
end
|
|
|
|
end
|
2013-06-19 15:20:30 +08:00
|
|
|
|
2014-04-04 23:10:47 +08:00
|
|
|
def rebake_posts(opts = {})
|
2014-09-10 01:25:20 +08:00
|
|
|
puts "Rebaking post markdown for '#{RailsMultisite::ConnectionManagement.current_db}'"
|
|
|
|
|
2015-04-02 05:08:29 +08:00
|
|
|
disable_edit_notifications = SiteSetting.disable_edit_notifications
|
|
|
|
SiteSetting.disable_edit_notifications = true
|
|
|
|
|
2014-09-10 01:25:20 +08:00
|
|
|
total = Post.count
|
|
|
|
rebaked = 0
|
2013-02-06 03:16:51 +08:00
|
|
|
|
2016-08-25 16:10:27 +08:00
|
|
|
Post.find_each do |post|
|
2014-09-10 01:25:20 +08:00
|
|
|
rebake_post(post, opts)
|
|
|
|
print_status(rebaked += 1, total)
|
2013-02-06 03:16:51 +08:00
|
|
|
end
|
2014-04-04 23:10:47 +08:00
|
|
|
|
2015-04-02 05:08:29 +08:00
|
|
|
SiteSetting.disable_edit_notifications = disable_edit_notifications
|
|
|
|
|
2014-09-10 01:25:20 +08:00
|
|
|
puts "", "#{rebaked} posts done!", "-" * 50
|
2013-02-06 03:16:51 +08:00
|
|
|
end
|
2014-06-06 12:08:39 +08:00
|
|
|
|
2015-09-24 04:44:53 +08:00
|
|
|
def rebake_post(post, opts = {})
|
2014-09-10 01:25:20 +08:00
|
|
|
post.rebake!(opts)
|
|
|
|
rescue => e
|
|
|
|
puts "", "Failed to rebake (topic_id: #{post.topic_id}, post_id: #{post.id})", e, e.backtrace.join("\n")
|
|
|
|
end
|
|
|
|
|
|
|
|
def print_status(current, max)
|
|
|
|
print "\r%9d / %d (%5.1f%%)" % [current, max, ((current.to_f / max.to_f) * 100).round(1)]
|
|
|
|
end
|
2014-06-06 12:08:39 +08:00
|
|
|
|
|
|
|
desc 'normalize all markdown so <pre><code> is not used and instead backticks'
|
|
|
|
task 'posts:normalize_code' => :environment do
|
|
|
|
lang = ENV['CODE_LANG'] || ''
|
|
|
|
require 'import/normalize'
|
|
|
|
|
|
|
|
puts "Normalizing"
|
|
|
|
i = 0
|
|
|
|
Post.where("raw like '%<pre>%<code>%'").each do |p|
|
|
|
|
normalized = Import::Normalize.normalize_code_blocks(p.raw, lang)
|
|
|
|
if normalized != p.raw
|
2017-07-28 09:20:09 +08:00
|
|
|
p.revise(Discourse.system_user, raw: normalized)
|
2014-06-06 12:08:39 +08:00
|
|
|
putc "."
|
|
|
|
i += 1
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
puts
|
|
|
|
puts "#{i} posts normalized!"
|
|
|
|
end
|
2016-08-17 15:17:07 +08:00
|
|
|
|
2017-10-04 08:47:53 +08:00
|
|
|
def remap_posts(find, type, replace = "")
|
2017-01-26 00:20:21 +08:00
|
|
|
i = 0
|
2017-10-04 08:47:53 +08:00
|
|
|
|
|
|
|
Post.raw_match(find, type).find_each do |p|
|
|
|
|
new_raw =
|
|
|
|
case type
|
|
|
|
when 'string' then p.raw.gsub(/#{Regexp.escape(find)}/, replace)
|
|
|
|
when 'regex' then p.raw.gsub(/#{find}/, replace)
|
|
|
|
end
|
2017-01-26 00:20:21 +08:00
|
|
|
|
|
|
|
if new_raw != p.raw
|
2017-12-28 22:13:01 +08:00
|
|
|
begin
|
|
|
|
p.revise(Discourse.system_user, { raw: new_raw }, bypass_bump: true, skip_revision: true)
|
|
|
|
putc "."
|
|
|
|
i += 1
|
|
|
|
rescue
|
|
|
|
puts "\nFailed to remap post (topic_id: #{p.topic_id}, post_id: #{p.id})\n"
|
|
|
|
end
|
2017-01-26 00:20:21 +08:00
|
|
|
end
|
|
|
|
end
|
2017-10-04 08:47:53 +08:00
|
|
|
|
2017-01-26 00:20:21 +08:00
|
|
|
i
|
|
|
|
end
|
|
|
|
|
2016-08-17 15:17:07 +08:00
|
|
|
desc 'Remap all posts matching specific string'
|
2017-10-04 08:47:53 +08:00
|
|
|
task 'posts:remap', [:find, :replace, :type] => [:environment] do |_, args|
|
|
|
|
require 'highline/import'
|
2017-01-24 14:30:57 +08:00
|
|
|
|
2017-10-04 08:47:53 +08:00
|
|
|
args.with_defaults(type: 'string')
|
2016-08-17 15:17:07 +08:00
|
|
|
find = args[:find]
|
|
|
|
replace = args[:replace]
|
2017-10-04 08:47:53 +08:00
|
|
|
type = args[:type]&.downcase
|
|
|
|
|
2017-01-24 14:30:57 +08:00
|
|
|
if !find
|
2017-01-26 00:20:21 +08:00
|
|
|
puts "ERROR: Expecting rake posts:remap['find','replace']"
|
2016-08-17 15:17:07 +08:00
|
|
|
exit 1
|
2017-01-24 14:30:57 +08:00
|
|
|
elsif !replace
|
2017-01-26 00:20:21 +08:00
|
|
|
puts "ERROR: Expecting rake posts:remap['find','replace']. Want to delete a word/string instead? Try rake posts:delete_word['word-to-delete']"
|
|
|
|
exit 1
|
2017-10-04 08:47:53 +08:00
|
|
|
elsif type != 'string' && type != 'regex'
|
|
|
|
puts "ERROR: Expecting rake posts:delete_word[pattern, type] where type is string or regex"
|
|
|
|
exit 1
|
|
|
|
else
|
|
|
|
confirm_replace = ask("Are you sure you want to replace all #{type} occurrences of '#{find}' with '#{replace}'? (Y/n)")
|
|
|
|
exit 1 unless (confirm_replace == "" || confirm_replace.downcase == 'y')
|
2016-08-17 15:17:07 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
puts "Remapping"
|
2017-10-04 08:47:53 +08:00
|
|
|
total = remap_posts(find, type, replace)
|
2017-01-26 00:20:21 +08:00
|
|
|
puts "", "#{total} posts remapped!", ""
|
|
|
|
end
|
2016-08-17 15:17:07 +08:00
|
|
|
|
2017-01-26 00:20:21 +08:00
|
|
|
desc 'Delete occurrence of a word/string'
|
2017-10-04 08:47:53 +08:00
|
|
|
task 'posts:delete_word', [:find, :type] => [:environment] do |_, args|
|
2017-01-26 00:20:21 +08:00
|
|
|
require 'highline/import'
|
|
|
|
|
2017-10-04 08:47:53 +08:00
|
|
|
args.with_defaults(type: 'string')
|
2017-01-26 00:20:21 +08:00
|
|
|
find = args[:find]
|
2017-10-04 08:47:53 +08:00
|
|
|
type = args[:type]&.downcase
|
|
|
|
|
2017-01-26 00:20:21 +08:00
|
|
|
if !find
|
|
|
|
puts "ERROR: Expecting rake posts:delete_word['word-to-delete']"
|
|
|
|
exit 1
|
2017-10-04 08:47:53 +08:00
|
|
|
elsif type != 'string' && type != 'regex'
|
|
|
|
puts "ERROR: Expecting rake posts:delete_word[pattern, type] where type is string or regex"
|
|
|
|
exit 1
|
2017-01-26 00:20:21 +08:00
|
|
|
else
|
2017-10-04 08:47:53 +08:00
|
|
|
confirm_delete = ask("Are you sure you want to remove all #{type} occurrences of '#{find}'? (Y/n)")
|
|
|
|
exit 1 unless (confirm_delete == "" || confirm_delete.downcase == 'y')
|
2016-08-17 15:17:07 +08:00
|
|
|
end
|
2017-01-26 00:20:21 +08:00
|
|
|
|
|
|
|
puts "Processing"
|
2017-10-04 08:47:53 +08:00
|
|
|
total = remap_posts(find, type)
|
2017-01-26 00:20:21 +08:00
|
|
|
puts "", "#{total} posts updated!", ""
|
2016-08-17 15:17:07 +08:00
|
|
|
end
|
2017-06-23 23:33:44 +08:00
|
|
|
|
|
|
|
desc 'Delete all likes'
|
|
|
|
task 'posts:delete_all_likes' => :environment do
|
|
|
|
|
|
|
|
post_actions = PostAction.where(post_action_type_id: PostActionType.types[:like])
|
|
|
|
|
|
|
|
likes_deleted = 0
|
|
|
|
total = post_actions.count
|
|
|
|
|
|
|
|
post_actions.each do |post_action|
|
|
|
|
begin
|
|
|
|
post_action.remove_act!(Discourse.system_user)
|
|
|
|
print_status(likes_deleted += 1, total)
|
|
|
|
rescue
|
|
|
|
# skip
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
UserStat.update_all(likes_given: 0, likes_received: 0) # clear user likes stats
|
2017-06-26 13:49:03 +08:00
|
|
|
DirectoryItem.update_all(likes_given: 0, likes_received: 0) # clear user directory likes stats
|
2017-06-23 23:33:44 +08:00
|
|
|
puts "", "#{likes_deleted} likes deleted!", ""
|
|
|
|
end
|
2017-09-15 16:48:24 +08:00
|
|
|
|
|
|
|
desc 'Defer all flags'
|
|
|
|
task 'posts:defer_all_flags' => :environment do
|
|
|
|
|
|
|
|
active_flags = FlagQuery.flagged_post_actions('active')
|
|
|
|
|
|
|
|
flags_deferred = 0
|
|
|
|
total = active_flags.count
|
|
|
|
|
|
|
|
active_flags.each do |post_action|
|
|
|
|
begin
|
|
|
|
PostAction.defer_flags!(Post.find(post_action.post_id), Discourse.system_user)
|
|
|
|
print_status(flags_deferred += 1, total)
|
|
|
|
rescue
|
|
|
|
# skip
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
puts "", "#{flags_deferred} flags deferred!", ""
|
|
|
|
end
|
2017-10-06 20:28:26 +08:00
|
|
|
|
|
|
|
desc 'Refreshes each post that was received via email'
|
|
|
|
task 'posts:refresh_emails', [:topic_id] => [:environment] do |_, args|
|
|
|
|
posts = Post.where.not(raw_email: nil).where(via_email: true)
|
|
|
|
posts = posts.where(topic_id: args[:topic_id]) if args[:topic_id]
|
|
|
|
|
|
|
|
updated = 0
|
|
|
|
total = posts.count
|
|
|
|
|
|
|
|
posts.find_each do |post|
|
2017-12-14 05:01:31 +08:00
|
|
|
begin
|
|
|
|
receiver = Email::Receiver.new(post.raw_email)
|
|
|
|
|
|
|
|
body, elided = receiver.select_body
|
|
|
|
body = receiver.add_attachments(body || '', post.user_id)
|
|
|
|
body << Email::Receiver.elided_html(elided) if elided.present?
|
2017-10-06 20:28:26 +08:00
|
|
|
|
2017-12-14 05:01:31 +08:00
|
|
|
post.revise(Discourse.system_user, { raw: body, cook_method: Post.cook_methods[:regular] },
|
|
|
|
skip_revision: true, skip_validations: true, bypass_bump: true)
|
|
|
|
rescue
|
|
|
|
puts "Failed to refresh post (topic_id: #{post.topic_id}, post_id: #{post.id})"
|
|
|
|
end
|
2017-10-06 20:28:26 +08:00
|
|
|
|
|
|
|
updated += 1
|
|
|
|
|
|
|
|
print_status(updated, total)
|
|
|
|
end
|
|
|
|
|
|
|
|
puts "", "Done. #{updated} posts updated.", ""
|
|
|
|
end
|