FIX: Store custom emojis as uploads.

* Depending on a hardcoded directory was a flawed design
  which made it impossible to debug when custom emojis go
  missing.
This commit is contained in:
Guo Xiang Tan 2017-02-02 17:41:57 +08:00
parent 99943ec769
commit 1a7e954e09
17 changed files with 224 additions and 126 deletions

View File

@ -14,25 +14,50 @@ class Admin::EmojisController < Admin::AdminController
.gsub(/_{2,}/, '_') .gsub(/_{2,}/, '_')
.downcase .downcase
data = if Emoji.exists?(name) upload = Upload.create_for(
failed_json.merge(errors: [I18n.t("emoji.errors.name_already_exists", name: name)]) current_user.id,
elsif emoji = Emoji.create_for(file, name) file.tempfile,
emoji file.original_filename,
else File.size(file.tempfile.path),
failed_json.merge(errors: [I18n.t("emoji.errors.error_while_storing_emoji")]) image_type: 'custom_emoji'
end )
data =
if upload.persisted?
custom_emoji = CustomEmoji.new(name: name, upload: upload)
if custom_emoji.save
Emoji.clear_cache
{ name: custom_emoji.name, url: custom_emoji.upload.url }
else
failed_json.merge(errors: custom_emoji.errors.full_messages)
end
else
failed_json.merge(errors: upload.errors.full_messages)
end
MessageBus.publish("/uploads/emoji", data.as_json, user_ids: [current_user.id]) MessageBus.publish("/uploads/emoji", data.as_json, user_ids: [current_user.id])
end end
render json: success_json render json: success_json
end end
def destroy def destroy
name = params.require(:id) name = params.require(:id)
Emoji[name].try(:remove)
render nothing: true custom_emoji = CustomEmoji.find_by(name: name)
raise Discourse::InvalidParameters unless custom_emoji
CustomEmoji.transaction do
custom_emoji.upload.destroy!
custom_emoji.destroy!
end
Emoji.clear_cache
Jobs.enqueue(:rebake_custom_emoji_posts, name: name)
render json: success_json
end end
end end

View File

@ -0,0 +1,35 @@
module Jobs
class MigrateCustomEmojis < Jobs::Onceoff
def execute_onceoff(args)
return if Rails.env.test?
CustomEmoji.transaction do
Dir["#{Rails.root}/#{Emoji.base_directory}/*.{png,gif}"].each do |path|
name = File.basename(path, File.extname(path))
File.open(path) do |file|
upload = Upload.create_for(
Discourse.system_user.id,
file,
File.basename(path),
file.size,
image_type: 'custom_emoji'
)
if upload.persisted?
CustomEmoji.create!(name: name, upload: upload)
else
raise "Failed to create upload for '#{name}' custom emoji"
end
end
end
Emoji.clear_cache
Post.where("cooked LIKE '%#{Emoji.base_url}%'").find_each do |post|
post.rebake!
end
end
end
end
end

View File

@ -0,0 +1,8 @@
module Jobs
class RebakeCustomEmojiPosts < Jobs::Base
def execute(args)
name = args[:name]
Post.where("raw LIKE '%:#{name}:%'").find_each { |post| post.rebake! }
end
end
end

View File

@ -1,18 +0,0 @@
module Jobs
class ResizeEmoji < Jobs::Base
def execute(args)
path = args[:path]
return unless File.exists?(path)
opts = {
allow_animation: true,
force_aspect_ratio: SiteSetting.enforce_square_emoji
}
# make sure emoji aren't too big
OptimizedImage.downsize(path, path, "100x100", opts)
end
end
end

View File

@ -22,11 +22,13 @@ module Jobs
.joins("LEFT JOIN user_avatars ua ON (ua.gravatar_upload_id = uploads.id OR ua.custom_upload_id = uploads.id)") .joins("LEFT JOIN user_avatars ua ON (ua.gravatar_upload_id = uploads.id OR ua.custom_upload_id = uploads.id)")
.joins("LEFT JOIN user_profiles up ON up.profile_background = uploads.url OR up.card_background = uploads.url") .joins("LEFT JOIN user_profiles up ON up.profile_background = uploads.url OR up.card_background = uploads.url")
.joins("LEFT JOIN categories c ON c.uploaded_logo_id = uploads.id OR c.uploaded_background_id = uploads.id") .joins("LEFT JOIN categories c ON c.uploaded_logo_id = uploads.id OR c.uploaded_background_id = uploads.id")
.joins("LEFT JOIN custom_emojis ce ON ce.upload_id = uploads.id")
.where("pu.upload_id IS NULL") .where("pu.upload_id IS NULL")
.where("u.uploaded_avatar_id IS NULL") .where("u.uploaded_avatar_id IS NULL")
.where("ua.gravatar_upload_id IS NULL AND ua.custom_upload_id IS NULL") .where("ua.gravatar_upload_id IS NULL AND ua.custom_upload_id IS NULL")
.where("up.profile_background IS NULL AND up.card_background IS NULL") .where("up.profile_background IS NULL AND up.card_background IS NULL")
.where("c.uploaded_logo_id IS NULL AND c.uploaded_background_id IS NULL") .where("c.uploaded_logo_id IS NULL AND c.uploaded_background_id IS NULL")
.where("ce.upload_id IS NULL")
.where("uploads.url NOT IN (?)", ignore_urls) .where("uploads.url NOT IN (?)", ignore_urls)
result.find_each do |upload| result.find_each do |upload|

