diff --git a/.gitignore b/.gitignore
index 8aa11f5f..c66d1cc4 100644
--- a/.gitignore
+++ b/.gitignore
@@ -334,3 +334,5 @@ ASALocalRun/
/Test/coverage.netcoreapp3.1.cobertura.xml
.DS_Store
/testEnvironments.json
+
+Demo/Conformance/
\ No newline at end of file
diff --git a/Demo/TestController.cs b/Demo/TestController.cs
index d0d67163..5967a8af 100644
--- a/Demo/TestController.cs
+++ b/Demo/TestController.cs
@@ -1,6 +1,5 @@
using System.Text;
using System.Text.Json;
-using System.Text.Json.Serialization;
using Fido2NetLib;
using Fido2NetLib.Development;
using Fido2NetLib.Objects;
diff --git a/Src/Fido2.Models/AuthenticatorAttestationRawResponse.cs b/Src/Fido2.Models/AuthenticatorAttestationRawResponse.cs
index 278e8eb2..80337144 100644
--- a/Src/Fido2.Models/AuthenticatorAttestationRawResponse.cs
+++ b/Src/Fido2.Models/AuthenticatorAttestationRawResponse.cs
@@ -15,7 +15,7 @@ public sealed class AuthenticatorAttestationRawResponse
public byte[] RawId { get; set; }
[JsonPropertyName("type")]
- public PublicKeyCredentialType Type { get; set; } = PublicKeyCredentialType.PublicKey;
+ public PublicKeyCredentialType? Type { get; set; }
[JsonPropertyName("response")]
public AttestationResponse Response { get; set; }
diff --git a/Src/Fido2.Models/CredentialCreateOptions.cs b/Src/Fido2.Models/CredentialCreateOptions.cs
index 1e5b2239..43de046b 100644
--- a/Src/Fido2.Models/CredentialCreateOptions.cs
+++ b/Src/Fido2.Models/CredentialCreateOptions.cs
@@ -134,6 +134,7 @@ public static CredentialCreateOptions Create(
PubKeyCredParam.ES512,
PubKeyCredParam.RS512,
PubKeyCredParam.PS512,
+ PubKeyCredParam.RS1
],
AuthenticatorSelection = authenticatorSelection,
Attestation = attestationConveyancePreference,
@@ -185,6 +186,7 @@ public sealed class PubKeyCredParam(
public static readonly PubKeyCredParam PS384 = new(COSE.Algorithm.PS384);
public static readonly PubKeyCredParam PS512 = new(COSE.Algorithm.PS512);
public static readonly PubKeyCredParam Ed25519 = new(COSE.Algorithm.EdDSA);
+ public static readonly PubKeyCredParam RS1 = new(COSE.Algorithm.RS1);
}
///
diff --git a/Src/Fido2.Models/Objects/AuthenticationExtensionsClientInputs.cs b/Src/Fido2.Models/Objects/AuthenticationExtensionsClientInputs.cs
index db10e328..f6f8e4d1 100644
--- a/Src/Fido2.Models/Objects/AuthenticationExtensionsClientInputs.cs
+++ b/Src/Fido2.Models/Objects/AuthenticationExtensionsClientInputs.cs
@@ -12,14 +12,14 @@ public sealed class AuthenticationExtensionsClientInputs
///
[JsonPropertyName("example.extension.bool")]
[JsonIgnore(Condition = JsonIgnoreCondition.WhenWritingNull)]
- public object Example { get; set; }
+ public bool? Example { get; set; }
///
/// This extension allows WebAuthn Relying Parties that have previously registered a credential using the legacy FIDO JavaScript APIs to request an assertion.
/// https://www.w3.org/TR/webauthn/#sctn-appid-extension
///
[JsonPropertyName("appid")]
- [JsonIgnore(Condition = JsonIgnoreCondition.WhenWritingNull)]
+ [JsonIgnore(Condition = JsonIgnoreCondition.Always)]
public string AppID { get; set; }
///
@@ -33,10 +33,11 @@ public sealed class AuthenticationExtensionsClientInputs
///
/// This extension enables use of a user verification method.
/// https://www.w3.org/TR/webauthn/#sctn-uvm-extension
+ /// TODO: Remove this completely as it's removed in L3
///
[JsonPropertyName("uvm")]
[JsonIgnore(Condition = JsonIgnoreCondition.WhenWritingNull)]
- public bool? UserVerificationMethod { get; set; }
+ public bool? UserVerificationMethod { private get; set; }
#nullable enable
///
diff --git a/Src/Fido2.Models/Objects/AuthenticationExtensionsClientOutputs.cs b/Src/Fido2.Models/Objects/AuthenticationExtensionsClientOutputs.cs
index edcf8059..b5f15b59 100644
--- a/Src/Fido2.Models/Objects/AuthenticationExtensionsClientOutputs.cs
+++ b/Src/Fido2.Models/Objects/AuthenticationExtensionsClientOutputs.cs
@@ -9,7 +9,7 @@ public class AuthenticationExtensionsClientOutputs
///
[JsonPropertyName("example.extension.bool")]
[JsonIgnore(Condition = JsonIgnoreCondition.WhenWritingNull)]
- public object Example { get; set; }
+ public bool? Example { get; set; }
#nullable enable
diff --git a/Src/Fido2/AuthenticatorAssertionResponse.cs b/Src/Fido2/AuthenticatorAssertionResponse.cs
index e2b30b13..166901a2 100644
--- a/Src/Fido2/AuthenticatorAssertionResponse.cs
+++ b/Src/Fido2/AuthenticatorAssertionResponse.cs
@@ -52,6 +52,7 @@ public static AuthenticatorAssertionResponse Parse(AuthenticatorAssertionRawResp
/// The stored counter value for this CredentialId
/// A function that returns if user handle is owned by the credential ID.
///
+ /// DO NOT USE - Deprecated, but kept in code due to conformance testing tool
/// The used to propagate notifications that the operation should be canceled.
public async Task VerifyAsync(
AssertionOptions options,
@@ -61,9 +62,10 @@ public async Task VerifyAsync(
uint storedSignatureCounter,
IsUserHandleOwnerOfCredentialIdAsync isUserHandleOwnerOfCredId,
IMetadataService? metadataService,
+ byte[]? requestTokenBindingId,
CancellationToken cancellationToken = default)
{
- BaseVerify(config.FullyQualifiedOrigins, options.Challenge);
+ BaseVerify(config.FullyQualifiedOrigins, options.Challenge, requestTokenBindingId);
if (Raw.Type != PublicKeyCredentialType.PublicKey)
throw new Fido2VerificationException(Fido2ErrorCode.InvalidAssertionResponse, Fido2ErrorMessages.AssertionResponseNotPublicKey);
@@ -115,15 +117,20 @@ public async Task VerifyAsync(
// https://www.w3.org/TR/webauthn/#sctn-appid-extension
// FIDO AppID Extension:
// If true, the AppID was used and thus, when verifying an assertion, the Relying Party MUST expect the rpIdHash to be the hash of the AppID, not the RP ID.
+
var rpid = Raw.ClientExtensionResults?.AppID ?? false ? options.Extensions?.AppID : options.RpId;
+
byte[] hashedRpId = SHA256.HashData(Encoding.UTF8.GetBytes(rpid ?? string.Empty));
byte[] hash = SHA256.HashData(Raw.Response.ClientDataJson);
if (!authData.RpIdHash.SequenceEqual(hashedRpId))
throw new Fido2VerificationException(Fido2ErrorCode.InvalidRpidHash, Fido2ErrorMessages.InvalidRpidHash);
+ var conformanceTesting = metadataService != null && metadataService.ConformanceTesting();
+
// 14. Verify that the UP bit of the flags in authData is set.
- if (!authData.UserPresent)
+ // Todo: Conformance testing verifies the UVP flags differently than W3C spec, simplify this by removing the mention of conformanceTesting when conformance tools are updated)
+ if (!authData.UserPresent && !conformanceTesting)
throw new Fido2VerificationException(Fido2ErrorCode.UserPresentFlagNotSet, Fido2ErrorMessages.UserPresentFlagNotSet);
// 15. If the Relying Party requires user verification for this assertion, verify that the UV bit of the flags in authData is set.
@@ -174,6 +181,7 @@ public async Task VerifyAsync(
if (authData.SignCount > 0 && authData.SignCount <= storedSignatureCounter)
throw new Fido2VerificationException(Fido2ErrorCode.InvalidSignCount, Fido2ErrorMessages.SignCountIsLessThanSignatureCounter);
+
return new VerifyAssertionResult
{
CredentialId = Raw.Id,
diff --git a/Src/Fido2/AuthenticatorAttestationResponse.cs b/Src/Fido2/AuthenticatorAttestationResponse.cs
index d5665ae8..1522e54b 100644
--- a/Src/Fido2/AuthenticatorAttestationResponse.cs
+++ b/Src/Fido2/AuthenticatorAttestationResponse.cs
@@ -60,6 +60,7 @@ public async Task VerifyAsync(
Fido2Configuration config,
IsCredentialIdUniqueToUserAsyncDelegate isCredentialIdUniqueToUser,
IMetadataService? metadataService,
+ byte[]? requestTokenBindingId,
CancellationToken cancellationToken = default)
{
// https://www.w3.org/TR/webauthn/#registering-a-new-credential
@@ -74,7 +75,10 @@ public async Task VerifyAsync(
// 8. Verify that the value of C.challenge matches the challenge that was sent to the authenticator in the create() call.
// 9. Verify that the value of C.origin matches the Relying Party's origin.
- BaseVerify(config.FullyQualifiedOrigins, originalOptions.Challenge);
+ // 9.5. Verify that the value of C.tokenBinding.status matches the state of Token Binding for the TLS connection over which the attestation was obtained.
+ // If Token Binding was used on that TLS connection, also verify that C.tokenBinding.id matches the base64url encoding of the Token Binding ID for the connection.
+ // Validated in BaseVerify.
+ BaseVerify(config.FullyQualifiedOrigins, originalOptions.Challenge, requestTokenBindingId);
if (Raw.Id is null || Raw.Id.Length == 0)
throw new Fido2VerificationException(Fido2ErrorCode.InvalidAttestationResponse, Fido2ErrorMessages.AttestationResponseIdMissing);
@@ -149,7 +153,7 @@ public async Task VerifyAsync(
if (metadataService?.ConformanceTesting() is true && metadataEntry is null && attType != AttestationType.None && AttestationObject.Fmt is not "fido-u2f")
throw new Fido2VerificationException(Fido2ErrorCode.AaGuidNotFound, "AAGUID not found in MDS test metadata");
- TrustAnchor.Verify(metadataEntry, trustPath);
+ TrustAnchor.Verify(metadataEntry, trustPath, metadataService?.ConformanceTesting() is true ? FidoValidationMode.FidoConformance2024 : FidoValidationMode.Default);
// 22. Assess the attestation trustworthiness using the outputs of the verification procedure in step 14, as follows:
// If self attestation was used, check if self attestation is acceptable under Relying Party policy.
@@ -186,7 +190,7 @@ public async Task VerifyAsync(
return new RegisteredPublicKeyCredential
{
- Type = Raw.Type,
+ Type = Raw.Type.Value,
Id = authData.AttestedCredentialData.CredentialId,
PublicKey = authData.AttestedCredentialData.CredentialPublicKey.GetBytes(),
SignCount = authData.SignCount,
@@ -253,7 +257,7 @@ private async Task DevicePublicKeyRegistrationAsync(
if (metadataService?.ConformanceTesting() is true && metadataEntry is null && attType != AttestationType.None && devicePublicKeyAuthenticatorOutput.Fmt is not "fido-u2f")
throw new Fido2VerificationException(Fido2ErrorCode.AaGuidNotFound, "AAGUID not found in MDS test metadata");
- TrustAnchor.Verify(metadataEntry, trustPath);
+ TrustAnchor.Verify(metadataEntry, trustPath, metadataService?.ConformanceTesting() is true ? FidoValidationMode.FidoConformance2024 : FidoValidationMode.Default);
// Check status reports for authenticator with undesirable status
var latestStatusReport = metadataEntry?.GetLatestStatusReport();
diff --git a/Src/Fido2/AuthenticatorResponse.cs b/Src/Fido2/AuthenticatorResponse.cs
index 22c31635..eddee047 100644
--- a/Src/Fido2/AuthenticatorResponse.cs
+++ b/Src/Fido2/AuthenticatorResponse.cs
@@ -48,6 +48,7 @@ protected AuthenticatorResponse(ReadOnlySpan utf8EncodedJson)
Type = response.Type;
Challenge = response.Challenge;
Origin = response.Origin;
+ TokenBinding = response.TokenBinding;
}
public const int MAX_ORIGINS_TO_PRINT = 5;
@@ -62,7 +63,11 @@ protected AuthenticatorResponse(ReadOnlySpan utf8EncodedJson)
[JsonPropertyName("origin")]
public string Origin { get; }
- protected void BaseVerify(IReadOnlySet fullyQualifiedExpectedOrigins, ReadOnlySpan originalChallenge)
+ // [Obsolete("This property is not used and will be removed in a future version once the conformance tool stops testing for it.")]
+ [JsonPropertyName("tokenBinding")]
+ public TokenBindingDto? TokenBinding { get; set; }
+
+ protected void BaseVerify(IReadOnlySet fullyQualifiedExpectedOrigins, ReadOnlySpan originalChallenge, byte[]? requestTokenBindingId)
{
if (Type is not "webauthn.create" && Type is not "webauthn.get")
throw new Fido2VerificationException(Fido2ErrorCode.InvalidAuthenticatorResponse, $"Type must be 'webauthn.create' or 'webauthn.get'. Was '{Type}'");
@@ -79,6 +84,10 @@ protected void BaseVerify(IReadOnlySet fullyQualifiedExpectedOrigins, Re
// 12. Verify that the value of C.origin matches the Relying Party's origin.
if (!fullyQualifiedExpectedOrigins.Contains(fullyQualifiedOrigin))
throw new Fido2VerificationException($"Fully qualified origin {fullyQualifiedOrigin} of {Origin} not equal to fully qualified original origin {string.Join(", ", fullyQualifiedExpectedOrigins.Take(MAX_ORIGINS_TO_PRINT))} ({fullyQualifiedExpectedOrigins.Count})");
+
+ // 13?. Verify that the value of C.tokenBinding.status matches the state of Token Binding for the TLS connection over which the assertion was obtained.
+ // If Token Binding was used on that TLS connection, also verify that C.tokenBinding.id matches the base64url encoding of the Token Binding ID for the connection.
+ TokenBinding?.Verify(requestTokenBindingId);
}
/*
diff --git a/Src/Fido2/Extensions/CryptoUtils.cs b/Src/Fido2/Extensions/CryptoUtils.cs
index 719d5eaf..8714becf 100644
--- a/Src/Fido2/Extensions/CryptoUtils.cs
+++ b/Src/Fido2/Extensions/CryptoUtils.cs
@@ -49,7 +49,7 @@ public static HashAlgorithmName HashAlgFromCOSEAlg(COSE.Algorithm alg)
};
}
- public static bool ValidateTrustChain(X509Certificate2[] trustPath, X509Certificate2[] attestationRootCertificates)
+ public static bool ValidateTrustChain(X509Certificate2[] trustPath, X509Certificate2[] attestationRootCertificates, FidoValidationMode validationMode = FidoValidationMode.Default)
{
// https://fidoalliance.org/specs/fido-v2.0-id-20180227/fido-metadata-statement-v2.0-id-20180227.html#widl-MetadataStatement-attestationRootCertificates
@@ -59,6 +59,8 @@ public static bool ValidateTrustChain(X509Certificate2[] trustPath, X509Certific
// A trust anchor can be a root certificate, an intermediate CA certificate or even the attestation certificate itself.
// Let's check the simplest case first. If subject and issuer are the same, and the attestation cert is in the list, that's all the validation we need
+
+ // We have the same singular root cert in trustpath and it is in attestationRootCertificates
if (trustPath.Length == 1 && trustPath[0].Subject.Equals(trustPath[0].Issuer, StringComparison.Ordinal))
{
foreach (X509Certificate2 cert in attestationRootCertificates)
@@ -68,7 +70,6 @@ public static bool ValidateTrustChain(X509Certificate2[] trustPath, X509Certific
return true;
}
}
- return false;
}
// If the attestation cert is not self signed, we will need to build a chain
@@ -101,7 +102,7 @@ public static bool ValidateTrustChain(X509Certificate2[] trustPath, X509Certific
chain.ChainPolicy.TrustMode = X509ChainTrustMode.CustomRootTrust;
// if the attestation cert has a CDP extension, go ahead and turn on online revocation checking
- if (!string.IsNullOrEmpty(CDPFromCertificateExts(trustPath[0].Extensions)))
+ if (!string.IsNullOrEmpty(CDPFromCertificateExts(trustPath[0].Extensions)) && validationMode != FidoValidationMode.FidoConformance2024)
chain.ChainPolicy.RevocationMode = X509RevocationMode.Online;
// don't allow unknown root now that we have a custom root
diff --git a/Src/Fido2/Fido2.cs b/Src/Fido2/Fido2.cs
index 11dc1b93..8ebf338b 100644
--- a/Src/Fido2/Fido2.cs
+++ b/Src/Fido2/Fido2.cs
@@ -65,16 +65,18 @@ public CredentialCreateOptions RequestNewCredential(
/// The attestation response from the authenticator.
/// The original options that was sent to the client.
/// The delegate used to validate that the CredentialID is unique to this user.
+ /// DO NOT USE - Deprecated, but kept in code due to conformance testing tool
/// The used to propagate notifications that the operation should be canceled.
///
public async Task MakeNewCredentialAsync(
AuthenticatorAttestationRawResponse attestationResponse,
CredentialCreateOptions originalOptions,
IsCredentialIdUniqueToUserAsyncDelegate isCredentialIdUniqueToUser,
+ byte[]? requestTokenBindingId = null,
CancellationToken cancellationToken = default)
{
var parsedResponse = AuthenticatorAttestationResponse.Parse(attestationResponse);
- var credential = await parsedResponse.VerifyAsync(originalOptions, _config, isCredentialIdUniqueToUser, _metadataService, cancellationToken);
+ var credential = await parsedResponse.VerifyAsync(originalOptions, _config, isCredentialIdUniqueToUser, _metadataService, requestTokenBindingId, cancellationToken);
return credential;
}
@@ -105,6 +107,7 @@ public AssertionOptions GetAssertionOptions(
/// The stored device public keys.
/// The stored value of the signature counter.
/// The delegate used to validate that the user handle is indeed owned of the CredentialId.
+ /// DO NOT USE - Deprecated, but kept in code due to conformance testing tool
/// The used to propagate notifications that the operation should be canceled.
///
public async Task MakeAssertionAsync(
@@ -114,6 +117,7 @@ public async Task MakeAssertionAsync(
IReadOnlyList storedDevicePublicKeys,
uint storedSignatureCounter,
IsUserHandleOwnerOfCredentialIdAsync isUserHandleOwnerOfCredentialIdCallback,
+ byte[]? requestTokenBindingId = null,
CancellationToken cancellationToken = default)
{
var parsedResponse = AuthenticatorAssertionResponse.Parse(assertionResponse);
@@ -125,6 +129,7 @@ public async Task MakeAssertionAsync(
storedSignatureCounter,
isUserHandleOwnerOfCredentialIdCallback,
_metadataService,
+ requestTokenBindingId,
cancellationToken);
return result;
diff --git a/Src/Fido2/FidoValidationMode.cs b/Src/Fido2/FidoValidationMode.cs
new file mode 100644
index 00000000..fbaf736f
--- /dev/null
+++ b/Src/Fido2/FidoValidationMode.cs
@@ -0,0 +1,6 @@
+public enum FidoValidationMode
+{
+ WebAuthNLevel3,
+ FidoConformance2024,
+ Default = WebAuthNLevel3
+}
diff --git a/Src/Fido2/IFido2.cs b/Src/Fido2/IFido2.cs
index a536bbef..8b58de02 100644
--- a/Src/Fido2/IFido2.cs
+++ b/Src/Fido2/IFido2.cs
@@ -20,12 +20,14 @@ Task MakeAssertionAsync(
IReadOnlyList storedDevicePublicKeys,
uint storedSignatureCounter,
IsUserHandleOwnerOfCredentialIdAsync isUserHandleOwnerOfCredentialIdCallback,
+ byte[]? requestTokenBindingId = null,
CancellationToken cancellationToken = default);
Task MakeNewCredentialAsync(
AuthenticatorAttestationRawResponse attestationResponse,
CredentialCreateOptions originalOptions,
IsCredentialIdUniqueToUserAsyncDelegate isCredentialIdUniqueToUser,
+ byte[]? requestTokenBindingId = null,
CancellationToken cancellationToken = default);
CredentialCreateOptions RequestNewCredential(
diff --git a/Src/Fido2/TokenBindingDto.cs b/Src/Fido2/TokenBindingDto.cs
new file mode 100644
index 00000000..3651b301
--- /dev/null
+++ b/Src/Fido2/TokenBindingDto.cs
@@ -0,0 +1,39 @@
+using System.Text.Json.Serialization;
+namespace Fido2NetLib;
+
+public class TokenBindingDto
+{
+ ///
+ /// Either "present" or "supported". https://www.w3.org/TR/webauthn/#enumdef-tokenbindingstatus
+ /// supported: Indicates the client supports token binding, but it was not negotiated when communicating with the Relying Party.
+ /// present: Indicates token binding was used when communicating with the Relying Party. In this case, the id member MUST be present
+ ///
+ [JsonPropertyName("status")]
+ public string? Status { get; set; }
+
+ ///
+ /// This member MUST be present if status is present, and MUST a base64url encoding of the Token Binding ID that was used when communicating with the Relying Party.
+ ///
+ [JsonPropertyName("id")]
+ public string? Id { get; set; }
+
+ public void Verify(byte[]? requestTokenbinding)
+ {
+ // validation by the FIDO conformance tool (more than spec says)
+ switch (Status)
+ {
+ case "present":
+ if (string.IsNullOrEmpty(Id))
+ throw new Fido2VerificationException("TokenBinding status was present but Id is missing");
+ var b64 = Base64Url.Encode(requestTokenbinding);
+ if (Id != b64)
+ throw new Fido2VerificationException("Tokenbinding Id does not match");
+ break;
+ case "supported":
+ case "not-supported":
+ break;
+ default:
+ throw new Fido2VerificationException("Malformed tokenbinding status field");
+ }
+ }
+}
diff --git a/Src/Fido2/TrustAnchor.cs b/Src/Fido2/TrustAnchor.cs
index 6a02daed..cbdbcdb3 100644
--- a/Src/Fido2/TrustAnchor.cs
+++ b/Src/Fido2/TrustAnchor.cs
@@ -1,60 +1,65 @@
-using System;
-using System.Linq;
-using System.Security.Cryptography.X509Certificates;
-
-using Fido2NetLib.Exceptions;
-
-namespace Fido2NetLib;
-
-public static class TrustAnchor
-{
- public static void Verify(MetadataBLOBPayloadEntry? metadataEntry, X509Certificate2[] trustPath)
- {
- if (trustPath != null && metadataEntry?.MetadataStatement?.AttestationTypes is not null)
- {
- static bool ContainsAttestationType(MetadataBLOBPayloadEntry entry, MetadataAttestationType type)
- {
- return entry.MetadataStatement.AttestationTypes.Contains(type.ToEnumMemberValue());
- }
-
- // If the authenticator's metadata requires basic full attestation, build and verify the chain
- if (ContainsAttestationType(metadataEntry, MetadataAttestationType.ATTESTATION_BASIC_FULL) ||
- ContainsAttestationType(metadataEntry, MetadataAttestationType.ATTESTATION_PRIVACY_CA))
- {
- string[] certStrings = metadataEntry.MetadataStatement.AttestationRootCertificates;
- var attestationRootCertificates = new X509Certificate2[certStrings.Length];
-
- for (int i = 0; i < attestationRootCertificates.Length; i++)
- {
- attestationRootCertificates[i] = new X509Certificate2(Convert.FromBase64String(certStrings[i]));
- }
-
- if (!CryptoUtils.ValidateTrustChain(trustPath, attestationRootCertificates))
- {
- throw new Fido2VerificationException(Fido2ErrorMessages.InvalidCertificateChain);
- }
- }
-
- else if (ContainsAttestationType(metadataEntry, MetadataAttestationType.ATTESTATION_ANONCA))
- {
- // skip verification for Anonymization CA (AnonCA)
- }
- else // otherwise, ensure the certificate is self signed
- {
- var trustPath0 = trustPath[0];
-
- if (!string.Equals(trustPath0.Subject, trustPath0.Issuer, StringComparison.Ordinal))
- {
- // TODO: Improve this error message
- throw new Fido2VerificationException("Attestation with full attestation from authenticator that does not support full attestation");
- }
- }
-
- // TODO: Verify all MetadataAttestationTypes are correctly handled
-
- // [ ] ATTESTATION_ECDAA "ecdaa" | currently handled as self signed w/ no test coverage
- // [ ] ATTESTATION_ANONCA "anonca" | currently not verified w/ no test coverage
- // [ ] ATTESTATION_NONE "none" | currently handled as self signed w/ no test coverage
- }
- }
-}
+using System;
+using System.Linq;
+using System.Security.Cryptography.X509Certificates;
+
+using Fido2NetLib.Exceptions;
+
+namespace Fido2NetLib;
+
+public static class TrustAnchor
+{
+ public static void Verify(MetadataBLOBPayloadEntry? metadataEntry, X509Certificate2[] trustPath, FidoValidationMode validationMode = FidoValidationMode.Default)
+ {
+ if (trustPath != null && metadataEntry?.MetadataStatement?.AttestationTypes is not null)
+ {
+ static bool ContainsAttestationType(MetadataBLOBPayloadEntry entry, MetadataAttestationType type)
+ {
+ return entry.MetadataStatement.AttestationTypes.Contains(type.ToEnumMemberValue());
+ }
+
+ // If the authenticator's metadata requires basic full attestation, build and verify the chain
+ if (ContainsAttestationType(metadataEntry, MetadataAttestationType.ATTESTATION_BASIC_FULL) ||
+ ContainsAttestationType(metadataEntry, MetadataAttestationType.ATTESTATION_PRIVACY_CA))
+ {
+ string[] certStrings = metadataEntry.MetadataStatement.AttestationRootCertificates;
+ var attestationRootCertificates = new X509Certificate2[certStrings.Length];
+
+ for (int i = 0; i < attestationRootCertificates.Length; i++)
+ {
+ attestationRootCertificates[i] = new X509Certificate2(Convert.FromBase64String(certStrings[i]));
+ }
+
+ if (trustPath.Length > 1 && attestationRootCertificates.Any(c => string.Equals(c.Thumbprint, trustPath[^1].Thumbprint, StringComparison.Ordinal)))
+ {
+ throw new Fido2VerificationException(Fido2ErrorMessages.InvalidCertificateChain);
+ }
+
+ if (!CryptoUtils.ValidateTrustChain(trustPath, attestationRootCertificates, validationMode))
+ {
+ throw new Fido2VerificationException(Fido2ErrorMessages.InvalidCertificateChain);
+ }
+ }
+
+ else if (ContainsAttestationType(metadataEntry, MetadataAttestationType.ATTESTATION_ANONCA))
+ {
+ // skip verification for Anonymization CA (AnonCA)
+ }
+ else // otherwise, ensure the certificate is self signed
+ {
+ var trustPath0 = trustPath[0];
+
+ if (!string.Equals(trustPath0.Subject, trustPath0.Issuer, StringComparison.Ordinal))
+ {
+ // TODO: Improve this error message
+ throw new Fido2VerificationException("Attestation with full attestation from authenticator that does not support full attestation");
+ }
+ }
+
+ // TODO: Verify all MetadataAttestationTypes are correctly handled
+
+ // [ ] ATTESTATION_ECDAA "ecdaa" | currently handled as self signed w/ no test coverage
+ // [ ] ATTESTATION_ANONCA "anonca" | currently not verified w/ no test coverage
+ // [ ] ATTESTATION_NONE "none" | currently handled as self signed w/ no test coverage
+ }
+ }
+}
diff --git a/Test/AuthenticatorResponse.cs b/Test/AuthenticatorResponse.cs
index 50ff27d0..93b6a064 100644
--- a/Test/AuthenticatorResponse.cs
+++ b/Test/AuthenticatorResponse.cs
@@ -253,7 +253,7 @@ public void TestAuthenticatorAttestationRawResponse()
{
AppID = true,
Extensions = ["foo", "bar"],
- Example = "test",
+ Example = true,
UserVerificationMethod = new ulong[][]
{
new ulong[]
@@ -279,7 +279,7 @@ public void TestAuthenticatorAttestationRawResponse()
Assert.Equal(clientDataJson, rawResponse.Response.ClientDataJson);
Assert.True(rawResponse.ClientExtensionResults.AppID);
Assert.Equal(new string[] { "foo", "bar" }, rawResponse.ClientExtensionResults.Extensions);
- Assert.Equal("test", rawResponse.ClientExtensionResults.Example);
+ Assert.True(rawResponse.ClientExtensionResults.Example);
Assert.Equal((ulong)4, rawResponse.ClientExtensionResults.UserVerificationMethod[0][0]);
Assert.True(rawResponse.ClientExtensionResults.PRF.Enabled);
Assert.Equal(rawResponse.ClientExtensionResults.PRF.Results.First, [0xf1, 0xd0]);
@@ -1212,7 +1212,7 @@ public void TestAuthenticatorAssertionRawResponse()
{
AppID = true,
Extensions = ["foo", "bar"],
- Example = "test",
+ Example = true,
UserVerificationMethod = new ulong[][]
{
new ulong[]
@@ -1240,11 +1240,12 @@ public void TestAuthenticatorAssertionRawResponse()
Assert.Equal([0xf1, 0xd0], assertionResponse.Response.UserHandle);
Assert.True(assertionResponse.ClientExtensionResults.AppID);
Assert.Equal(new string[] { "foo", "bar" }, assertionResponse.ClientExtensionResults.Extensions);
- Assert.Equal("test", assertionResponse.ClientExtensionResults.Example);
+ Assert.True(assertionResponse.ClientExtensionResults.Example);
Assert.Equal((ulong)4, assertionResponse.ClientExtensionResults.UserVerificationMethod[0][0]);
Assert.True(assertionResponse.ClientExtensionResults.PRF.Enabled);
Assert.Equal([0xf1, 0xd0], assertionResponse.ClientExtensionResults.PRF.Results.First);
Assert.Equal([0xf1, 0xd0], assertionResponse.ClientExtensionResults.PRF.Results.Second);
+
}
[Fact]
@@ -1288,7 +1289,7 @@ public async Task TestAuthenticatorAssertionTypeNotPublicKey()
{
AppID = false,
Extensions = ["foo", "bar"],
- Example = "test",
+ Example = true,
UserVerificationMethod = new ulong[][]
{
new ulong[]
@@ -1356,7 +1357,7 @@ public async Task TestAuthenticatorAssertionIdMissing()
{
AppID = false,
Extensions = ["foo", "bar"],
- Example = "test",
+ Example = true,
UserVerificationMethod = new ulong[][]
{
new ulong[]
@@ -1425,7 +1426,7 @@ public async Task TestAuthenticatorAssertionRawIdMissing()
{
AppID = false,
Extensions = ["foo", "bar"],
- Example = "test",
+ Example = true,
UserVerificationMethod = new ulong[][]
{
new ulong[]
@@ -1494,7 +1495,7 @@ public async Task TestAuthenticatorAssertionUserHandleEmpty()
{
AppID = false,
Extensions = ["foo", "bar"],
- Example = "test",
+ Example = true,
UserVerificationMethod = new ulong[][]
{
new ulong[]
@@ -1563,7 +1564,7 @@ public async Task TestAuthenticatorAssertionUserHandleNotOwnerOfPublicKey()
{
AppID = false,
Extensions = ["foo", "bar"],
- Example = "test",
+ Example = true,
UserVerificationMethod = new ulong[][]
{
new ulong[]
@@ -1632,7 +1633,7 @@ public async Task TestAuthenticatorAssertionTypeNotWebAuthnGet()
{
AppID = false,
Extensions = ["foo", "bar"],
- Example = "test",
+ Example = true,
UserVerificationMethod = new ulong[][]
{
new ulong[]
@@ -1703,7 +1704,7 @@ public async Task TestAuthenticatorAssertionAppId()
{
AppID = true,
Extensions = ["foo", "bar"],
- Example = "test",
+ Example = true,
UserVerificationMethod = new ulong[][]
{
new ulong[]
@@ -1773,7 +1774,7 @@ public async Task TestAuthenticatorAssertionInvalidRpIdHash()
{
AppID = false,
Extensions = ["foo", "bar"],
- Example = "test",
+ Example = true,
UserVerificationMethod = new ulong[][]
{
new ulong[]
@@ -1844,7 +1845,7 @@ public async Task TestAuthenticatorAssertionUPRequirementNotMet()
{
AppID = false,
Extensions = ["foo", "bar"],
- Example = "test",
+ Example = true,
UserVerificationMethod = new ulong[][]
{
new ulong[]
@@ -1914,7 +1915,7 @@ public async Task TestAuthenticatorAssertionUVPolicyNotMet()
{
AppID = false,
Extensions = ["foo", "bar"],
- Example = "test",
+ Example = true,
UserVerificationMethod = new ulong[][]
{
new ulong[]
@@ -1982,7 +1983,7 @@ public async Task TestAuthenticatorAssertionBEPolicyRequired()
{
AppID = false,
Extensions = ["foo", "bar"],
- Example = "test",
+ Example = true,
UserVerificationMethod = new ulong[][]
{
new ulong[]
@@ -2051,7 +2052,7 @@ public async Task TestAuthenticatorAssertionBEPolicyDisallow()
{
AppID = false,
Extensions = ["foo", "bar"],
- Example = "test",
+ Example = true,
UserVerificationMethod = new ulong[][]
{
new ulong[]
@@ -2120,7 +2121,7 @@ public async Task TestAuthenticatorAssertionBSPolicyRequired()
{
AppID = false,
Extensions = ["foo", "bar"],
- Example = "test",
+ Example = true,
UserVerificationMethod = new ulong[][]
{
new ulong[]
@@ -2189,7 +2190,7 @@ public async Task TestAuthenticatorAssertionBSPolicyDisallow()
{
AppID = false,
Extensions = ["foo", "bar"],
- Example = "test",
+ Example = true,
UserVerificationMethod = new ulong[][]
{
new ulong[]
@@ -2259,7 +2260,7 @@ public async Task TestAuthenticatorAssertionStoredPublicKeyMissing()
{
AppID = false,
Extensions = ["foo", "bar"],
- Example = "test",
+ Example = true,
UserVerificationMethod = new ulong[][]
{
new ulong[]
@@ -2328,7 +2329,7 @@ public async Task TestAuthenticatorAssertionInvalidSignature()
{
AppID = false,
Extensions = ["foo", "bar"],
- Example = "test",
+ Example = true,
UserVerificationMethod = new ulong[][]
{
new ulong[]
@@ -2404,7 +2405,7 @@ public async Task TestAuthenticatorAssertionSignCountSignature()
{
AppID = false,
Extensions = ["foo", "bar"],
- Example = "test",
+ Example = true,
UserVerificationMethod = new ulong[][]
{
new ulong[]
diff --git a/Test/Fido2Tests.cs b/Test/Fido2Tests.cs
index a6b578d7..db69354c 100644
--- a/Test/Fido2Tests.cs
+++ b/Test/Fido2Tests.cs
@@ -16,7 +16,7 @@
using Moq;
using NSec.Cryptography;
-
+using Test;
using static Fido2NetLib.AuthenticatorAttestationResponse;
namespace fido2_net_lib.Test;
@@ -76,6 +76,17 @@ static Fido2Tests()
];
}
+ private TestMetadataService CreateMetadataService(string cacheDir)
+ {
+ var repos = new List
+ {
+ new FileSystemMetadataRepository(cacheDir)
+ };
+ var simpleService = new TestMetadataService(repos);
+ simpleService.InitializeAsync().Wait();
+ return simpleService;
+ }
+
private async Task GetAsync(string filename)
{
return JsonSerializer.Deserialize(await File.ReadAllTextAsync(filename));
@@ -168,7 +179,7 @@ public async Task MakeAttestationResponseAsync()
{
AppID = true,
Extensions = ["foo", "bar"],
- Example = "test",
+ Example = true,
UserVerificationMethod = new ulong[][]
{
new ulong[]
@@ -411,7 +422,7 @@ public async Task TestFido2AssertionAsync()
var response = JsonSerializer.Deserialize(await File.ReadAllTextAsync("./attestationNoneResponse.json"));
var o = AuthenticatorAttestationResponse.Parse(response);
- await o.VerifyAsync(options, _config, (x, cancellationToken) => Task.FromResult(true), _metadataService, CancellationToken.None);
+ await o.VerifyAsync(options, _config, (x, cancellationToken) => Task.FromResult(true), _metadataService, null, CancellationToken.None);
var credId = "F1-3C-7F-08-3C-A2-29-E0-B4-03-E8-87-34-6E-FC-7F-98-53-10-3A-30-91-75-67-39-7A-D1-D8-AF-87-04-61-87-EF-95-31-85-60-F3-5A-1A-2A-CF-7D-B0-1D-06-B9-69-F9-AB-F4-EC-F3-07-3E-CF-0F-71-E8-84-E8-41-20";
var allowedCreds = new List() {
@@ -480,7 +491,7 @@ public async Task TestParsingAsync()
Assert.NotNull(jsonPost);
var o = AuthenticatorAttestationResponse.Parse(jsonPost);
- await o.VerifyAsync(options, _config, (x, cancellationToken) => Task.FromResult(true), _metadataService, CancellationToken.None);
+ await o.VerifyAsync(options, _config, (x, cancellationToken) => Task.FromResult(true), _metadataService, null, CancellationToken.None);
}
[Fact]
@@ -531,7 +542,60 @@ public async Task TestU2FAttestationAsync()
var jsonPost = JsonSerializer.Deserialize(await File.ReadAllTextAsync("./attestationResultsU2F.json"));
var options = JsonSerializer.Deserialize(await File.ReadAllTextAsync("./attestationOptionsU2F.json"));
var o = AuthenticatorAttestationResponse.Parse(jsonPost);
- await o.VerifyAsync(options, _config, (x, cancellationToken) => Task.FromResult(true), _metadataService, CancellationToken.None);
+ await o.VerifyAsync(options, _config, (x, cancellationToken) => Task.FromResult(true), _metadataService, null, CancellationToken.None);
+ }
+
+ [Fact]
+ public async Task TestPackedttestationAsyncFailTrustAnchorOnRootCertInTrustPath()
+ {
+ if (!OperatingSystem.IsWindows())
+ return;
+
+ var targetGuid = new Guid("42383245-4437-3343-3846-423445354132");
+ var metadataService = CreateMetadataService("./metadata");
+ metadataService.ChangeEntryGuid(new Guid("00000000-0000-0000-0000-000000000004"), targetGuid);
+ var jsonPost = JsonSerializer.Deserialize(await File.ReadAllTextAsync("./attestationResultsPacked.json"));
+ var options = JsonSerializer.Deserialize(await File.ReadAllTextAsync("./attestationOptionsPacked.json"));
+ var o = AuthenticatorAttestationResponse.Parse(jsonPost);
+ CborArray X5c = o.AttestationObject.AttStmt["x5c"] as CborArray;
+ var entry = await metadataService.GetEntryAsync(targetGuid);
+ foreach (var attRootCert in entry.MetadataStatement.AttestationRootCertificates)
+ X5c.Add(Encoding.UTF8.GetBytes(attRootCert));
+
+ await Assert.ThrowsAsync(() => o.VerifyAsync(options, _config, (x, cancellationToken) => Task.FromResult(true), metadataService, null, CancellationToken.None));
+ }
+
+ [Fact]
+ public async Task TestU2FAttestationAsyncFailTrustAnchorBasicFull()
+ {
+ var metadataService = CreateMetadataService("./metadata");
+ metadataService.ChangeEntryGuid(new Guid("00000000-0000-0000-0000-000000000001"), new Guid("00000000-0000-0000-0000-000000000000"));
+ var jsonPost = JsonSerializer.Deserialize(await File.ReadAllTextAsync("./attestationResultsU2F.json"));
+ var options = JsonSerializer.Deserialize(await File.ReadAllTextAsync("./attestationOptionsU2F.json"));
+ var o = AuthenticatorAttestationResponse.Parse(jsonPost);
+ await Assert.ThrowsAsync(() => o.VerifyAsync(options, _config, (x, cancellationToken) => Task.FromResult(true), metadataService, null, CancellationToken.None));
+ }
+
+ [Fact]
+ public async Task TestU2FAttestationAsyncCantFailTrustAnchorAnonca()
+ {
+ var metadataService = CreateMetadataService("./metadata");
+ metadataService.ChangeEntryGuid(new Guid("00000000-0000-0000-0000-000000000002"), new Guid("00000000-0000-0000-0000-000000000000"));
+ var jsonPost = JsonSerializer.Deserialize(await File.ReadAllTextAsync("./attestationResultsU2F.json"));
+ var options = JsonSerializer.Deserialize(await File.ReadAllTextAsync("./attestationOptionsU2F.json"));
+ var o = AuthenticatorAttestationResponse.Parse(jsonPost);
+ await o.VerifyAsync(options, _config, (x, cancellationToken) => Task.FromResult(true), metadataService, null, CancellationToken.None);
+ }
+
+ [Fact]
+ public async Task TestU2FAttestationAsyncFailTrustAnchorBasicSurrogate()
+ {
+ var metadataService = CreateMetadataService("./metadata");
+ metadataService.ChangeEntryGuid(new Guid("00000000-0000-0000-0000-000000000003"), new Guid("00000000-0000-0000-0000-000000000000"));
+ var jsonPost = JsonSerializer.Deserialize(await File.ReadAllTextAsync("./attestationResultsU2F.json"));
+ var options = JsonSerializer.Deserialize(await File.ReadAllTextAsync("./attestationOptionsU2F.json"));
+ var o = AuthenticatorAttestationResponse.Parse(jsonPost);
+ await Assert.ThrowsAsync(() => o.VerifyAsync(options, _config, (x, cancellationToken) => Task.FromResult(true), metadataService, null, CancellationToken.None));
}
[Fact]
@@ -541,7 +605,7 @@ public async Task TestPackedAttestationAsync()
var options = JsonSerializer.Deserialize(await File.ReadAllTextAsync("./attestationOptionsPacked.json"));
var o = AuthenticatorAttestationResponse.Parse(jsonPost);
options.PubKeyCredParams.Add(new PubKeyCredParam(COSE.Algorithm.RS1, PublicKeyCredentialType.PublicKey));
- await o.VerifyAsync(options, _config, (x, cancellationToken) => Task.FromResult(true), _metadataService, CancellationToken.None);
+ await o.VerifyAsync(options, _config, (x, cancellationToken) => Task.FromResult(true), _metadataService, null, CancellationToken.None);
var authData = o.AttestationObject.AuthData;
var acdBytes = authData.AttestedCredentialData.ToByteArray();
var acd = AttestedCredentialData.Parse(acdBytes);
@@ -555,7 +619,7 @@ public async Task TestNoneAttestationAsync()
var options = JsonSerializer.Deserialize(await File.ReadAllTextAsync("./attestationOptionsNone.json"));
var o = AuthenticatorAttestationResponse.Parse(jsonPost);
- await o.VerifyAsync(options, _config, (x, cancellationToken) => Task.FromResult(true), _metadataService, CancellationToken.None);
+ await o.VerifyAsync(options, _config, (x, cancellationToken) => Task.FromResult(true), _metadataService, null, CancellationToken.None);
}
[Fact]
@@ -564,7 +628,7 @@ public async Task TestTPMSHA256AttestationAsync()
var jsonPost = JsonSerializer.Deserialize(await File.ReadAllTextAsync("./attestationTPMSHA256Response.json"));
var options = JsonSerializer.Deserialize(await File.ReadAllTextAsync("./attestationTPMSHA256Options.json"));
var o = AuthenticatorAttestationResponse.Parse(jsonPost);
- await o.VerifyAsync(options, _config, (x, cancellationToken) => Task.FromResult(true), _metadataService, CancellationToken.None);
+ await o.VerifyAsync(options, _config, (x, cancellationToken) => Task.FromResult(true), _metadataService, null, CancellationToken.None);
}
[Fact]
@@ -574,7 +638,7 @@ public async Task TestTPMSHA1AttestationAsync()
var options = JsonSerializer.Deserialize(await File.ReadAllTextAsync("./attestationTPMSHA1Options.json"));
var o = AuthenticatorAttestationResponse.Parse(jsonPost);
options.PubKeyCredParams.Add(new PubKeyCredParam(COSE.Algorithm.RS1, PublicKeyCredentialType.PublicKey));
- await o.VerifyAsync(options, _config, (x, cancellationToken) => Task.FromResult(true), _metadataService, CancellationToken.None);
+ await o.VerifyAsync(options, _config, (x, cancellationToken) => Task.FromResult(true), _metadataService, null, CancellationToken.None);
}
[Fact]
@@ -583,7 +647,7 @@ public async Task TestAndroidKeyAttestationAsync()
var jsonPost = JsonSerializer.Deserialize(await File.ReadAllTextAsync("./attestationAndroidKeyResponse.json"));
var options = JsonSerializer.Deserialize(await File.ReadAllTextAsync("./attestationAndroidKeyOptions.json"));
var o = AuthenticatorAttestationResponse.Parse(jsonPost);
- await o.VerifyAsync(options, _config, (x, cancellationToken) => Task.FromResult(true), _metadataService, CancellationToken.None);
+ await o.VerifyAsync(options, _config, (x, cancellationToken) => Task.FromResult(true), _metadataService, null, CancellationToken.None);
}
[Fact]
@@ -592,7 +656,7 @@ public async Task TaskPackedAttestation512()
var jsonPost = JsonSerializer.Deserialize(await File.ReadAllTextAsync("./attestationResultsPacked512.json"));
var options = JsonSerializer.Deserialize(await File.ReadAllTextAsync("./attestationOptionsPacked512.json"));
var o = AuthenticatorAttestationResponse.Parse(jsonPost);
- await o.VerifyAsync(options, _config, (x, cancellationToken) => Task.FromResult(true), _metadataService, CancellationToken.None);
+ await o.VerifyAsync(options, _config, (x, cancellationToken) => Task.FromResult(true), _metadataService, null, CancellationToken.None);
}
[Fact]
@@ -601,7 +665,7 @@ public async Task TestTrustKeyAttestationAsync()
var jsonPost = JsonSerializer.Deserialize(await File.ReadAllTextAsync("./attestationResultTrustKeyT110.json"));
var options = JsonSerializer.Deserialize(await File.ReadAllTextAsync("./attestationOptionsTrustKeyT110.json"));
var o = AuthenticatorAttestationResponse.Parse(jsonPost);
- await o.VerifyAsync(options, _config, (x, cancellationToken) => Task.FromResult(true), _metadataService, CancellationToken.None);
+ await o.VerifyAsync(options, _config, (x, cancellationToken) => Task.FromResult(true), _metadataService, null, CancellationToken.None);
var authData = o.AttestationObject.AuthData;
var acdBytes = authData.AttestedCredentialData.ToByteArray();
var acd = AttestedCredentialData.Parse(acdBytes);
@@ -618,7 +682,7 @@ public async Task TestInvalidU2FAttestationAsync()
var jsonPost = JsonSerializer.Deserialize(await File.ReadAllTextAsync("./attestationResultsATKey.json"));
var options = JsonSerializer.Deserialize(await File.ReadAllTextAsync("./attestationOptionsATKey.json"));
var o = AuthenticatorAttestationResponse.Parse(jsonPost);
- await o.VerifyAsync(options, _config, (x, cancellationToken) => Task.FromResult(true), _metadataService, CancellationToken.None);
+ await o.VerifyAsync(options, _config, (x, cancellationToken) => Task.FromResult(true), _metadataService, null, CancellationToken.None);
var authData = o.AttestationObject.AuthData;
var acdBytes = authData.AttestedCredentialData.ToByteArray();
var acd = AttestedCredentialData.Parse(acdBytes);
@@ -643,7 +707,7 @@ public async Task TestMdsStatusReportsSuccessAsync()
mockMetadataService.Setup(m => m.ConformanceTesting()).Returns(false);
var o = AuthenticatorAttestationResponse.Parse(response);
- await o.VerifyAsync(options, _config, (x, cancellationToken) => Task.FromResult(true), mockMetadataService.Object, CancellationToken.None);
+ await o.VerifyAsync(options, _config, (x, cancellationToken) => Task.FromResult(true), mockMetadataService.Object, null, CancellationToken.None);
}
[Fact]
@@ -666,7 +730,7 @@ public async Task TestMdsStatusReportsUndesiredAsync()
var o = AuthenticatorAttestationResponse.Parse(response);
await Assert.ThrowsAsync(() =>
- o.VerifyAsync(options, _config, (x, cancellationToken) => Task.FromResult(true), mockMetadataService.Object, CancellationToken.None));
+ o.VerifyAsync(options, _config, (x, cancellationToken) => Task.FromResult(true), mockMetadataService.Object, null, CancellationToken.None));
}
[Fact]
@@ -689,7 +753,7 @@ public async Task TestMdsStatusReportsUndesiredFixedAsync()
mockMetadataService.Setup(m => m.ConformanceTesting()).Returns(false);
var o = AuthenticatorAttestationResponse.Parse(response);
- await o.VerifyAsync(options, _config, (x, cancellationToken) => Task.FromResult(true), mockMetadataService.Object, CancellationToken.None);
+ await o.VerifyAsync(options, _config, (x, cancellationToken) => Task.FromResult(true), mockMetadataService.Object, null, CancellationToken.None);
}
[Fact]
@@ -703,7 +767,7 @@ public async Task TestMdsStatusReportsNullAsync()
mockMetadataService.Setup(m => m.ConformanceTesting()).Returns(false);
var o = AuthenticatorAttestationResponse.Parse(response);
- await o.VerifyAsync(options, _config, (x, cancellationToken) => Task.FromResult(true), mockMetadataService.Object, CancellationToken.None);
+ await o.VerifyAsync(options, _config, (x, cancellationToken) => Task.FromResult(true), mockMetadataService.Object, null, CancellationToken.None);
}
//public void TestHasCorrentAAguid()
diff --git a/Test/Test.csproj b/Test/Test.csproj
index 5fb30e72..12faf566 100644
--- a/Test/Test.csproj
+++ b/Test/Test.csproj
@@ -17,6 +17,7 @@
+
diff --git a/Test/TestFiles/metadata/256K1 U2F Authenticator anonca.json b/Test/TestFiles/metadata/256K1 U2F Authenticator anonca.json
new file mode 100644
index 00000000..e844de94
--- /dev/null
+++ b/Test/TestFiles/metadata/256K1 U2F Authenticator anonca.json
@@ -0,0 +1,68 @@
+{
+ "legalHeader": "https://fidoalliance.org/metadata/metadata-statement-legal-header/",
+ "description": "Virtual Secp256K1 U2F authenticator",
+ "aaguid": "00000000-0000-0000-0000-000000000002",
+ "alternativeDescriptions": {
+ "en-GB": "Virtual Secp256K1 U2F authenticator"
+ },
+ "attestationCertificateKeyIdentifiers": [
+ "564df7c0f8c655b6a11f6c4d19f3bf41e2fd0179"
+ ],
+ "protocolFamily": "u2f",
+ "schema": 3,
+ "authenticatorVersion": 2,
+ "upv": [
+ {
+ "major": 1,
+ "minor": 0
+ },
+ {
+ "major": 1,
+ "minor": 1
+ },
+ {
+ "major": 1,
+ "minor": 2
+ }
+ ],
+ "authenticationAlgorithms": [
+ "secp256r1_ecdsa_sha256_raw"
+ ],
+ "publicKeyAlgAndEncodings": [
+ "ecc_x962_raw"
+ ],
+ "attestationTypes": [
+ "anonca"
+ ],
+ "userVerificationDetails": [
+ [
+ {
+ "userVerificationMethod": "none"
+ }
+ ],
+ [
+ {
+ "userVerificationMethod": "presence_internal"
+ }
+ ]
+ ],
+ "keyProtection": [
+ "hardware",
+ "secure_element"
+ ],
+ "matcherProtection": [
+ "on_chip"
+ ],
+ "cryptoStrength": 128,
+ "attachmentHint": [
+ "external",
+ "wired",
+ "nfc",
+ "wireless"
+ ],
+ "tcDisplay": [],
+ "attestationRootCertificates": [
+ "MIIFwDCCA6gCCQCNm1u56oRwXTANBgkqhkiG9w0BAQsFADCBoTEYMBYGA1UEAwwPRklETzIgVEVTVCBST09UMTEwLwYJKoZIhvcNAQkBFiJjb25mb3JtYW5jZS10b29sc0BmaWRvYWxsaWFuY2Uub3JnMRYwFAYDVQQKDA1GSURPIEFsbGlhbmNlMQwwCgYDVQQLDANDV0cxCzAJBgNVBAYTAlVTMQswCQYDVQQIDAJNWTESMBAGA1UEBwwJV2FrZWZpZWxkMB4XDTE4MDMxNjE0MzUyN1oXDTQ1MDgwMTE0MzUyN1owgaExGDAWBgNVBAMMD0ZJRE8yIFRFU1QgUk9PVDExMC8GCSqGSIb3DQEJARYiY29uZm9ybWFuY2UtdG9vbHNAZmlkb2FsbGlhbmNlLm9yZzEWMBQGA1UECgwNRklETyBBbGxpYW5jZTEMMAoGA1UECwwDQ1dHMQswCQYDVQQGEwJVUzELMAkGA1UECAwCTVkxEjAQBgNVBAcMCVdha2VmaWVsZDCCAiIwDQYJKoZIhvcNAQEBBQADggIPADCCAgoCggIBAL11U5yAIVLMrL3xS8u8ysMSdOkDeoTO+RcAy+uXXp6k4SC+jOy37gICEtYI+MKQV1EMeMMf3rM1ueZAO3iPFa0NEdi/oQ7npnGjBNI8wMzD8FfNe6rWtzkDaHpsZW///MwWDpGyJR+Xyjcq6U4vS9bS6zZ7jslw0Oczx4UsYgOsIUXSSBaGOrRbxJ/JC5gnDYEYvtNM+PDPczLNKAyhdvBZWNWHr7MZ0P5TeJQcXsAoShRX2Y8U8fRNJm7SeiFKDP0Nn/QKxOSt7zGP4xt9nMasE1q2ZTdar2+W13CRz37RI0ZWpq/+YquoEbZ7Uj7NmBTcqhb260nmDER2FpwwYwPSark92IZbamozB8d7OEI1jJgsrjJhKan0EmRaWVBpHT4xYKdEu7r09S0JhKyU+52WDmmVQTMpYLrm4Xl7hRxyPyBYkalrozsGmPs8vlhNq3VsVbyBSMSpEmUaeAa7LLE9/Vh0agJLVFHh1ehYKJpzHnmmBXUqx0Fz3afmDm1NX0sr3O/6xIx1VSTViT3KNxBYpVH1qjHATLzuxcWmm+75fcJMiPYPSMXVmRb3Q1l91AM4BBeWhlP3Fbc7gDy0r+s7m0sGS6PT2J2rGog2rUxnJ+zCM11M7DeO0XM2nny4uRYPPk9w2EXzfvtdvieYU/5RB4RDm5TGxHhGXVZUgac5AgMBAAEwDQYJKoZIhvcNAQELBQADggIBAFt2XGd3k5GpbO1EUm3u60zT1fE6u6pOscp156k5VnsHgaHRHdIAPNLeLNmR7y5OnrXbh13CrGwU1q84jjJXpv+v14xUCc5i01yopFTQFLr4A7NHp2nNYfNhhIVSFAgW43EflJflbLEelCJzxLlWb5BoDsZeeNmEQsXIM1mJ26R3r0dzsHBb0uy+8LNR1gdVqdjhC8BLy3gh4+BWuidyZNt07LveDsSFW5rcj5wRrSx9hXPIyVpjQSljNvY7MVTouqJzNAAQMsTKkXPkTXldCop9Qo9UPkHRRm0l7LLtdaOoXrct0Ymocf8zxf9bFNiw9f4WRYQM6sMhzt8+s/oDilo4QhcUgeJEiEPESi6ynYTV62SHA4eMunUJ5dlCaRnFiR9DTImFa5IRzie326/nW/SPCaKc/yrFIihMMjJoSAPhpTb/K6yHOUG8r+KiQut7NzqGV301pQ9u62dGL5Oi1VXmCFlE2ramZs15BNOUyAo2CBbRJg3jKcdu/8QC6ojjDvQ863+7LPtn74wJC5RpUJsS0GhQWgq5pAXO3wA61Uobxi6MkOpCC0zBWx/d4CqpS4j4hFgxWBTXX48ihPu+hIxIF/AxbqtPvqLMExW/xZITn6ArpWyQ9e4SUVr3n3F33ap1XdDyZ0vwFcm18JQAtsvXT6qCLrWOXnHUgfn/+Viu"
+ ],
+ "icon": ""
+}
\ No newline at end of file
diff --git a/Test/TestFiles/metadata/256K1 U2F Authenticator basic_full.json b/Test/TestFiles/metadata/256K1 U2F Authenticator basic_full.json
new file mode 100644
index 00000000..c09f42b7
--- /dev/null
+++ b/Test/TestFiles/metadata/256K1 U2F Authenticator basic_full.json
@@ -0,0 +1,68 @@
+{
+ "legalHeader": "https://fidoalliance.org/metadata/metadata-statement-legal-header/",
+ "description": "Virtual Secp256K1 U2F authenticator",
+ "aaguid": "00000000-0000-0000-0000-000000000001",
+ "alternativeDescriptions": {
+ "en-GB": "Virtual Secp256K1 U2F authenticator"
+ },
+ "attestationCertificateKeyIdentifiers": [
+ "564df7c0f8c655b6a11f6c4d19f3bf41e2fd0179"
+ ],
+ "protocolFamily": "u2f",
+ "schema": 3,
+ "authenticatorVersion": 2,
+ "upv": [
+ {
+ "major": 1,
+ "minor": 0
+ },
+ {
+ "major": 1,
+ "minor": 1
+ },
+ {
+ "major": 1,
+ "minor": 2
+ }
+ ],
+ "authenticationAlgorithms": [
+ "secp256r1_ecdsa_sha256_raw"
+ ],
+ "publicKeyAlgAndEncodings": [
+ "ecc_x962_raw"
+ ],
+ "attestationTypes": [
+ "basic_full"
+ ],
+ "userVerificationDetails": [
+ [
+ {
+ "userVerificationMethod": "none"
+ }
+ ],
+ [
+ {
+ "userVerificationMethod": "presence_internal"
+ }
+ ]
+ ],
+ "keyProtection": [
+ "hardware",
+ "secure_element"
+ ],
+ "matcherProtection": [
+ "on_chip"
+ ],
+ "cryptoStrength": 128,
+ "attachmentHint": [
+ "external",
+ "wired",
+ "nfc",
+ "wireless"
+ ],
+ "tcDisplay": [],
+ "attestationRootCertificates": [
+ "MIIFwDCCA6gCCQCNm1u56oRwXTANBgkqhkiG9w0BAQsFADCBoTEYMBYGA1UEAwwPRklETzIgVEVTVCBST09UMTEwLwYJKoZIhvcNAQkBFiJjb25mb3JtYW5jZS10b29sc0BmaWRvYWxsaWFuY2Uub3JnMRYwFAYDVQQKDA1GSURPIEFsbGlhbmNlMQwwCgYDVQQLDANDV0cxCzAJBgNVBAYTAlVTMQswCQYDVQQIDAJNWTESMBAGA1UEBwwJV2FrZWZpZWxkMB4XDTE4MDMxNjE0MzUyN1oXDTQ1MDgwMTE0MzUyN1owgaExGDAWBgNVBAMMD0ZJRE8yIFRFU1QgUk9PVDExMC8GCSqGSIb3DQEJARYiY29uZm9ybWFuY2UtdG9vbHNAZmlkb2FsbGlhbmNlLm9yZzEWMBQGA1UECgwNRklETyBBbGxpYW5jZTEMMAoGA1UECwwDQ1dHMQswCQYDVQQGEwJVUzELMAkGA1UECAwCTVkxEjAQBgNVBAcMCVdha2VmaWVsZDCCAiIwDQYJKoZIhvcNAQEBBQADggIPADCCAgoCggIBAL11U5yAIVLMrL3xS8u8ysMSdOkDeoTO+RcAy+uXXp6k4SC+jOy37gICEtYI+MKQV1EMeMMf3rM1ueZAO3iPFa0NEdi/oQ7npnGjBNI8wMzD8FfNe6rWtzkDaHpsZW///MwWDpGyJR+Xyjcq6U4vS9bS6zZ7jslw0Oczx4UsYgOsIUXSSBaGOrRbxJ/JC5gnDYEYvtNM+PDPczLNKAyhdvBZWNWHr7MZ0P5TeJQcXsAoShRX2Y8U8fRNJm7SeiFKDP0Nn/QKxOSt7zGP4xt9nMasE1q2ZTdar2+W13CRz37RI0ZWpq/+YquoEbZ7Uj7NmBTcqhb260nmDER2FpwwYwPSark92IZbamozB8d7OEI1jJgsrjJhKan0EmRaWVBpHT4xYKdEu7r09S0JhKyU+52WDmmVQTMpYLrm4Xl7hRxyPyBYkalrozsGmPs8vlhNq3VsVbyBSMSpEmUaeAa7LLE9/Vh0agJLVFHh1ehYKJpzHnmmBXUqx0Fz3afmDm1NX0sr3O/6xIx1VSTViT3KNxBYpVH1qjHATLzuxcWmm+75fcJMiPYPSMXVmRb3Q1l91AM4BBeWhlP3Fbc7gDy0r+s7m0sGS6PT2J2rGog2rUxnJ+zCM11M7DeO0XM2nny4uRYPPk9w2EXzfvtdvieYU/5RB4RDm5TGxHhGXVZUgac5AgMBAAEwDQYJKoZIhvcNAQELBQADggIBAFt2XGd3k5GpbO1EUm3u60zT1fE6u6pOscp156k5VnsHgaHRHdIAPNLeLNmR7y5OnrXbh13CrGwU1q84jjJXpv+v14xUCc5i01yopFTQFLr4A7NHp2nNYfNhhIVSFAgW43EflJflbLEelCJzxLlWb5BoDsZeeNmEQsXIM1mJ26R3r0dzsHBb0uy+8LNR1gdVqdjhC8BLy3gh4+BWuidyZNt07LveDsSFW5rcj5wRrSx9hXPIyVpjQSljNvY7MVTouqJzNAAQMsTKkXPkTXldCop9Qo9UPkHRRm0l7LLtdaOoXrct0Ymocf8zxf9bFNiw9f4WRYQM6sMhzt8+s/oDilo4QhcUgeJEiEPESi6ynYTV62SHA4eMunUJ5dlCaRnFiR9DTImFa5IRzie326/nW/SPCaKc/yrFIihMMjJoSAPhpTb/K6yHOUG8r+KiQut7NzqGV301pQ9u62dGL5Oi1VXmCFlE2ramZs15BNOUyAo2CBbRJg3jKcdu/8QC6ojjDvQ863+7LPtn74wJC5RpUJsS0GhQWgq5pAXO3wA61Uobxi6MkOpCC0zBWx/d4CqpS4j4hFgxWBTXX48ihPu+hIxIF/AxbqtPvqLMExW/xZITn6ArpWyQ9e4SUVr3n3F33ap1XdDyZ0vwFcm18JQAtsvXT6qCLrWOXnHUgfn/+Viu"
+ ],
+ "icon": ""
+}
\ No newline at end of file
diff --git a/Test/TestFiles/metadata/256K1 U2F Authenticator basic_surrogate.json b/Test/TestFiles/metadata/256K1 U2F Authenticator basic_surrogate.json
new file mode 100644
index 00000000..c2c55c06
--- /dev/null
+++ b/Test/TestFiles/metadata/256K1 U2F Authenticator basic_surrogate.json
@@ -0,0 +1,68 @@
+{
+ "legalHeader": "https://fidoalliance.org/metadata/metadata-statement-legal-header/",
+ "description": "Virtual Secp256K1 U2F authenticator",
+ "aaguid": "00000000-0000-0000-0000-000000000003",
+ "alternativeDescriptions": {
+ "en-GB": "Virtual Secp256K1 U2F authenticator"
+ },
+ "attestationCertificateKeyIdentifiers": [
+ "564df7c0f8c655b6a11f6c4d19f3bf41e2fd0179"
+ ],
+ "protocolFamily": "u2f",
+ "schema": 3,
+ "authenticatorVersion": 2,
+ "upv": [
+ {
+ "major": 1,
+ "minor": 0
+ },
+ {
+ "major": 1,
+ "minor": 1
+ },
+ {
+ "major": 1,
+ "minor": 2
+ }
+ ],
+ "authenticationAlgorithms": [
+ "secp256r1_ecdsa_sha256_raw"
+ ],
+ "publicKeyAlgAndEncodings": [
+ "ecc_x962_raw"
+ ],
+ "attestationTypes": [
+ "basic_surrogate"
+ ],
+ "userVerificationDetails": [
+ [
+ {
+ "userVerificationMethod": "none"
+ }
+ ],
+ [
+ {
+ "userVerificationMethod": "presence_internal"
+ }
+ ]
+ ],
+ "keyProtection": [
+ "hardware",
+ "secure_element"
+ ],
+ "matcherProtection": [
+ "on_chip"
+ ],
+ "cryptoStrength": 128,
+ "attachmentHint": [
+ "external",
+ "wired",
+ "nfc",
+ "wireless"
+ ],
+ "tcDisplay": [],
+ "attestationRootCertificates": [
+ "MIIFwDCCA6gCCQCNm1u56oRwXTANBgkqhkiG9w0BAQsFADCBoTEYMBYGA1UEAwwPRklETzIgVEVTVCBST09UMTEwLwYJKoZIhvcNAQkBFiJjb25mb3JtYW5jZS10b29sc0BmaWRvYWxsaWFuY2Uub3JnMRYwFAYDVQQKDA1GSURPIEFsbGlhbmNlMQwwCgYDVQQLDANDV0cxCzAJBgNVBAYTAlVTMQswCQYDVQQIDAJNWTESMBAGA1UEBwwJV2FrZWZpZWxkMB4XDTE4MDMxNjE0MzUyN1oXDTQ1MDgwMTE0MzUyN1owgaExGDAWBgNVBAMMD0ZJRE8yIFRFU1QgUk9PVDExMC8GCSqGSIb3DQEJARYiY29uZm9ybWFuY2UtdG9vbHNAZmlkb2FsbGlhbmNlLm9yZzEWMBQGA1UECgwNRklETyBBbGxpYW5jZTEMMAoGA1UECwwDQ1dHMQswCQYDVQQGEwJVUzELMAkGA1UECAwCTVkxEjAQBgNVBAcMCVdha2VmaWVsZDCCAiIwDQYJKoZIhvcNAQEBBQADggIPADCCAgoCggIBAL11U5yAIVLMrL3xS8u8ysMSdOkDeoTO+RcAy+uXXp6k4SC+jOy37gICEtYI+MKQV1EMeMMf3rM1ueZAO3iPFa0NEdi/oQ7npnGjBNI8wMzD8FfNe6rWtzkDaHpsZW///MwWDpGyJR+Xyjcq6U4vS9bS6zZ7jslw0Oczx4UsYgOsIUXSSBaGOrRbxJ/JC5gnDYEYvtNM+PDPczLNKAyhdvBZWNWHr7MZ0P5TeJQcXsAoShRX2Y8U8fRNJm7SeiFKDP0Nn/QKxOSt7zGP4xt9nMasE1q2ZTdar2+W13CRz37RI0ZWpq/+YquoEbZ7Uj7NmBTcqhb260nmDER2FpwwYwPSark92IZbamozB8d7OEI1jJgsrjJhKan0EmRaWVBpHT4xYKdEu7r09S0JhKyU+52WDmmVQTMpYLrm4Xl7hRxyPyBYkalrozsGmPs8vlhNq3VsVbyBSMSpEmUaeAa7LLE9/Vh0agJLVFHh1ehYKJpzHnmmBXUqx0Fz3afmDm1NX0sr3O/6xIx1VSTViT3KNxBYpVH1qjHATLzuxcWmm+75fcJMiPYPSMXVmRb3Q1l91AM4BBeWhlP3Fbc7gDy0r+s7m0sGS6PT2J2rGog2rUxnJ+zCM11M7DeO0XM2nny4uRYPPk9w2EXzfvtdvieYU/5RB4RDm5TGxHhGXVZUgac5AgMBAAEwDQYJKoZIhvcNAQELBQADggIBAFt2XGd3k5GpbO1EUm3u60zT1fE6u6pOscp156k5VnsHgaHRHdIAPNLeLNmR7y5OnrXbh13CrGwU1q84jjJXpv+v14xUCc5i01yopFTQFLr4A7NHp2nNYfNhhIVSFAgW43EflJflbLEelCJzxLlWb5BoDsZeeNmEQsXIM1mJ26R3r0dzsHBb0uy+8LNR1gdVqdjhC8BLy3gh4+BWuidyZNt07LveDsSFW5rcj5wRrSx9hXPIyVpjQSljNvY7MVTouqJzNAAQMsTKkXPkTXldCop9Qo9UPkHRRm0l7LLtdaOoXrct0Ymocf8zxf9bFNiw9f4WRYQM6sMhzt8+s/oDilo4QhcUgeJEiEPESi6ynYTV62SHA4eMunUJ5dlCaRnFiR9DTImFa5IRzie326/nW/SPCaKc/yrFIihMMjJoSAPhpTb/K6yHOUG8r+KiQut7NzqGV301pQ9u62dGL5Oi1VXmCFlE2ramZs15BNOUyAo2CBbRJg3jKcdu/8QC6ojjDvQ863+7LPtn74wJC5RpUJsS0GhQWgq5pAXO3wA61Uobxi6MkOpCC0zBWx/d4CqpS4j4hFgxWBTXX48ihPu+hIxIF/AxbqtPvqLMExW/xZITn6ArpWyQ9e4SUVr3n3F33ap1XdDyZ0vwFcm18JQAtsvXT6qCLrWOXnHUgfn/+Viu"
+ ],
+ "icon": ""
+}
\ No newline at end of file
diff --git a/Test/TestFiles/metadata/Secp256R1 Packed Authenticator.json b/Test/TestFiles/metadata/Secp256R1 Packed Authenticator.json
new file mode 100644
index 00000000..db5b0b68
--- /dev/null
+++ b/Test/TestFiles/metadata/Secp256R1 Packed Authenticator.json
@@ -0,0 +1,128 @@
+{
+ "legalHeader": "https://fidoalliance.org/metadata/metadata-statement-legal-header/",
+ "description": "Virtual Packed authenticator",
+ "aaguid": "00000000-0000-0000-0000-000000000004",
+ "alternativeDescriptions": {
+ "en-GB": "Virtual Packed authenticator"
+ },
+ "protocolFamily": "fido2",
+ "authenticatorVersion": 2,
+ "upv": [
+ {
+ "major": 1,
+ "minor": 0
+ }
+ ],
+ "authenticationAlgorithms": [
+ "secp256r1_ecdsa_sha256_raw"
+ ],
+ "publicKeyAlgAndEncodings": [
+ "cose"
+ ],
+ "attestationTypes": [
+ "basic_full",
+ "basic_surrogate"
+ ],
+ "schema": 3,
+ "userVerificationDetails": [
+ [
+ {
+ "userVerificationMethod": "none"
+ }
+ ],
+ [
+ {
+ "userVerificationMethod": "presence_internal"
+ }
+ ],
+ [
+ {
+ "userVerificationMethod": "passcode_external",
+ "caDesc": {
+ "base": 10,
+ "minLength": 4
+ }
+ }
+ ],
+ [
+ {
+ "userVerificationMethod": "passcode_external",
+ "caDesc": {
+ "base": 10,
+ "minLength": 4
+ }
+ },
+ {
+ "userVerificationMethod": "presence_internal"
+ }
+ ]
+ ],
+ "keyProtection": [
+ "hardware",
+ "secure_element"
+ ],
+ "matcherProtection": [
+ "on_chip"
+ ],
+ "cryptoStrength": 128,
+ "attachmentHint": [
+ "external",
+ "wired",
+ "wireless",
+ "nfc"
+ ],
+ "tcDisplay": [],
+ "attestationRootCertificates": [
+ "MIIFwDCCA6gCCQCNm1u56oRwXTANBgkqhkiG9w0BAQsFADCBoTEYMBYGA1UEAwwPRklETzIgVEVTVCBST09UMTEwLwYJKoZIhvcNAQkBFiJjb25mb3JtYW5jZS10b29sc0BmaWRvYWxsaWFuY2Uub3JnMRYwFAYDVQQKDA1GSURPIEFsbGlhbmNlMQwwCgYDVQQLDANDV0cxCzAJBgNVBAYTAlVTMQswCQYDVQQIDAJNWTESMBAGA1UEBwwJV2FrZWZpZWxkMB4XDTE4MDMxNjE0MzUyN1oXDTQ1MDgwMTE0MzUyN1owgaExGDAWBgNVBAMMD0ZJRE8yIFRFU1QgUk9PVDExMC8GCSqGSIb3DQEJARYiY29uZm9ybWFuY2UtdG9vbHNAZmlkb2FsbGlhbmNlLm9yZzEWMBQGA1UECgwNRklETyBBbGxpYW5jZTEMMAoGA1UECwwDQ1dHMQswCQYDVQQGEwJVUzELMAkGA1UECAwCTVkxEjAQBgNVBAcMCVdha2VmaWVsZDCCAiIwDQYJKoZIhvcNAQEBBQADggIPADCCAgoCggIBAL11U5yAIVLMrL3xS8u8ysMSdOkDeoTO+RcAy+uXXp6k4SC+jOy37gICEtYI+MKQV1EMeMMf3rM1ueZAO3iPFa0NEdi/oQ7npnGjBNI8wMzD8FfNe6rWtzkDaHpsZW///MwWDpGyJR+Xyjcq6U4vS9bS6zZ7jslw0Oczx4UsYgOsIUXSSBaGOrRbxJ/JC5gnDYEYvtNM+PDPczLNKAyhdvBZWNWHr7MZ0P5TeJQcXsAoShRX2Y8U8fRNJm7SeiFKDP0Nn/QKxOSt7zGP4xt9nMasE1q2ZTdar2+W13CRz37RI0ZWpq/+YquoEbZ7Uj7NmBTcqhb260nmDER2FpwwYwPSark92IZbamozB8d7OEI1jJgsrjJhKan0EmRaWVBpHT4xYKdEu7r09S0JhKyU+52WDmmVQTMpYLrm4Xl7hRxyPyBYkalrozsGmPs8vlhNq3VsVbyBSMSpEmUaeAa7LLE9/Vh0agJLVFHh1ehYKJpzHnmmBXUqx0Fz3afmDm1NX0sr3O/6xIx1VSTViT3KNxBYpVH1qjHATLzuxcWmm+75fcJMiPYPSMXVmRb3Q1l91AM4BBeWhlP3Fbc7gDy0r+s7m0sGS6PT2J2rGog2rUxnJ+zCM11M7DeO0XM2nny4uRYPPk9w2EXzfvtdvieYU/5RB4RDm5TGxHhGXVZUgac5AgMBAAEwDQYJKoZIhvcNAQELBQADggIBAFt2XGd3k5GpbO1EUm3u60zT1fE6u6pOscp156k5VnsHgaHRHdIAPNLeLNmR7y5OnrXbh13CrGwU1q84jjJXpv+v14xUCc5i01yopFTQFLr4A7NHp2nNYfNhhIVSFAgW43EflJflbLEelCJzxLlWb5BoDsZeeNmEQsXIM1mJ26R3r0dzsHBb0uy+8LNR1gdVqdjhC8BLy3gh4+BWuidyZNt07LveDsSFW5rcj5wRrSx9hXPIyVpjQSljNvY7MVTouqJzNAAQMsTKkXPkTXldCop9Qo9UPkHRRm0l7LLtdaOoXrct0Ymocf8zxf9bFNiw9f4WRYQM6sMhzt8+s/oDilo4QhcUgeJEiEPESi6ynYTV62SHA4eMunUJ5dlCaRnFiR9DTImFa5IRzie326/nW/SPCaKc/yrFIihMMjJoSAPhpTb/K6yHOUG8r+KiQut7NzqGV301pQ9u62dGL5Oi1VXmCFlE2ramZs15BNOUyAo2CBbRJg3jKcdu/8QC6ojjDvQ863+7LPtn74wJC5RpUJsS0GhQWgq5pAXO3wA61Uobxi6MkOpCC0zBWx/d4CqpS4j4hFgxWBTXX48ihPu+hIxIF/AxbqtPvqLMExW/xZITn6ArpWyQ9e4SUVr3n3F33ap1XdDyZ0vwFcm18JQAtsvXT6qCLrWOXnHUgfn/+Viu"
+ ],
+ "icon": "",
+ "supportedExtensions": [
+ {
+ "id": "hmac-secret",
+ "fail_if_unknown": false
+ },
+ {
+ "id": "credProtect",
+ "fail_if_unknown": false
+ }
+ ],
+ "authenticatorGetInfo": {
+ "versions": [
+ "U2F_V2",
+ "FIDO_2_0"
+ ],
+ "extensions": [
+ "credProtect",
+ "hmac-secret"
+ ],
+ "aaguid": "326adcf00cef46d0939298d6c4a84a72",
+ "options": {
+ "plat": false,
+ "rk": true,
+ "clientPin": true,
+ "up": true,
+ "uv": true,
+ "uvToken": false,
+ "config": false
+ },
+ "maxMsgSize": 1200,
+ "pinUvAuthProtocols": [
+ 1
+ ],
+ "maxCredentialCountInList": 16,
+ "maxCredentialIdLength": 128,
+ "transports": [
+ "usb",
+ "nfc"
+ ],
+ "algorithms": [
+ {
+ "type": "public-key",
+ "alg": -7
+ }
+ ],
+ "maxAuthenticatorConfigLength": 1024,
+ "defaultCredProtect": 2,
+ "firmwareVersion": 5
+ }
+}
\ No newline at end of file
diff --git a/Test/TestMetadataService.cs b/Test/TestMetadataService.cs
new file mode 100644
index 00000000..760656cc
--- /dev/null
+++ b/Test/TestMetadataService.cs
@@ -0,0 +1,21 @@
+using Fido2NetLib;
+
+namespace Test
+{
+ public class TestMetadataService : ConformanceMetadataService
+ {
+ public TestMetadataService(IEnumerable repositories) : base(repositories)
+ {
+ }
+
+ public void ChangeEntryGuid(Guid oldGuid, Guid newGuid)
+ {
+ if (!_entries.ContainsKey(oldGuid))
+ return;
+
+ _entries.Remove(oldGuid, out var entry);
+ entry.AaGuid = newGuid;
+ _entries.TryAdd(newGuid, entry);
+ }
+ }
+}