2019-10-01 22:08:41 -04:00
|
|
|
# frozen_string_literal: true
|
2020-01-09 19:45:56 -05:00
|
|
|
require "webauthn/challenge_generator"
|
2019-10-01 22:08:41 -04:00
|
|
|
require "webauthn/security_key_base_validation_service"
|
|
|
|
require "webauthn/security_key_registration_service"
|
|
|
|
require "webauthn/security_key_authentication_service"
|
|
|
|
|
2023-08-18 08:39:10 -04:00
|
|
|
module DiscourseWebauthn
|
2020-04-30 02:48:34 -04:00
|
|
|
ACCEPTABLE_REGISTRATION_TYPE = "webauthn.create"
|
|
|
|
ACCEPTABLE_AUTHENTICATION_TYPE = "webauthn.get"
|
2019-11-21 00:45:36 -05:00
|
|
|
|
|
|
|
# -7 - ES256
|
|
|
|
# -257 - RS256 (Windows Hello supported alg.)
|
2022-02-08 07:07:47 -05:00
|
|
|
SUPPORTED_ALGORITHMS = COSE::Algorithm.registered_algorithm_ids.freeze
|
2019-10-01 22:08:41 -04:00
|
|
|
VALID_ATTESTATION_FORMATS = %w[none packed fido-u2f].freeze
|
|
|
|
|
|
|
|
class SecurityKeyError < StandardError
|
|
|
|
end
|
|
|
|
|
|
|
|
class InvalidOriginError < SecurityKeyError
|
|
|
|
end
|
|
|
|
class InvalidRelyingPartyIdError < SecurityKeyError
|
|
|
|
end
|
|
|
|
class UserVerificationError < SecurityKeyError
|
|
|
|
end
|
|
|
|
class ChallengeMismatchError < SecurityKeyError
|
|
|
|
end
|
|
|
|
class InvalidTypeError < SecurityKeyError
|
|
|
|
end
|
|
|
|
class UnsupportedPublicKeyAlgorithmError < SecurityKeyError
|
|
|
|
end
|
|
|
|
class UnsupportedAttestationFormatError < SecurityKeyError
|
|
|
|
end
|
|
|
|
class CredentialIdInUseError < SecurityKeyError
|
|
|
|
end
|
|
|
|
class MalformedAttestationError < SecurityKeyError
|
|
|
|
end
|
|
|
|
class NotFoundError < SecurityKeyError
|
|
|
|
end
|
|
|
|
class OwnershipError < SecurityKeyError
|
|
|
|
end
|
|
|
|
class PublicKeyError < SecurityKeyError
|
|
|
|
end
|
2020-01-01 19:14:22 -05:00
|
|
|
class UnknownCOSEAlgorithmError < SecurityKeyError
|
|
|
|
end
|
2020-01-09 19:45:56 -05:00
|
|
|
|
|
|
|
##
|
|
|
|
# Usage:
|
|
|
|
#
|
|
|
|
# These methods should be used in controllers where we
|
|
|
|
# are challenging the user that has a security key, and
|
|
|
|
# they must respond with a valid webauthn response and
|
|
|
|
# credentials.
|
|
|
|
def self.stage_challenge(user, secure_session)
|
2023-08-18 08:39:10 -04:00
|
|
|
::DiscourseWebauthn::ChallengeGenerator.generate.commit_to_session(secure_session, user)
|
2020-01-09 19:45:56 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
def self.allowed_credentials(user, secure_session)
|
|
|
|
return {} if !user.security_keys_enabled?
|
|
|
|
credential_ids = user.second_factor_security_key_credential_ids
|
|
|
|
{
|
|
|
|
allowed_credential_ids: credential_ids,
|
|
|
|
challenge:
|
2023-08-18 08:39:10 -04:00
|
|
|
secure_session[
|
|
|
|
DiscourseWebauthn::ChallengeGenerator::ChallengeSession.session_challenge_key(user)
|
|
|
|
],
|
2020-01-09 19:45:56 -05:00
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.rp_id(user, secure_session)
|
2023-08-18 08:39:10 -04:00
|
|
|
secure_session[DiscourseWebauthn::ChallengeGenerator::ChallengeSession.session_rp_id_key(user)]
|
2020-01-09 19:45:56 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
def self.rp_name(user, secure_session)
|
2023-08-18 08:39:10 -04:00
|
|
|
secure_session[
|
|
|
|
DiscourseWebauthn::ChallengeGenerator::ChallengeSession.session_rp_name_key(user)
|
|
|
|
]
|
2020-01-09 19:45:56 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
def self.challenge(user, secure_session)
|
2023-08-18 08:39:10 -04:00
|
|
|
secure_session[
|
|
|
|
DiscourseWebauthn::ChallengeGenerator::ChallengeSession.session_challenge_key(user)
|
|
|
|
]
|
|
|
|
end
|
2019-10-01 22:08:41 -04:00
|
|
|
end
|