FIX: log proper error message when SSO nonce verification fails (#14077)

This commit is contained in:
Arpit Jalan 2021-08-18 18:44:12 +05:30 committed by GitHub
parent 4380ba34d5
commit 7db3888f17
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
4 changed files with 41 additions and 1 deletions

View File

@ -156,7 +156,7 @@ class SessionController < ApplicationController
if !sso.nonce_valid? if !sso.nonce_valid?
if SiteSetting.verbose_discourse_connect_logging if SiteSetting.verbose_discourse_connect_logging
Rails.logger.warn("Verbose SSO log: Nonce has already expired\n\n#{sso.diagnostics}") Rails.logger.warn("Verbose SSO log: #{sso.nonce_error}\n\n#{sso.diagnostics}")
end end
return render_sso_error(text: I18n.t("discourse_connect.timeout_expired"), status: 419) return render_sso_error(text: I18n.t("discourse_connect.timeout_expired"), status: 419)
end end

View File

@ -47,6 +47,14 @@ class DiscourseSingleSignOn < SingleSignOn
end end
end end
def nonce_error
if Discourse.cache.read(used_nonce_key).present?
"Nonce has already been used"
else
"Nonce has expired"
end
end
def return_path def return_path
if SiteSetting.discourse_connect_csrf_protection if SiteSetting.discourse_connect_csrf_protection
@secure_session[nonce_key] || "/" @secure_session[nonce_key] || "/"
@ -62,6 +70,8 @@ class DiscourseSingleSignOn < SingleSignOn
else else
Discourse.cache.delete nonce_key Discourse.cache.delete nonce_key
end end
Discourse.cache.write(used_nonce_key, return_path, expires_in: SingleSignOn.used_nonce_expiry_time)
end end
end end
@ -69,6 +79,10 @@ class DiscourseSingleSignOn < SingleSignOn
"SSO_NONCE_#{nonce}" "SSO_NONCE_#{nonce}"
end end
def used_nonce_key
"USED_SSO_NONCE_#{nonce}"
end
BANNED_EXTERNAL_IDS = %w{none nil blank null} BANNED_EXTERNAL_IDS = %w{none nil blank null}
def lookup_or_create_user(ip_address = nil) def lookup_or_create_user(ip_address = nil)

View File

@ -50,6 +50,10 @@ class SingleSignOn
@nonce_expiry_time = v @nonce_expiry_time = v
end end
def self.used_nonce_expiry_time
24.hours
end
attr_accessor(*ACCESSORS) attr_accessor(*ACCESSORS)
attr_writer :sso_secret, :sso_url attr_writer :sso_secret, :sso_url

View File

@ -406,6 +406,28 @@ describe DiscourseSingleSignOn do
expect(sso.nonce).to_not be_nil expect(sso.nonce).to_not be_nil
end end
context 'nonce error' do
it "generates correct error message when nonce has already been used" do
_ , payload = DiscourseSingleSignOn.generate_url(secure_session: secure_session).split("?")
sso = DiscourseSingleSignOn.parse(payload, secure_session: secure_session)
expect(sso.nonce_valid?).to eq true
sso.expire_nonce!
expect(sso.nonce_error).to eq("Nonce has already been used")
end
it "generates correct error message when nonce is expired" do
_ , payload = DiscourseSingleSignOn.generate_url(secure_session: secure_session).split("?")
sso = DiscourseSingleSignOn.parse(payload, secure_session: secure_session)
expect(sso.nonce_valid?).to eq true
Discourse.cache.delete(sso.used_nonce_key)
expect(sso.nonce_error).to eq("Nonce has expired")
end
end
context 'user locale' do context 'user locale' do
it 'sets default user locale if specified' do it 'sets default user locale if specified' do
SiteSetting.allow_user_locale = true SiteSetting.allow_user_locale = true