Commit ba771af3 authored by Imre Farkas's avatar Imre Farkas

Improve logging around U2F to WebAuthn conversion

Some U2fRegistrations are not migrated to WebAuthn despite the
after_commit hook in U2fRegistration model. This commit adds more
logging to gain more visibility.
parent 97c2def3
...@@ -4,11 +4,19 @@ ...@@ -4,11 +4,19 @@
class U2fRegistration < ApplicationRecord class U2fRegistration < ApplicationRecord
belongs_to :user belongs_to :user
after_commit :schedule_webauthn_migration, on: :create
after_commit :update_webauthn_registration, on: :update, if: :counter_changed?
def schedule_webauthn_migration after_create :create_webauthn_registration
BackgroundMigrationWorker.perform_async('MigrateU2fWebauthn', [id, id]) after_update :update_webauthn_registration, if: :counter_changed?
def create_webauthn_registration
converter = Gitlab::Auth::U2fWebauthnConverter.new(self)
WebauthnRegistration.create!(converter.convert)
rescue StandardError => ex
Gitlab::AppJsonLogger.error(
event: 'u2f_migration',
error: ex.class.name,
backtrace: ::Gitlab::BacktraceCleaner.clean_backtrace(ex.backtrace),
message: "U2F to WebAuthn conversion failed")
end end
def update_webauthn_registration def update_webauthn_registration
......
# frozen_string_literal: true
module Gitlab
module Auth
class U2fWebauthnConverter
def initialize(u2f_registration)
@u2f_registration = u2f_registration
end
def convert
now = Time.current
converted_credential = WebAuthn::U2fMigrator.new(
app_id: Gitlab.config.gitlab.url,
certificate: u2f_registration.certificate,
key_handle: u2f_registration.key_handle,
public_key: u2f_registration.public_key,
counter: u2f_registration.counter
).credential
{
credential_xid: Base64.strict_encode64(converted_credential.id),
public_key: Base64.strict_encode64(converted_credential.public_key),
counter: u2f_registration.counter || 0,
name: u2f_registration.name || '',
user_id: u2f_registration.user_id,
u2f_registration_id: u2f_registration.id,
created_at: now,
updated_at: now
}
end
private
attr_reader :u2f_registration
end
end
end
...@@ -16,26 +16,9 @@ module Gitlab ...@@ -16,26 +16,9 @@ module Gitlab
def perform(start_id, end_id) def perform(start_id, end_id)
old_registrations = U2fRegistration.where(id: start_id..end_id) old_registrations = U2fRegistration.where(id: start_id..end_id)
old_registrations.each_slice(100) do |slice| old_registrations.each_slice(100) do |slice|
now = Time.now
values = slice.map do |u2f_registration| values = slice.map do |u2f_registration|
converted_credential = WebAuthn::U2fMigrator.new( converter = Gitlab::Auth::U2fWebauthnConverter.new(u2f_registration)
app_id: Gitlab.config.gitlab.url, converter.convert
certificate: u2f_registration.certificate,
key_handle: u2f_registration.key_handle,
public_key: u2f_registration.public_key,
counter: u2f_registration.counter
).credential
{
credential_xid: Base64.strict_encode64(converted_credential.id),
public_key: Base64.strict_encode64(converted_credential.public_key),
counter: u2f_registration.counter || 0,
name: u2f_registration.name || '',
user_id: u2f_registration.user_id,
u2f_registration_id: u2f_registration.id,
created_at: now,
updated_at: now
}
end end
WebauthnRegistration.insert_all(values, unique_by: :credential_xid, returning: false) WebauthnRegistration.insert_all(values, unique_by: :credential_xid, returning: false)
......
...@@ -2,6 +2,8 @@ ...@@ -2,6 +2,8 @@
FactoryBot.define do FactoryBot.define do
factory :u2f_registration do factory :u2f_registration do
user
certificate { FFaker::BaconIpsum.characters(728) } certificate { FFaker::BaconIpsum.characters(728) }
key_handle { FFaker::BaconIpsum.characters(86) } key_handle { FFaker::BaconIpsum.characters(86) }
public_key { FFaker::BaconIpsum.characters(88) } public_key { FFaker::BaconIpsum.characters(88) }
......
...@@ -39,6 +39,11 @@ RSpec.describe 'Using U2F (Universal 2nd Factor) Devices for Authentication', :j ...@@ -39,6 +39,11 @@ RSpec.describe 'Using U2F (Universal 2nd Factor) Devices for Authentication', :j
end end
it 'allows the same device to be registered for multiple users' do it 'allows the same device to be registered for multiple users' do
# U2f specs will be removed after WebAuthn migration completed
pending('FakeU2fDevice has static key handle, '\
'leading to duplicate credential_xid for WebAuthn during migration, '\
'resulting in unique constraint violation')
# First user # First user
visit profile_account_path visit profile_account_path
manage_two_factor_authentication manage_two_factor_authentication
...@@ -148,6 +153,11 @@ RSpec.describe 'Using U2F (Universal 2nd Factor) Devices for Authentication', :j ...@@ -148,6 +153,11 @@ RSpec.describe 'Using U2F (Universal 2nd Factor) Devices for Authentication', :j
describe "and also the current user" do describe "and also the current user" do
it "allows logging in with that particular device" do it "allows logging in with that particular device" do
# U2f specs will be removed after WebAuthn migration completed
pending('FakeU2fDevice has static key handle, '\
'leading to duplicate credential_xid for WebAuthn during migration, '\
'resulting in unique constraint violation')
# Register current user with the same U2F device # Register current user with the same U2F device
current_user = gitlab_sign_in(:user) current_user = gitlab_sign_in(:user)
current_user.update_attribute(:otp_required_for_login, true) current_user.update_attribute(:otp_required_for_login, true)
......
...@@ -129,6 +129,10 @@ RSpec.describe 'Using WebAuthn Devices for Authentication', :js do ...@@ -129,6 +129,10 @@ RSpec.describe 'Using WebAuthn Devices for Authentication', :js do
end end
it 'falls back to U2F' do it 'falls back to U2F' do
# WebAuthn registration is automatically created with the U2fRegistration because of the after_create callback
# so we need to delete it
WebauthnRegistration.delete_all
gitlab_sign_in(user) gitlab_sign_in(user)
u2f_device.respond_to_u2f_authentication u2f_device.respond_to_u2f_authentication
......
# frozen_string_literal: true
require 'spec_helper'
RSpec.describe Gitlab::Auth::U2fWebauthnConverter do
let_it_be(:u2f_registration) do
device = U2F::FakeU2F.new(FFaker::BaconIpsum.characters(5))
create(:u2f_registration, name: 'u2f_device',
certificate: Base64.strict_encode64(device.cert_raw),
key_handle: U2F.urlsafe_encode64(device.key_handle_raw),
public_key: Base64.strict_encode64(device.origin_public_key_raw))
end
it 'converts u2f registration' do
webauthn_credential = WebAuthn::U2fMigrator.new(
app_id: Gitlab.config.gitlab.url,
certificate: u2f_registration.certificate,
key_handle: u2f_registration.key_handle,
public_key: u2f_registration.public_key,
counter: u2f_registration.counter
).credential
converted_webauthn = described_class.new(u2f_registration).convert
expect(converted_webauthn).to(
include(user_id: u2f_registration.user_id,
credential_xid: Base64.strict_encode64(webauthn_credential.id)))
end
end
# frozen_string_literal: true
require 'spec_helper'
RSpec.describe U2fRegistration do
let_it_be(:user) { create(:user) }
let(:u2f_registration) do
device = U2F::FakeU2F.new(FFaker::BaconIpsum.characters(5))
create(:u2f_registration, name: 'u2f_device',
user: user,
certificate: Base64.strict_encode64(device.cert_raw),
key_handle: U2F.urlsafe_encode64(device.key_handle_raw),
public_key: Base64.strict_encode64(device.origin_public_key_raw))
end
describe 'callbacks' do
describe '#create_webauthn_registration' do
it 'creates webauthn registration' do
u2f_registration.save!
webauthn_registration = WebauthnRegistration.where(u2f_registration_id: u2f_registration.id)
expect(webauthn_registration).to exist
end
it 'logs error' do
allow(Gitlab::Auth::U2fWebauthnConverter).to receive(:new).and_raise('boom!')
expect(Gitlab::AppJsonLogger).to(
receive(:error).with(a_hash_including(event: 'u2f_migration',
error: 'RuntimeError',
message: 'U2F to WebAuthn conversion failed'))
)
u2f_registration.save!
end
end
end
end
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