FIX: respect automatic group membership when sso changes email

Previously we were only updating group membership when a user record was
first created in an SSO setting.

This corrects it so we also update it if SSO changes the email
This commit is contained in:
Sam Saffron 2020-04-08 16:33:50 +10:00
parent 0375a5ac0b
commit b824898f61
No known key found for this signature in database
GPG Key ID: B9606168D2FFD9F5
3 changed files with 36 additions and 1 deletions

View File

@ -100,6 +100,10 @@ class DiscourseSingleSignOn < SingleSignOn
user.user_avatar.save! if user.user_avatar user.user_avatar.save! if user.user_avatar
user.save! user.save!
if @email_changed && user.active
user.set_automatic_groups
end
# The user might require approval # The user might require approval
user.create_reviewable user.create_reviewable
@ -251,9 +255,12 @@ class DiscourseSingleSignOn < SingleSignOn
end end
def change_external_attributes_and_override(sso_record, user) def change_external_attributes_and_override(sso_record, user)
@email_changed = false
if SiteSetting.sso_overrides_email && user.email != Email.downcase(email) if SiteSetting.sso_overrides_email && user.email != Email.downcase(email)
user.email = email user.email = email
user.active = false if require_activation user.active = false if require_activation
@email_changed = true
end end
if SiteSetting.sso_overrides_username? && username.present? if SiteSetting.sso_overrides_username? && username.present?

View File

@ -584,8 +584,9 @@ RSpec.describe SessionController do
end end
it 'can take over an account' do it 'can take over an account' do
user = Fabricate(:user, email: 'bill@bill.com')
sso = get_sso("/") sso = get_sso("/")
user = Fabricate(:user)
sso.email = user.email sso.email = user.email
sso.external_id = 'abc' sso.external_id = 'abc'
sso.username = 'sam' sso.username = 'sam'
@ -597,6 +598,25 @@ RSpec.describe SessionController do
expect(logged_on_user.email).to eq(user.email) expect(logged_on_user.email).to eq(user.email)
expect(logged_on_user.single_sign_on_record.external_id).to eq("abc") expect(logged_on_user.single_sign_on_record.external_id).to eq("abc")
expect(logged_on_user.single_sign_on_record.external_username).to eq('sam') expect(logged_on_user.single_sign_on_record.external_username).to eq('sam')
# we are updating the email ... ensure auto group membership works
sign_out
SiteSetting.email_editable = false
SiteSetting.sso_overrides_email = true
group = Fabricate(:group, name: :bob, automatic_membership_email_domains: 'jane.com')
sso = get_sso("/")
sso.email = "hello@jane.com"
sso.external_id = 'abc'
get "/session/sso_login", params: Rack::Utils.parse_query(sso.payload), headers: headers
logged_on_user = Discourse.current_user_provider.new(request.env).current_user
expect(logged_on_user.email).to eq('hello@jane.com')
expect(group.users.count).to eq(1)
end end
def sso_for_ip_specs def sso_for_ip_specs
@ -722,6 +742,8 @@ RSpec.describe SessionController do
end end
it 'allows you to create an account' do it 'allows you to create an account' do
group = Fabricate(:group, name: :bob, automatic_membership_email_domains: 'bob.com')
sso = get_sso('/a/') sso = get_sso('/a/')
sso.external_id = '666' # the number of the beast sso.external_id = '666' # the number of the beast
sso.email = 'bob@bob.com' sso.email = 'bob@bob.com'
@ -742,6 +764,8 @@ RSpec.describe SessionController do
logged_on_user = Discourse.current_user_provider.new(request.env).current_user logged_on_user = Discourse.current_user_provider.new(request.env).current_user
expect(group.users.where(id: logged_on_user.id).count).to eq(1)
# ensure nothing is transient # ensure nothing is transient
logged_on_user = User.find(logged_on_user.id) logged_on_user = User.find(logged_on_user.id)

View File

@ -30,6 +30,10 @@ module IntegrationHelpers
user user
end end
def sign_out
delete "/session"
end
def read_secure_session def read_secure_session
SecureSession.new(session[:secure_session_id]) SecureSession.new(session[:secure_session_id])
end end