Refactor Users#upload_avatar method
Moved avatar file upload to ```AvatarUploadService``` class and ```AvatarUploadPolicy``` Address review comments + require missing file in spec
This commit is contained in:
parent
20f06f3efc
commit
58f78e9001
|
@ -1,6 +1,7 @@
|
||||||
require_dependency 'discourse_hub'
|
require_dependency 'discourse_hub'
|
||||||
require_dependency 'user_name_suggester'
|
require_dependency 'user_name_suggester'
|
||||||
require_dependency 'user_activator'
|
require_dependency 'user_activator'
|
||||||
|
require_dependency 'avatar_upload_service'
|
||||||
|
|
||||||
class UsersController < ApplicationController
|
class UsersController < ApplicationController
|
||||||
|
|
||||||
|
@ -168,14 +169,16 @@ class UsersController < ApplicationController
|
||||||
end
|
end
|
||||||
|
|
||||||
def logon_after_password_reset
|
def logon_after_password_reset
|
||||||
if Guardian.new(@user).can_access_forum?
|
message = if Guardian.new(@user).can_access_forum?
|
||||||
# Log in the user
|
# Log in the user
|
||||||
log_on_user(@user)
|
log_on_user(@user)
|
||||||
flash[:success] = I18n.t('password_reset.success')
|
'password_reset.success'
|
||||||
else
|
else
|
||||||
@requires_approval = true
|
@requires_approval = true
|
||||||
flash[:success] = I18n.t('password_reset.success_unapproved')
|
'password_reset.success_unapproved'
|
||||||
end
|
end
|
||||||
|
|
||||||
|
flash[:success] = I18n.t(message)
|
||||||
end
|
end
|
||||||
|
|
||||||
def change_email
|
def change_email
|
||||||
|
@ -285,35 +288,18 @@ class UsersController < ApplicationController
|
||||||
# Only allow url uploading for API users
|
# Only allow url uploading for API users
|
||||||
# TODO: Does not protect from huge uploads
|
# TODO: Does not protect from huge uploads
|
||||||
# https://github.com/discourse/discourse/pull/1512
|
# https://github.com/discourse/discourse/pull/1512
|
||||||
if file.is_a?(String) && is_api?
|
|
||||||
adapted = ::UriAdapter.new(file)
|
|
||||||
file = adapted.build_uploaded_file
|
|
||||||
filesize = adapted.file_size
|
|
||||||
elsif file.is_a?(String)
|
|
||||||
return render status: 422, text: I18n.t("upload.images.unknown_image_type")
|
|
||||||
end
|
|
||||||
|
|
||||||
# check the file size (note: this might also be done in the web server)
|
# check the file size (note: this might also be done in the web server)
|
||||||
filesize ||= File.size(file.tempfile)
|
avatar = build_avatar_from(file)
|
||||||
max_size_kb = SiteSetting.max_image_size_kb * 1024
|
avatar_policy = AvatarUploadPolicy.new(avatar)
|
||||||
if filesize > max_size_kb
|
|
||||||
return render status: 413, text: I18n.t("upload.images.too_large", max_size_kb: max_size_kb)
|
if avatar_policy.too_big?
|
||||||
else
|
return render status: 413, text: I18n.t("upload.images.too_large",
|
||||||
filesize
|
max_size_kb: avatar_policy.max_size_kb)
|
||||||
end
|
end
|
||||||
|
|
||||||
return render status: 422, text: I18n.t("upload.images.unknown_image_type") unless SiteSetting.authorized_image?(file)
|
raise FastImage::UnknownImageType unless SiteSetting.authorized_image?(avatar.file)
|
||||||
|
|
||||||
upload = Upload.create_for(user.id, file, filesize)
|
upload_avatar_for(user, avatar)
|
||||||
user.upload_avatar(upload)
|
|
||||||
|
|
||||||
Jobs.enqueue(:generate_avatars, user_id: user.id, upload_id: upload.id)
|
|
||||||
|
|
||||||
render json: {
|
|
||||||
url: upload.url,
|
|
||||||
width: upload.width,
|
|
||||||
height: upload.height,
|
|
||||||
}
|
|
||||||
|
|
||||||
rescue Discourse::InvalidParameters
|
rescue Discourse::InvalidParameters
|
||||||
render status: 422, text: I18n.t("upload.images.unknown_image_type")
|
render status: 422, text: I18n.t("upload.images.unknown_image_type")
|
||||||
|
@ -413,4 +399,21 @@ class UsersController < ApplicationController
|
||||||
auth
|
auth
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def build_avatar_from(file)
|
||||||
|
source = if file.is_a?(String)
|
||||||
|
is_api? ? :url : (raise FastImage::UnknownImageType)
|
||||||
|
else
|
||||||
|
:image
|
||||||
|
end
|
||||||
|
AvatarUploadService.new(file, source)
|
||||||
|
end
|
||||||
|
|
||||||
|
def upload_avatar_for(user, avatar)
|
||||||
|
upload = Upload.create_for(user.id, avatar.file, avatar.filesize)
|
||||||
|
user.upload_avatar(upload)
|
||||||
|
|
||||||
|
Jobs.enqueue(:generate_avatars, user_id: user.id, upload_id: upload.id)
|
||||||
|
render json: { url: upload.url, width: upload.width, height: upload.height }
|
||||||
|
end
|
||||||
|
|
||||||
end
|
end
|
||||||
|
|
|
@ -0,0 +1,43 @@
|
||||||
|
class AvatarUploadService
|
||||||
|
|
||||||
|
attr_accessor :source
|
||||||
|
attr_reader :filesize, :file
|
||||||
|
|
||||||
|
def initialize(file, source)
|
||||||
|
@source = source
|
||||||
|
@file , @filesize = construct(file)
|
||||||
|
end
|
||||||
|
|
||||||
|
def construct(file)
|
||||||
|
case source
|
||||||
|
when :url
|
||||||
|
build_from_url(file)
|
||||||
|
when :image
|
||||||
|
[file, File.size(file.tempfile)]
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def build_from_url(url)
|
||||||
|
temp = ::UriAdapter.new(url)
|
||||||
|
return temp.build_uploaded_file, temp.file_size
|
||||||
|
end
|
||||||
|
|
||||||
|
end
|
||||||
|
|
||||||
|
class AvatarUploadPolicy
|
||||||
|
|
||||||
|
def initialize(avatar)
|
||||||
|
@avatar = avatar
|
||||||
|
end
|
||||||
|
|
||||||
|
def max_size_kb
|
||||||
|
SiteSetting.max_image_size_kb * 1024
|
||||||
|
end
|
||||||
|
|
||||||
|
def too_big?
|
||||||
|
@avatar.filesize > max_size_kb
|
||||||
|
end
|
||||||
|
|
||||||
|
end
|
|
@ -0,0 +1,66 @@
|
||||||
|
require "spec_helper"
|
||||||
|
require "avatar_upload_service"
|
||||||
|
|
||||||
|
describe AvatarUploadService do
|
||||||
|
let(:file) do
|
||||||
|
ActionDispatch::Http::UploadedFile.new({
|
||||||
|
filename: 'logo.png',
|
||||||
|
tempfile: File.new("#{Rails.root}/spec/fixtures/images/logo.png")
|
||||||
|
})
|
||||||
|
end
|
||||||
|
|
||||||
|
let(:url) { "http://cdn.discourse.org/assets/logo.png" }
|
||||||
|
|
||||||
|
describe "#construct" do
|
||||||
|
context "when avatar is in the form of a file upload" do
|
||||||
|
let(:avatar_file) { AvatarUploadService.new(file, :image) }
|
||||||
|
|
||||||
|
it "should have a filesize" do
|
||||||
|
expect(avatar_file.filesize).to eq(2290)
|
||||||
|
end
|
||||||
|
|
||||||
|
it "should have a source as 'image'" do
|
||||||
|
expect(avatar_file.source).to eq(:image)
|
||||||
|
end
|
||||||
|
|
||||||
|
it "is an instance of File class" do
|
||||||
|
file = avatar_file.file
|
||||||
|
expect(file.tempfile).to be_instance_of File
|
||||||
|
end
|
||||||
|
|
||||||
|
it "returns the file object built from File" do
|
||||||
|
file = avatar_file.file
|
||||||
|
file.should be_instance_of(ActionDispatch::Http::UploadedFile)
|
||||||
|
file.original_filename.should == "logo.png"
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
context "when file is in the form of a URL" do
|
||||||
|
let(:avatar_file) { AvatarUploadService.new(url, :url) }
|
||||||
|
|
||||||
|
before :each do
|
||||||
|
UriAdapter.any_instance.stubs(:open).returns StringIO.new(fixture_file("images/logo.png"))
|
||||||
|
end
|
||||||
|
|
||||||
|
it "should have a filesize" do
|
||||||
|
expect(avatar_file.filesize).to eq(2290)
|
||||||
|
end
|
||||||
|
|
||||||
|
it "should have a source as 'url'" do
|
||||||
|
expect(avatar_file.source).to eq(:url)
|
||||||
|
end
|
||||||
|
|
||||||
|
it "is an instance of Tempfile class" do
|
||||||
|
file = avatar_file.file
|
||||||
|
expect(file.tempfile).to be_instance_of Tempfile
|
||||||
|
end
|
||||||
|
|
||||||
|
it "returns the file object built from URL" do
|
||||||
|
file = avatar_file.file
|
||||||
|
file.should be_instance_of(ActionDispatch::Http::UploadedFile)
|
||||||
|
file.original_filename.should == "logo.png"
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
end
|
|
@ -996,7 +996,7 @@ describe UsersController do
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'rejects large images' do
|
it 'rejects large images' do
|
||||||
SiteSetting.stubs(:max_image_size_kb).returns(1)
|
AvatarUploadPolicy.any_instance.stubs(:too_big?).returns(true)
|
||||||
xhr :post, :upload_avatar, username: user.username, file: avatar
|
xhr :post, :upload_avatar, username: user.username, file: avatar
|
||||||
response.status.should eq 413
|
response.status.should eq 413
|
||||||
end
|
end
|
||||||
|
@ -1041,7 +1041,7 @@ describe UsersController do
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'rejects large images' do
|
it 'rejects large images' do
|
||||||
SiteSetting.stubs(:max_image_size_kb).returns(1)
|
AvatarUploadPolicy.any_instance.stubs(:too_big?).returns(true)
|
||||||
xhr :post, :upload_avatar, username: user.username, file: avatar_url
|
xhr :post, :upload_avatar, username: user.username, file: avatar_url
|
||||||
response.status.should eq 413
|
response.status.should eq 413
|
||||||
end
|
end
|
||||||
|
|
Loading…
Reference in New Issue