FIX: `UserAvatar#update_gravatar!` does not update `User#uploaded_avatar`.
https://meta.discourse.org/t/missing-user-profile-pictures/93844/4
This commit is contained in:
parent
18b396ad56
commit
4e11811321
|
@ -0,0 +1,23 @@
|
||||||
|
module Jobs
|
||||||
|
class FixOutOfSyncUserUploadedAvatar < Jobs::Onceoff
|
||||||
|
def execute_onceoff(args)
|
||||||
|
DB.exec(<<~SQL)
|
||||||
|
WITH X AS (
|
||||||
|
SELECT
|
||||||
|
u.id AS user_id,
|
||||||
|
ua.gravatar_upload_id AS gravatar_upload_id
|
||||||
|
FROM users u
|
||||||
|
JOIN user_avatars ua ON ua.user_id = u.id
|
||||||
|
LEFT JOIN uploads ON uploads.id = u.uploaded_avatar_id
|
||||||
|
WHERE u.uploaded_avatar_id IS NOT NULL
|
||||||
|
AND uploads.id IS NULL
|
||||||
|
)
|
||||||
|
UPDATE users
|
||||||
|
SET uploaded_avatar_id = X.gravatar_upload_id
|
||||||
|
FROM X
|
||||||
|
WHERE users.id = X.user_id
|
||||||
|
AND coalesce(uploaded_avatar_id,-1) <> coalesce(X.gravatar_upload_id,-1)
|
||||||
|
SQL
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
|
@ -42,10 +42,18 @@ class UserAvatar < ActiveRecord::Base
|
||||||
type: "avatar"
|
type: "avatar"
|
||||||
).create_for(user_id)
|
).create_for(user_id)
|
||||||
|
|
||||||
if gravatar_upload_id != upload.id
|
upload_id = upload.id
|
||||||
gravatar_upload&.destroy!
|
|
||||||
self.gravatar_upload = upload
|
if gravatar_upload_id != upload_id
|
||||||
save!
|
User.transaction do
|
||||||
|
if gravatar_upload_id && user.uploaded_avatar_id == gravatar_upload_id
|
||||||
|
user.update!(uploaded_avatar_id: upload_id)
|
||||||
|
end
|
||||||
|
|
||||||
|
gravatar_upload&.destroy!
|
||||||
|
self.gravatar_upload = upload
|
||||||
|
save!
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
rescue OpenURI::HTTPError
|
rescue OpenURI::HTTPError
|
||||||
|
|
|
@ -0,0 +1,42 @@
|
||||||
|
require 'rails_helper'
|
||||||
|
|
||||||
|
RSpec.describe Jobs::FixOutOfSyncUserUploadedAvatar do
|
||||||
|
it 'should fix out of sync user uploaded avatars' do
|
||||||
|
user_with_custom_upload = Fabricate(:user)
|
||||||
|
custom_upload1 = Fabricate(:upload, user: user_with_custom_upload)
|
||||||
|
gravatar_upload1 = Fabricate(:upload, user: user_with_custom_upload)
|
||||||
|
user_with_custom_upload.update!(uploaded_avatar: custom_upload1)
|
||||||
|
|
||||||
|
user_with_custom_upload.user_avatar.update!(
|
||||||
|
custom_upload: custom_upload1,
|
||||||
|
gravatar_upload: gravatar_upload1
|
||||||
|
)
|
||||||
|
|
||||||
|
user_out_of_sync = Fabricate(:user)
|
||||||
|
custom_upload2 = Fabricate(:upload, user: user_out_of_sync)
|
||||||
|
gravatar_upload2 = Fabricate(:upload, user: user_out_of_sync)
|
||||||
|
prev_gravatar_upload = Fabricate(:upload, user: user_out_of_sync)
|
||||||
|
user_out_of_sync.update!(uploaded_avatar: prev_gravatar_upload)
|
||||||
|
prev_gravatar_upload.destroy!
|
||||||
|
|
||||||
|
user_out_of_sync.user_avatar.update!(
|
||||||
|
custom_upload: custom_upload2,
|
||||||
|
gravatar_upload: gravatar_upload2
|
||||||
|
)
|
||||||
|
|
||||||
|
user_without_uploaded_avatar = Fabricate(:user)
|
||||||
|
gravatar_upload3 = Fabricate(:upload, user: user_without_uploaded_avatar)
|
||||||
|
|
||||||
|
user_without_uploaded_avatar.user_avatar.update!(
|
||||||
|
gravatar_upload: gravatar_upload3
|
||||||
|
)
|
||||||
|
|
||||||
|
described_class.new.execute_onceoff({})
|
||||||
|
|
||||||
|
expect(user_with_custom_upload.reload.uploaded_avatar).to eq(custom_upload1)
|
||||||
|
expect(user_out_of_sync.reload.uploaded_avatar).to eq(gravatar_upload2)
|
||||||
|
|
||||||
|
expect(user_without_uploaded_avatar.reload.uploaded_avatar)
|
||||||
|
.to eq(nil)
|
||||||
|
end
|
||||||
|
end
|
|
@ -4,16 +4,66 @@ describe UserAvatar do
|
||||||
let(:user) { Fabricate(:user) }
|
let(:user) { Fabricate(:user) }
|
||||||
let(:avatar) { user.create_user_avatar! }
|
let(:avatar) { user.create_user_avatar! }
|
||||||
|
|
||||||
it 'can update gravatars' do
|
describe '#update_gravatar!' do
|
||||||
temp = Tempfile.new('test')
|
let(:temp) { Tempfile.new('test') }
|
||||||
temp.binmode
|
let(:upload) { Fabricate(:upload, user: user) }
|
||||||
# tiny valid png
|
|
||||||
temp.write(Base64.decode64("R0lGODlhAQABALMAAAAAAIAAAACAAICAAAAAgIAAgACAgMDAwICAgP8AAAD/AP//AAAA//8A/wD//wBiZCH5BAEAAA8ALAAAAAABAAEAAAQC8EUAOw=="))
|
before do
|
||||||
temp.rewind
|
temp.binmode
|
||||||
FileHelper.expects(:download).returns(temp)
|
# tiny valid png
|
||||||
avatar.update_gravatar!
|
temp.write(Base64.decode64("R0lGODlhAQABALMAAAAAAIAAAACAAICAAAAAgIAAgACAgMDAwICAgP8AAAD/AP//AAAA//8A/wD//wBiZCH5BAEAAA8ALAAAAAABAAEAAAQC8EUAOw=="))
|
||||||
temp.unlink
|
temp.rewind
|
||||||
expect(avatar.gravatar_upload).not_to eq(nil)
|
FileHelper.expects(:download).returns(temp)
|
||||||
|
end
|
||||||
|
|
||||||
|
after do
|
||||||
|
temp.unlink
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'can update gravatars' do
|
||||||
|
expect do
|
||||||
|
avatar.update_gravatar!
|
||||||
|
end.to change { Upload.count }.by(1)
|
||||||
|
|
||||||
|
upload = Upload.last
|
||||||
|
|
||||||
|
expect(avatar.gravatar_upload).to eq(upload)
|
||||||
|
expect(user.reload.uploaded_avatar).to eq(nil)
|
||||||
|
end
|
||||||
|
|
||||||
|
describe 'when user has an existing custom upload' do
|
||||||
|
it "should not change the user's uploaded avatar" do
|
||||||
|
user.update!(uploaded_avatar: upload)
|
||||||
|
|
||||||
|
avatar.update!(
|
||||||
|
custom_upload: upload,
|
||||||
|
gravatar_upload: Fabricate(:upload, user: user)
|
||||||
|
)
|
||||||
|
|
||||||
|
avatar.update_gravatar!
|
||||||
|
|
||||||
|
expect(upload.reload).to eq(upload)
|
||||||
|
expect(user.reload.uploaded_avatar).to eq(upload)
|
||||||
|
expect(avatar.reload.custom_upload).to eq(upload)
|
||||||
|
expect(avatar.gravatar_upload).to eq(Upload.last)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
describe 'when user has an existing gravatar' do
|
||||||
|
it "should update the user's uploaded avatar correctly" do
|
||||||
|
user.update!(uploaded_avatar: upload)
|
||||||
|
avatar.update!(gravatar_upload: upload)
|
||||||
|
|
||||||
|
avatar.update_gravatar!
|
||||||
|
|
||||||
|
expect(Upload.find_by(id: upload.id)).to eq(nil)
|
||||||
|
|
||||||
|
new_upload = Upload.last
|
||||||
|
|
||||||
|
expect(user.reload.uploaded_avatar).to eq(new_upload)
|
||||||
|
expect(avatar.reload.gravatar_upload).to eq(new_upload)
|
||||||
|
end
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
context '.import_url_for_user' do
|
context '.import_url_for_user' do
|
||||||
|
|
Loading…
Reference in New Issue