Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
G
gitlab-ce
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
1
Merge Requests
1
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
nexedi
gitlab-ce
Commits
1adb89c1
Commit
1adb89c1
authored
Aug 07, 2017
by
Sean McGivern
Browse files
Options
Browse Files
Download
Plain Diff
Merge branch 'feature/audit-events-failed-login' into 'master'
Audit failed login events See merge request !2587
parents
258792a0
e74c7e2a
Changes
10
Hide whitespace changes
Inline
Side-by-side
Showing
10 changed files
with
149 additions
and
12 deletions
+149
-12
app/controllers/omniauth_callbacks_controller.rb
app/controllers/omniauth_callbacks_controller.rb
+16
-6
app/controllers/sessions_controller.rb
app/controllers/sessions_controller.rb
+1
-0
changelogs/unreleased-ee/feature-audit-events-failed-login.yml
...elogs/unreleased-ee/feature-audit-events-failed-login.yml
+4
-0
doc/administration/audit_events.md
doc/administration/audit_events.md
+1
-0
ee/app/controllers/ee/omniauth_callbacks_controller.rb
ee/app/controllers/ee/omniauth_callbacks_controller.rb
+31
-0
ee/app/controllers/ee/sessions_controller.rb
ee/app/controllers/ee/sessions_controller.rb
+20
-0
ee/app/services/ee/audit_event_service.rb
ee/app/services/ee/audit_event_service.rb
+27
-0
lib/audit/details.rb
lib/audit/details.rb
+8
-6
spec/ee/spec/services/ee/audit_event_service_spec.rb
spec/ee/spec/services/ee/audit_event_service_spec.rb
+31
-0
spec/features/login_spec.rb
spec/features/login_spec.rb
+10
-0
No files found.
app/controllers/omniauth_callbacks_controller.rb
View file @
1adb89c1
class
OmniauthCallbacksController
<
Devise
::
OmniauthCallbacksController
class
OmniauthCallbacksController
<
Devise
::
OmniauthCallbacksController
include
AuthenticatesWithTwoFactor
include
AuthenticatesWithTwoFactor
include
Devise
::
Controllers
::
Rememberable
include
Devise
::
Controllers
::
Rememberable
prepend
EE
::
OmniauthCallbacksController
protect_from_forgery
except:
[
:kerberos
,
:saml
,
:cas3
]
protect_from_forgery
except:
[
:kerberos
,
:saml
,
:cas3
]
...
@@ -34,14 +35,13 @@ class OmniauthCallbacksController < Devise::OmniauthCallbacksController
...
@@ -34,14 +35,13 @@ class OmniauthCallbacksController < Devise::OmniauthCallbacksController
if
@user
.
two_factor_enabled?
if
@user
.
two_factor_enabled?
prompt_for_two_factor
(
@user
)
prompt_for_two_factor
(
@user
)
else
else
log_audit_event
(
@user
,
with:
:ldap
)
log_audit_event
(
@user
,
with:
oauth
[
'provider'
]
)
# The counter only gets incremented in `sign_in_and_redirect`
# The counter only gets incremented in `sign_in_and_redirect`
show_ldap_sync_flash
if
@user
.
sign_in_count
==
0
show_ldap_sync_flash
if
@user
.
sign_in_count
==
0
sign_in_and_redirect
(
@user
)
sign_in_and_redirect
(
@user
)
end
end
else
else
flash
[
:alert
]
=
"Access denied for your LDAP account."
fail_ldap_login
redirect_to
new_user_session_path
end
end
end
end
...
@@ -136,9 +136,7 @@ class OmniauthCallbacksController < Devise::OmniauthCallbacksController
...
@@ -136,9 +136,7 @@ class OmniauthCallbacksController < Devise::OmniauthCallbacksController
sign_in_and_redirect
(
@user
)
sign_in_and_redirect
(
@user
)
end
end
else
else
error_message
=
@user
.
errors
.
full_messages
.
to_sentence
fail_login
return
redirect_to
omniauth_error_path
(
oauth
[
'provider'
],
error:
error_message
)
end
end
end
end
...
@@ -159,6 +157,18 @@ class OmniauthCallbacksController < Devise::OmniauthCallbacksController
...
@@ -159,6 +157,18 @@ class OmniauthCallbacksController < Devise::OmniauthCallbacksController
@oauth
||=
request
.
env
[
'omniauth.auth'
]
@oauth
||=
request
.
env
[
'omniauth.auth'
]
end
end
def
fail_login
error_message
=
@user
.
errors
.
full_messages
.
to_sentence
return
redirect_to
omniauth_error_path
(
oauth
[
'provider'
],
error:
error_message
)
end
def
fail_ldap_login
flash
[
:alert
]
=
'Access denied for your LDAP account.'
redirect_to
new_user_session_path
end
def
log_audit_event
(
user
,
options
=
{})
def
log_audit_event
(
user
,
options
=
{})
AuditEventService
.
new
(
user
,
user
,
options
)
AuditEventService
.
new
(
user
,
user
,
options
)
.
for_authentication
.
security_event
.
for_authentication
.
security_event
...
...
app/controllers/sessions_controller.rb
View file @
1adb89c1
...
@@ -2,6 +2,7 @@ class SessionsController < Devise::SessionsController
...
@@ -2,6 +2,7 @@ class SessionsController < Devise::SessionsController
include
AuthenticatesWithTwoFactor
include
AuthenticatesWithTwoFactor
include
Devise
::
Controllers
::
Rememberable
include
Devise
::
Controllers
::
Rememberable
include
Recaptcha
::
ClientHelper
include
Recaptcha
::
ClientHelper
prepend
EE
::
SessionsController
skip_before_action
:check_two_factor_requirement
,
only:
[
:destroy
]
skip_before_action
:check_two_factor_requirement
,
only:
[
:destroy
]
...
...
changelogs/unreleased-ee/feature-audit-events-failed-login.yml
0 → 100644
View file @
1adb89c1
---
title
:
Audit failed login events
merge_request
:
2587
author
:
doc/administration/audit_events.md
View file @
1adb89c1
...
@@ -25,6 +25,7 @@ in your GitLab instance, and who gave them that permission level.
...
@@ -25,6 +25,7 @@ in your GitLab instance, and who gave them that permission level.
|--------------------------------|--------------------------------------------------------------------------------------------------|
|--------------------------------|--------------------------------------------------------------------------------------------------|
| User added to group or project | Notes the author of the change, target user |
| User added to group or project | Notes the author of the change, target user |
| User permission changed | Notes the author of the change, original permission and new permission, target user |
| User permission changed | Notes the author of the change, original permission and new permission, target user |
| User login failed | Notes the target username and IP address |
## Audit events in project
## Audit events in project
...
...
ee/app/controllers/ee/omniauth_callbacks_controller.rb
0 → 100644
View file @
1adb89c1
module
EE
module
OmniauthCallbacksController
protected
def
fail_login
log_failed_login
(
@user
.
username
,
oauth
[
'provider'
])
error_message
=
@user
.
errors
.
full_messages
.
to_sentence
return
redirect_to
omniauth_error_path
(
oauth
[
'provider'
],
error:
error_message
)
end
def
fail_ldap_login
log_failed_login
(
@user
.
username
,
oauth
[
'provider'
])
flash
[
:alert
]
=
'Access denied for your LDAP account.'
redirect_to
new_user_session_path
end
private
def
log_failed_login
(
author
,
provider
)
::
AuditEventService
.
new
(
author
,
nil
,
ip_address:
request
.
remote_ip
,
with:
provider
)
.
for_failed_login
.
unauth_security_event
end
end
end
ee/app/controllers/ee/sessions_controller.rb
0 → 100644
View file @
1adb89c1
module
EE
module
SessionsController
extend
ActiveSupport
::
Concern
prepended
do
after_action
:log_failed_login
,
only: :new
,
if: :failed_login?
end
private
def
log_failed_login
::
AuditEventService
.
new
(
request
.
filtered_parameters
[
'user'
][
'login'
],
nil
,
ip_address:
request
.
remote_ip
)
.
for_failed_login
.
unauth_security_event
end
def
failed_login?
env
[
'warden.options'
]
&&
env
[
'warden.options'
][
:action
]
==
'unauthenticated'
end
end
end
ee/app/services/ee/audit_event_service.rb
View file @
1adb89c1
...
@@ -68,6 +68,20 @@ module EE
...
@@ -68,6 +68,20 @@ module EE
self
self
end
end
def
for_failed_login
ip
=
@details
[
:ip_address
]
auth
=
@details
[
:with
]
||
'STANDARD'
@details
=
{
failed_login:
auth
.
upcase
,
author_name:
@author
,
target_details:
@author
,
ip_address:
ip
}
self
end
def
security_event
def
security_event
if
admin_audit_log_enabled?
if
admin_audit_log_enabled?
add_security_event_admin_details!
add_security_event_admin_details!
...
@@ -92,5 +106,18 @@ module EE
...
@@ -92,5 +106,18 @@ module EE
def
admin_audit_log_enabled?
def
admin_audit_log_enabled?
License
.
feature_available?
(
:admin_audit_log
)
License
.
feature_available?
(
:admin_audit_log
)
end
end
def
unauth_security_event
return
unless
audit_events_enabled?
@details
.
delete
(
:ip_address
)
unless
admin_audit_log_enabled?
SecurityEvent
.
create
(
author_id:
-
1
,
entity_id:
-
1
,
entity_type:
'User'
,
details:
@details
)
end
end
end
end
end
lib/audit/details.rb
View file @
1adb89c1
module
Audit
module
Audit
class
Details
class
Details
CRUD_ACTIONS
=
%i[add remove
change]
.
freeze
ACTIONS
=
%i[add remove failed_login
change]
.
freeze
def
self
.
humanize
(
*
args
)
def
self
.
humanize
(
*
args
)
new
(
*
args
).
humanize
new
(
*
args
).
humanize
...
@@ -14,21 +14,23 @@ module Audit
...
@@ -14,21 +14,23 @@ module Audit
if
@details
[
:with
]
if
@details
[
:with
]
"Signed in with
#{
@details
[
:with
].
upcase
}
authentication"
"Signed in with
#{
@details
[
:with
].
upcase
}
authentication"
else
else
crud_
action_text
action_text
end
end
end
end
private
private
def
crud_
action_text
def
action_text
action
=
@details
.
slice
(
*
CRUD_
ACTIONS
)
action
=
@details
.
slice
(
*
ACTIONS
)
value
=
@details
.
values
.
first
.
tr
(
'_'
,
' '
)
value
=
@details
.
values
.
first
.
tr
(
'_'
,
' '
)
case
action
.
keys
.
first
case
action
.
keys
.
first
when
:add
when
:add
"Added
#{
value
}#{
@details
[
:as
]
?
" as
#{
@details
[
:as
]
}
"
:
""
}
"
"Added
#{
value
}#{
@details
[
:as
]
?
" as
#{
@details
[
:as
]
}
"
:
''
}
"
when
:remove
when
:remove
"Removed
#{
value
}
"
"Removed
#{
value
}
"
when
:failed_login
"Failed to login with
#{
Gitlab
::
OAuth
::
Provider
.
label_for
(
value
).
upcase
}
authentication"
else
else
"Changed
#{
value
}
from
#{
@details
[
:from
]
}
to
#{
@details
[
:to
]
}
"
"Changed
#{
value
}
from
#{
@details
[
:from
]
}
to
#{
@details
[
:to
]
}
"
end
end
...
...
spec/ee/spec/services/ee/audit_event_service_spec.rb
0 → 100644
View file @
1adb89c1
require
'spec_helper'
describe
AuditEventService
do
describe
'#for_failed_login'
do
let
(
:author_name
)
{
'testuser'
}
let
(
:ip_address
)
{
'127.0.0.1'
}
let
(
:service
)
{
described_class
.
new
(
author_name
,
nil
,
ip_address:
ip_address
)
}
let
(
:event
)
{
service
.
for_failed_login
.
unauth_security_event
}
it
'has the right type'
do
expect
(
event
.
entity_type
).
to
eq
(
'User'
)
end
it
'has the right author'
do
expect
(
event
.
details
[
:author_name
]).
to
eq
(
author_name
)
end
it
'has the right IP address'
do
allow
(
service
).
to
receive
(
:admin_audit_log_enabled?
).
and_return
(
true
)
expect
(
event
.
details
[
:ip_address
]).
to
eq
(
ip_address
)
end
it
'has the right auth method for OAUTH'
do
oauth_service
=
described_class
.
new
(
author_name
,
nil
,
ip_address:
ip_address
,
with:
'ldap'
)
event
=
oauth_service
.
for_failed_login
.
unauth_security_event
expect
(
event
.
details
[
:failed_login
]).
to
eq
(
'LDAP'
)
end
end
end
spec/features/login_spec.rb
View file @
1adb89c1
...
@@ -152,6 +152,14 @@ feature 'Login' do
...
@@ -152,6 +152,14 @@ feature 'Login' do
enter_code
(
user
.
current_otp
)
enter_code
(
user
.
current_otp
)
expect
(
current_path
).
to
eq
root_path
expect
(
current_path
).
to
eq
root_path
end
end
it
'creates a security event after failed OAuth login'
do
stub_omniauth_saml_config
(
enabled:
true
,
auto_link_saml_user:
true
,
allow_single_sign_on:
[
'saml'
],
providers:
[
mock_saml_config
])
user
=
create
(
:omniauth_user
,
:two_factor
,
extern_uid:
'my-uid'
,
provider:
'saml'
)
gitlab_sign_in_via
(
'saml'
,
user
,
'wrong-uid'
)
expect
(
SecurityEvent
.
where
(
entity_id:
-
1
).
count
).
to
eq
(
1
)
end
end
end
end
end
...
@@ -173,6 +181,8 @@ feature 'Login' do
...
@@ -173,6 +181,8 @@ feature 'Login' do
gitlab_sign_in
(
user
)
gitlab_sign_in
(
user
)
expect
(
page
).
to
have_content
(
'Invalid Login or password.'
)
expect
(
page
).
to
have_content
(
'Invalid Login or password.'
)
expect
(
SecurityEvent
.
where
(
entity_id:
-
1
).
count
).
to
eq
(
1
)
end
end
end
end
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment