123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141 |
- # Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- require 'rotp'
- require 'webauthn'
- FactoryBot.define do
- factory :'user/two_factor_preference', aliases: %i[user_two_factor_preference] do
- transient do
- user { association(:user, preferences: { two_factor_authentication: { default: method } }) }
- end
- user_id { user.id }
- updated_by_id { user.id }
- created_by_id { user.id }
- trait :authenticator_app do
- add_attribute(:method) { 'authenticator_app' }
- transient do
- secret { ROTP::Base32.random_base32 }
- code { ROTP::TOTP.new(secret).now }
- end
- configuration do
- {
- secret: secret,
- code: code, # Store a valid code for usage from the tests.
- provisioning_uri: ROTP::TOTP.new(secret, issuer: 'Zammad CI').provisioning_uri(user.login),
- }
- end
- end
- trait :security_keys do
- add_attribute(:method) { 'security_keys' }
- transient do
- credential_external_id { Faker::Alphanumeric.alpha(number: 70) }
- credential_public_key { Faker::Alphanumeric.alpha(number: 128) }
- # A fake static key is enough for most of the tests.
- credential do
- {
- external_id: credential_external_id,
- public_key: credential_public_key,
- nickname: Faker::Lorem.unique.word,
- sign_count: '0',
- created_at: Time.zone.now,
- }
- end
- end
- configuration do
- {
- credentials: [credential],
- }
- end
- end
- trait :mocked_security_keys do
- add_attribute(:method) { 'security_keys' }
- transient do
- page { raise NotImplementedError, 'You must provide current page object for mocking credentials' }
- wrong_key { false }
- # We can mock a WebAuthn credential only within a running browser session.
- # The code below is a pretty heavy "hack" to get the credential information from the Selenium virtual
- # authenticator, by simulating the complete registration process.
- # First, the create options are generated via Ruby code.
- # Then, a virtual authenticator instance is set up within the browser session with a mocked U2F key.
- # Create options are passed to JS, which triggers the registration of the key.
- # Finally, returned key information is processed back in Ruby, and the mocked credential can be "stored" in
- # the emulated two factor preferences.
- credential do
- WebAuthn.configure do |config|
- config.origin = "#{Setting.get('http_type')}://#{Capybara.app_host.gsub(%r{^https?://}, '')}:#{Capybara.current_session.server.port}"
- config.rp_name = Setting.get('organization').presence || Setting.get('product_name').presence || 'Zammad'
- config.credential_options_timeout = 120_000
- end
- initiate_configuration = WebAuthn::Credential.options_for_create(
- user: {
- id: WebAuthn.generate_user_id,
- display_name: user.login,
- name: user.login,
- },
- )
- options = Selenium::WebDriver::VirtualAuthenticatorOptions.new(protocol: :u2f, transport: :usb,
- resident_key: false, user_consenting: true,
- user_verification: true, user_verified: true)
- page.driver.browser.add_virtual_authenticator(options)
- public_key_json = JSON.generate({ publicKey: initiate_configuration.as_json.to_h })
- public_key_credential = page.execute_script("return webauthnJSON.create(#{public_key_json}).then((publicKeyCredential) => publicKeyCredential);")
- webauthn_credential = WebAuthn::Credential.from_create(public_key_credential)
- if wrong_key
- {
- external_id: Faker::Alphanumeric.alpha(number: 70),
- public_key: Faker::Alphanumeric.alpha(number: 128),
- nickname: Faker::Lorem.unique.word,
- sign_count: '0',
- created_at: Time.zone.now,
- }
- else
- {
- external_id: webauthn_credential.id,
- public_key: webauthn_credential.public_key,
- nickname: Faker::Lorem.unique.word,
- sign_count: webauthn_credential.sign_count.to_s,
- created_at: Time.zone.now,
- }
- end
- end
- end
- configuration do
- {
- credentials: [credential],
- }
- end
- end
- trait :recovery_codes do
- add_attribute(:method) { 'recovery_codes' }
- transient do
- user { association :user }
- recovery_code { 'example' }
- end
- configuration do
- {
- codes: [PasswordHash.crypt(recovery_code)],
- }
- end
- end
- end
- end
|