View File

@ -0,0 +1,6 @@
class CustomEmoji < ActiveRecord::Base
belongs_to :upload
validates :name, presence: true, uniqueness: true
validates :upload_id, presence: true
end

View File

@ -14,14 +14,6 @@ class Emoji
@path = path @path = path
end end
def remove
return if path.blank?
if File.exists?(path)
File.delete(path) rescue nil
Emoji.clear_cache
end
end
def self.all def self.all
Discourse.cache.fetch(cache_key("all_emojis")) { standard | custom } Discourse.cache.fetch(cache_key("all_emojis")) { standard | custom }
end end
@ -46,14 +38,6 @@ class Emoji
Emoji.custom.detect { |e| e.name == name } Emoji.custom.detect { |e| e.name == name }
end end
def self.create_from_path(path)
extension = File.extname(path)
Emoji.new(path).tap do |e|
e.name = File.basename(path, ".*")
e.url = "#{base_url}/#{e.name}#{extension}"
end
end
def self.create_from_db_item(emoji) def self.create_from_db_item(emoji)
name = emoji["name"] name = emoji["name"]
filename = "#{emoji['filename'] || name}.png" filename = "#{emoji['filename'] || name}.png"
@ -63,22 +47,6 @@ class Emoji
end end
end end
def self.create_for(file, name)
extension = File.extname(file.original_filename)
path = "#{Emoji.base_directory}/#{name}#{extension}"
full_path = "#{Rails.root}/#{path}"
# store the emoji
FileUtils.mkdir_p(Pathname.new(path).dirname)
File.open(path, "wb") { |f| f << file.tempfile.read }
# clear the cache
Emoji.clear_cache
# launch resize job
Jobs.enqueue(:resize_emoji, path: full_path)
# return created emoji
Emoji[name]
end
def self.cache_key(name) def self.cache_key(name)
"#{name}:#{EMOJI_VERSION}:#{Plugin::CustomEmoji.cache_key}" "#{name}:#{EMOJI_VERSION}:#{Plugin::CustomEmoji.cache_key}"
end end
@ -124,9 +92,12 @@ class Emoji
def self.load_custom def self.load_custom
result = [] result = []
Dir.glob(File.join(Emoji.base_directory, "*.{png,gif}")) CustomEmoji.all.each do |emoji|
.sort result << Emoji.new.tap do |e|
.each { |emoji| result << Emoji.create_from_path(emoji) } e.name = emoji.name
e.url = emoji.upload.url
end
end
Plugin::CustomEmoji.emojis.each do |name, url| Plugin::CustomEmoji.emojis.each do |name, url|
result << Emoji.new.tap do |e| result << Emoji.new.tap do |e|

View File

