Commit 7e862f36 authored by Stan Hu's avatar Stan Hu

Do not attribute unverified commit e-mails to GitLab users

Previously we would identify any user with a matching e-mail address as
a commit author or committer, but this caused confusion when commits
contained invalid, unconfirmable e-mail addresses (e.g. my.local).

Now, we only assign a commit author or committer is a specific user if
the e-mail has been confirmed.

Closes https://gitlab.com/gitlab-org/gitlab/issues/28493
parent 54506cc8
...@@ -246,7 +246,7 @@ class Commit ...@@ -246,7 +246,7 @@ class Commit
def lazy_author def lazy_author
BatchLoader.for(author_email.downcase).batch do |emails, loader| BatchLoader.for(author_email.downcase).batch do |emails, loader|
users = User.by_any_email(emails).includes(:emails) users = User.by_any_email(emails, confirmed: true).includes(:emails)
emails.each do |email| emails.each do |email|
user = users.find { |u| u.any_email?(email) } user = users.find { |u| u.any_email?(email) }
...@@ -263,8 +263,8 @@ class Commit ...@@ -263,8 +263,8 @@ class Commit
end end
request_cache(:author) { author_email.downcase } request_cache(:author) { author_email.downcase }
def committer def committer(confirmed: true)
@committer ||= User.find_by_any_email(committer_email) @committer ||= User.find_by_any_email(committer_email, confirmed: confirmed)
end end
def parents def parents
......
---
title: Do not attribute unverified commit e-mails to GitLab users
merge_request: 21214
author:
type: fixed
...@@ -82,9 +82,13 @@ module EE ...@@ -82,9 +82,13 @@ module EE
def committer_check(commit) def committer_check(commit)
unless push_rule.committer_allowed?(commit.committer_email, user_access.user) unless push_rule.committer_allowed?(commit.committer_email, user_access.user)
committer_is_current_user = commit.committer == user_access.user # We can assume only one user holds an unconfirmed e-mail address. Since we want
# to give feedback whether this is an unconfirmed address, we look for any user that
# matches by disabling the confirmation requirement.
committer = commit.committer(confirmed: false)
committer_is_current_user = committer == user_access.user
if committer_is_current_user && !commit.committer.verified_email?(commit.committer_email) if committer_is_current_user && !committer.verified_email?(commit.committer_email)
ERROR_MESSAGES[:committer_not_verified] % { committer_email: commit.committer_email } ERROR_MESSAGES[:committer_not_verified] % { committer_email: commit.committer_email }
else else
ERROR_MESSAGES[:committer_not_allowed] % { committer_email: commit.committer_email } ERROR_MESSAGES[:committer_not_allowed] % { committer_email: commit.committer_email }
......
...@@ -61,7 +61,7 @@ describe 'Member autocomplete', :js do ...@@ -61,7 +61,7 @@ describe 'Member autocomplete', :js do
before do before do
allow(User).to receive(:find_by_any_email) allow(User).to receive(:find_by_any_email)
.with(noteable.author_email.downcase).and_return(author) .with(noteable.author_email.downcase, confirmed: true).and_return(author)
visit project_commit_path(project, noteable) visit project_commit_path(project, noteable)
end end
......
...@@ -76,16 +76,23 @@ describe 'User browses commits' do ...@@ -76,16 +76,23 @@ describe 'User browses commits' do
end end
context 'secondary email' do context 'secondary email' do
let(:user) { create(:user) }
it 'finds a commit by a secondary email' do it 'finds a commit by a secondary email' do
user = create(:email, :confirmed, user: user, email: 'dmitriy.zaporozhets@gmail.com')
create(:user) do |user|
create(:email, { user: user, email: 'dmitriy.zaporozhets@gmail.com' })
end
visit(project_commit_path(project, sample_commit.parent_id)) visit(project_commit_path(project, sample_commit.parent_id))
check_author_link(sample_commit.author_email, user) check_author_link(sample_commit.author_email, user)
end end
it 'links to an unverified e-mail address instead of the user' do
create(:email, user: user, email: 'dmitriy.zaporozhets@gmail.com')
visit(project_commit_path(project, sample_commit.parent_id))
check_author_email(sample_commit.author_email)
end
end end
context 'when the blob does not exist' do context 'when the blob does not exist' do
...@@ -263,3 +270,9 @@ def check_author_link(email, author) ...@@ -263,3 +270,9 @@ def check_author_link(email, author)
expect(author_link['href']).to eq(user_path(author)) expect(author_link['href']).to eq(user_path(author))
expect(find('.commit-author-name').text).to eq(author.name) expect(find('.commit-author-name').text).to eq(author.name)
end end
def check_author_email(email)
author_link = find('.commit-author-link')
expect(author_link['href']).to eq("mailto:#{email}")
end
...@@ -80,6 +80,17 @@ describe Commit do ...@@ -80,6 +80,17 @@ describe Commit do
expect(commit.author).to eq(user) expect(commit.author).to eq(user)
end end
context 'with a user with an unconfirmed e-mail' do
before do
user = create(:user)
create(:email, user: user, email: commit.author_email)
end
it 'returns no user' do
expect(commit.author).to be_nil
end
end
context 'using eager loading' do context 'using eager loading' do
let!(:alice) { create(:user, email: 'alice@example.com') } let!(:alice) { create(:user, email: 'alice@example.com') }
let!(:bob) { create(:user, email: 'hunter2@example.com') } let!(:bob) { create(:user, email: 'hunter2@example.com') }
...@@ -115,7 +126,7 @@ describe Commit do ...@@ -115,7 +126,7 @@ describe Commit do
let!(:commits) { [alice_commit, bob_commit, eve_commit, jeff_commit] } let!(:commits) { [alice_commit, bob_commit, eve_commit, jeff_commit] }
before do before do
create(:email, user: bob, email: 'bob@example.com') create(:email, :confirmed, user: bob, email: 'bob@example.com')
end end
it 'executes only two SQL queries' do it 'executes only two SQL queries' do
...@@ -179,6 +190,32 @@ describe Commit do ...@@ -179,6 +190,32 @@ describe Commit do
end end
end end
describe '#committer' do
context 'with a confirmed e-mail' do
it 'returns the user' do
user = create(:user, email: commit.committer_email)
expect(commit.committer).to eq(user)
end
end
context 'with an unconfirmed e-mail' do
let(:user) { create(:user) }
before do
create(:email, user: user, email: commit.committer_email)
end
it 'returns no user' do
expect(commit.committer).to be_nil
end
it 'returns the user' do
expect(commit.committer(confirmed: false)).to eq(user)
end
end
end
describe '#to_reference' do describe '#to_reference' do
let(:project) { create(:project, :repository, path: 'sample-project') } let(:project) { create(:project, :repository, path: 'sample-project') }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment