FIX: Perform better email validation (#12497)

Using UserEmail for validation is not sufficient because it checks the
emails of staged users too.
This commit is contained in:
Bianca Nenciu 2021-03-23 23:44:51 +02:00 committed by GitHub
parent be5ed73f08
commit f3eab6a86a
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 40 additions and 9 deletions

View File

@ -524,20 +524,38 @@ class UsersController < ApplicationController
end end
def check_email def check_email
RateLimiter.new(nil, "check-email-#{request.remote_ip}", 10, 1.minute).performed! begin
RateLimiter.new(nil, "check-email-#{request.remote_ip}", 10, 1.minute).performed!
if SiteSetting.hide_email_address_taken? rescue RateLimiter::LimitExceeded
return render json: success_json return render json: success_json
end end
user_email = UserEmail.new(email: params[:email]) email = Email.downcase((params[:email] || "").strip)
if user_email.valid? if email.blank? || SiteSetting.hide_email_address_taken?
render json: success_json return render json: success_json
else
render json: failed_json.merge(errors: user_email.errors.full_messages)
end end
rescue RateLimiter::LimitExceeded
if !(email =~ EmailValidator.email_regex)
error = User.new.errors.full_message(:email, I18n.t(:'user.email.invalid'))
return render json: failed_json.merge(errors: [error])
end
if !EmailValidator.allowed?(email)
error = User.new.errors.full_message(:email, I18n.t(:'user.email.not_allowed'))
return render json: failed_json.merge(errors: [error])
end
if ScreenedEmail.should_block?(email)
error = User.new.errors.full_message(:email, I18n.t(:'user.email.blocked'))
return render json: failed_json.merge(errors: [error])
end
if User.where(staged: false).find_by_email(email).present?
error = User.new.errors.full_message(:email, I18n.t(:'errors.messages.taken'))
return render json: failed_json.merge(errors: [error])
end
render json: success_json render json: success_json
end end

View File

@ -1592,6 +1592,11 @@ describe UsersController do
expect(response.parsed_body["success"]).to be_present expect(response.parsed_body["success"]).to be_present
end end
it 'returns success if email is empty' do
get "/u/check_email.json"
expect(response.parsed_body["success"]).to be_present
end
it 'returns failure if email is not valid' do it 'returns failure if email is not valid' do
get "/u/check_email.json", params: { email: "invalid" } get "/u/check_email.json", params: { email: "invalid" }
expect(response.parsed_body["failed"]).to be_present expect(response.parsed_body["failed"]).to be_present
@ -1600,12 +1605,20 @@ describe UsersController do
it 'returns failure if email exists' do it 'returns failure if email exists' do
get "/u/check_email.json", params: { email: user.email } get "/u/check_email.json", params: { email: user.email }
expect(response.parsed_body["failed"]).to be_present expect(response.parsed_body["failed"]).to be_present
get "/u/check_email.json", params: { email: user.email.upcase }
expect(response.parsed_body["failed"]).to be_present
end end
it 'returns success if email does not exists' do it 'returns success if email does not exists' do
get "/u/check_email.json", params: { email: "available@example.com" } get "/u/check_email.json", params: { email: "available@example.com" }
expect(response.parsed_body["success"]).to be_present expect(response.parsed_body["success"]).to be_present
end end
it 'return success if user email is taken by staged user' do
get "/u/check_email.json", params: { email: Fabricate(:staged).email }
expect(response.parsed_body["success"]).to be_present
end
end end
describe '#invited' do describe '#invited' do