@ -57,7 +57,12 @@ class Upload < ActiveRecord::Base
end end
# list of image types that will be cropped # list of image types that will be cropped
CROPPED_IMAGE_TYPES ||= %w{avatar profile_background card_background} CROPPED_IMAGE_TYPES ||= %w{
avatar
profile_background
card_background
custom_emoji
}
WHITELISTED_SVG_ELEMENTS ||= %w{ WHITELISTED_SVG_ELEMENTS ||= %w{
circle circle
@ -92,7 +97,7 @@ class Upload < ActiveRecord::Base
# options # options
# - content_type # - content_type
# - origin (url) # - origin (url)
# - image_type ("avatar", "profile_background", "card_background") # - image_type ("avatar", "profile_background", "card_background", "custom_emoji")
# - is_attachment_for_group_message (boolean) # - is_attachment_for_group_message (boolean)
def self.create_for(user_id, file, filename, filesize, options = {}) def self.create_for(user_id, file, filename, filesize, options = {})
upload = Upload.new upload = Upload.new
@ -145,6 +150,8 @@ class Upload < ActiveRecord::Base
max_width = 590 * max_pixel_ratio max_width = 590 * max_pixel_ratio
width, height = ImageSizer.resize(w, h, max_width: max_width, max_height: max_width) width, height = ImageSizer.resize(w, h, max_width: max_width, max_height: max_width)
OptimizedImage.downsize(file.path, file.path, "#{width}x#{height}", filename: filename, allow_animation: allow_animation) OptimizedImage.downsize(file.path, file.path, "#{width}x#{height}", filename: filename, allow_animation: allow_animation)
when "custom_emoji"
OptimizedImage.downsize(file.path, file.path, "100x100", filename: filename, allow_animation: allow_animation)
end end
end end

View File

@ -12,4 +12,3 @@ end
ActiveSupport.on_load(:active_record) do ActiveSupport.on_load(:active_record) do
self.include_root_in_json = false self.include_root_in_json = false
end end

View File

@ -391,6 +391,10 @@ en:
attributes: attributes:
payload_url: payload_url:
invalid: "URL is invalid. URL should includes http:// or https://. And no blank is allowed." invalid: "URL is invalid. URL should includes http:// or https://. And no blank is allowed."
custom_emoji:
attributes:
name:
taken: is already in use by another emoji
user_profile: user_profile:
no_info_me: "<div class='missing-profile'>the About Me field of your profile is currently blank, <a href='/users/%{username_lower}/preferences/about-me'>would you like to fill it out?</a></div>" no_info_me: "<div class='missing-profile'>the About Me field of your profile is currently blank, <a href='/users/%{username_lower}/preferences/about-me'>would you like to fill it out?</a></div>"
@ -1574,11 +1578,6 @@ en:
post_revision_text: "Ownership transferred from %{old_user} to %{new_user}" post_revision_text: "Ownership transferred from %{old_user} to %{new_user}"
deleted_user: "a deleted user" deleted_user: "a deleted user"
emoji:
errors:
name_already_exists: "Sorry, the name '%{name}' is already used by another emoji."
error_while_storing_emoji: "Sorry, there has been an error while storing the emoji."
topic_statuses: topic_statuses:
archived_enabled: "This topic is now archived. It is frozen and cannot be changed in any way." archived_enabled: "This topic is now archived. It is frozen and cannot be changed in any way."
archived_disabled: "This topic is now unarchived. It is no longer frozen, and can be changed." archived_disabled: "This topic is now unarchived. It is no longer frozen, and can be changed."

View File

@ -0,0 +1,12 @@
class CreateCustomEmojis < ActiveRecord::Migration
def change
create_table :custom_emojis do |t|
t.string :name, null: false
t.integer :upload_id, null: false
t.timestamps null: false
end
add_index :custom_emojis, :name, unique: true
end
end

View File

@ -6,6 +6,7 @@ end
# we need to run seed_fu every time we run rake db:migrate # we need to run seed_fu every time we run rake db:migrate
task 'db:migrate' => ['environment', 'set_locale'] do task 'db:migrate' => ['environment', 'set_locale'] do
SeedFu.seed SeedFu.seed
Jobs::Onceoff.enqueue_all
end end
task 'test:prepare' => 'environment' do task 'test:prepare' => 'environment' do

View File

@ -431,7 +431,7 @@ HTML
describe "custom emoji" do describe "custom emoji" do
it "replaces the custom emoji" do it "replaces the custom emoji" do
Emoji.stubs(:custom).returns([ Emoji.create_from_path('trout') ]) CustomEmoji.create!(name: 'trout', upload: Fabricate(:upload))
expect(PrettyText.cook("hello :trout:")).to match(/<img src[^>]+trout[^>]+>/) expect(PrettyText.cook("hello :trout:")).to match(/<img src[^>]+trout[^>]+>/)
end end
end end

View File

@ -16,10 +16,6 @@ describe Admin::EmojisController do
end end
end end
it "is a subclass of AdminController" do
expect(Admin::EmojisController < Admin::AdminController).to eq(true)
end
context "when logged in" do context "when logged in" do
let!(:user) { log_in(:admin) } let!(:user) { log_in(:admin) }
@ -33,56 +29,6 @@ describe Admin::EmojisController do
expect(json[0]["url"]).to eq(custom_emoji.url) expect(json[0]["url"]).to eq(custom_emoji.url)
end end
end end
context ".create" do
before { Emoji.expects(:custom).returns([custom_emoji]) }
context "name already exist" do
it "throws an error" do
message = MessageBus.track_publish do
xhr :post, :create, { name: "hello", file: "" }
end.first
expect(response).to be_success
expect(message.data["errors"]).to be
end
end
context "error while saving emoji" do
it "throws an error" do
Emoji.expects(:create_for).returns(nil)
message = MessageBus.track_publish do
xhr :post, :create, { name: "garbage", file: "" }
end.first
expect(response).to be_success
expect(message.data["errors"]).to be
end
end
it "works" do
Emoji.expects(:create_for).returns(custom_emoji2)
message = MessageBus.track_publish do
xhr :post, :create, { name: "hello2", file: ""}
end.first
expect(response).to be_success
expect(message.data["name"]).to eq(custom_emoji2.name)
expect(message.data["url"]).to eq(custom_emoji2.url)
end
end
context ".destroy" do
it "deletes the custom emoji" do
custom_emoji.expects(:remove)
Emoji.expects(:custom).returns([custom_emoji])
xhr :delete, :destroy, id: "hello"
expect(response).to be_success
end
end
end end
end end

View File

@ -0,0 +1,76 @@
require 'rails_helper'
RSpec.describe "Managing custom emojis" do
let(:admin) { Fabricate(:admin) }
let(:upload) { Fabricate(:upload) }
before do
sign_in(admin)
end
describe "creating a custom emoji" do
describe 'when upload is invalid' do
it 'should publish the right error' do
message = MessageBus.track_publish do
post("/admin/customize/emojis.json", {
name: 'test',
file: fixture_file_upload("#{Rails.root}/spec/fixtures/images/fake.jpg")
})
end.first
expect(message.channel).to eq("/uploads/emoji")
expect(message.data["errors"]).to eq([I18n.t('upload.images.size_not_found')])
end
end
describe 'when emoji name already exists' do
it 'should publish the right error' do
CustomEmoji.create!(name: 'test', upload: upload)
message = MessageBus.track_publish do
post("/admin/customize/emojis.json", {
name: 'test',
file: fixture_file_upload("#{Rails.root}/spec/fixtures/images/logo.png")
})
end.first
expect(message.channel).to eq("/uploads/emoji")
expect(message.data["errors"]).to eq([
"Name #{I18n.t('activerecord.errors.models.custom_emoji.attributes.name.taken')}"
])
end
end
it 'should allow an admin to add a custom emoji' do
Emoji.expects(:clear_cache)
message = MessageBus.track_publish do
post("/admin/customize/emojis.json", {
name: 'test',
file: fixture_file_upload("#{Rails.root}/spec/fixtures/images/logo.png")
})
end.first
custom_emoji = CustomEmoji.last
upload = custom_emoji.upload
expect(upload.original_filename).to eq('logo.png')
expect(message.channel).to eq("/uploads/emoji")
expect(message.data["errors"]).to eq(nil)
expect(message.data["name"]).to eq(custom_emoji.name)
expect(message.data["url"]).to eq(upload.url)
end
end
describe 'deleting a custom emoji' do
it 'should allow an admin to delete a custom emoji' do
custom_emoji = CustomEmoji.create!(name: 'test', upload: upload)
Emoji.clear_cache
expect { delete "/admin/customize/emojis/#{custom_emoji.name}.json", name: 'test' }
.to change { Upload.count }.by(-1)
.and change { CustomEmoji.count }.by(-1)
end
end
end

View File

@ -140,4 +140,14 @@ describe Jobs::CleanUpUploads do
expect(Upload.find_by(id: upload.id)).to eq(upload) expect(Upload.find_by(id: upload.id)).to eq(upload)
end end
it "does not delete custom emojis" do
upload = fabricate_upload
CustomEmoji.create!(name: 'test', upload: upload)
Jobs::CleanUpUploads.new.execute(nil)
expect(Upload.find_by(id: @upload.id)).to eq(nil)
expect(Upload.find_by(id: upload.id)).to eq(upload)
end
end end

View File

@ -0,0 +1,19 @@
require 'rails_helper'
RSpec.describe Jobs::RebakeCustomEmojiPosts do
it 'should rebake posts that are using a given custom emoji' do
custom_emoji = CustomEmoji.create!(name: 'test', upload: Fabricate(:upload))
Emoji.clear_cache
post = Fabricate(:post, raw: 'some post with :test: yay')
expect(post.reload.cooked).to eq(
"<p>some post with <img src=\"/uploads/default/0/1234567890123456.png?v=3\" title=\":test:\" class=\"emoji emoji-custom\" alt=\":test:\"> yay</p>"
)
custom_emoji.destroy!
Emoji.clear_cache
described_class.new.execute(name: 'test')
expect(post.reload.cooked).to eq('<p>some post with :test: yay</p>')
end
end