diff --git a/.github/workflows/build_and_functional_tests.yml b/.github/workflows/build_and_functional_tests.yml index 647f417b..25408fff 100644 --- a/.github/workflows/build_and_functional_tests.yml +++ b/.github/workflows/build_and_functional_tests.yml @@ -22,7 +22,7 @@ jobs: name: Build application using the reusable workflow uses: LedgerHQ/ledger-app-workflows/.github/workflows/reusable_build.yml@v1 with: - run_for_devices: '["nanox", "nanosp"]' + run_for_devices: '["nanox", "nanosp", "stax"]' upload_app_binaries_artifact: compiled_app_binaries ragger_tests: @@ -32,6 +32,8 @@ jobs: include: - model: nanox + - model: stax + - model: nanosp args: "--fast" diff --git a/.github/workflows/guidelines_enforcer.yml b/.github/workflows/guidelines_enforcer.yml index fabb7702..748c4806 100644 --- a/.github/workflows/guidelines_enforcer.yml +++ b/.github/workflows/guidelines_enforcer.yml @@ -22,4 +22,4 @@ jobs: name: Call Ledger guidelines_enforcer uses: LedgerHQ/ledger-app-workflows/.github/workflows/reusable_guidelines_enforcer.yml@v1 with: - run_for_devices: '["nanox", "nanosp"]' + run_for_devices: '["nanox", "nanosp", "stax"]' diff --git a/.github/workflows/misspellings_checks.yml b/.github/workflows/misspellings_checks.yml index 0333481f..60499ab7 100644 --- a/.github/workflows/misspellings_checks.yml +++ b/.github/workflows/misspellings_checks.yml @@ -22,7 +22,8 @@ jobs: uses: actions/checkout@v3 - name: Check misspellings - uses: codespell-project/actions-codespell@v1 + uses: codespell-project/actions-codespell@v2 with: builtin: clear,rare check_filenames: true + ignore_words_list: onTop, dummy diff --git a/Makefile b/Makefile index dd3d7fa2..1bef2814 100644 --- a/Makefile +++ b/Makefile @@ -21,7 +21,7 @@ endif include $(BOLOS_SDK)/Makefile.defines $(info TARGET_NAME=$(TARGET_NAME)) -ifneq ($(TARGET_NAME),$(filter $(TARGET_NAME),TARGET_NANOX TARGET_NANOS2)) +ifneq ($(TARGET_NAME),$(filter $(TARGET_NAME),TARGET_NANOX TARGET_NANOS2 TARGET_STAX)) $(error Environment variable TARGET_NAME is not valid or not supported) endif @@ -32,12 +32,13 @@ PATH_APP_LOAD_PARAMS = "5722689'" # int("WRA".encode("ascii").hex(), 16) PATH_APP_LOAD_PARAMS += "5262163'" # int("PKS".encode("ascii").hex(), 16) APPVERSION_M=1 -APPVERSION_N=3 +APPVERSION_N=4 APPVERSION_P=0 APPVERSION=$(APPVERSION_M).$(APPVERSION_N).$(APPVERSION_P) ICON_NANOX=icons/icon_security_key.gif ICON_NANOSP=icons/icon_security_key.gif +ICON_STAX=icons/icon_security_key_stax.gif ################ # Attestations # @@ -133,9 +134,6 @@ DEFINES += HAVE_DEBUG_THROWS # Compiler # ############## -# Remove warning on custom snprintf implementation usage -CFLAGS += -Wno-format-invalid-specifier -Wno-format-extra-args - # Application source files APP_SOURCE_PATH += src src-cbor SDK_SOURCE_PATH += lib_u2f diff --git a/conformance/prod-nanos.json b/conformance/prod-nanos.json index eea3de6f..c92fa2a4 100644 --- a/conformance/prod-nanos.json +++ b/conformance/prod-nanos.json @@ -73,7 +73,7 @@ "icon": "", "authenticatorGetInfo": { "versions": ["U2F_V2", "FIDO_2_0"], - "extensions": ["hmac-secret", "txAuthSimple"], + "extensions": ["hmac-secret"], "aaguid": "341e4da93c2e81035a9faad887135200", "options": { "rk": true, diff --git a/conformance/prod-nanosp.json b/conformance/prod-nanosp.json index 185ce908..beb38535 100644 --- a/conformance/prod-nanosp.json +++ b/conformance/prod-nanosp.json @@ -73,7 +73,7 @@ "icon": "", "authenticatorGetInfo": { "versions": ["U2F_V2", "FIDO_2_0"], - "extensions": ["hmac-secret", "txAuthSimple"], + "extensions": ["hmac-secret"], "aaguid": "58b44d0b0a7cf33afd48f7153c871352", "options": { "rk": true, diff --git a/conformance/prod-nanox.json b/conformance/prod-nanox.json index 85357095..7fd6d0ea 100644 --- a/conformance/prod-nanox.json +++ b/conformance/prod-nanox.json @@ -73,7 +73,7 @@ "icon": "", "authenticatorGetInfo": { "versions": ["U2F_V2", "FIDO_2_0"], - "extensions": ["hmac-secret", "txAuthSimple"], + "extensions": ["hmac-secret"], "aaguid": "fcb1bcb4f370078c6993bc24d0ae3fbe", "options": { "rk": true, diff --git a/conformance/prod-stax.json b/conformance/prod-stax.json new file mode 100644 index 00000000..da7afffa --- /dev/null +++ b/conformance/prod-stax.json @@ -0,0 +1,87 @@ +{ + "legalHeader": "Submission of this statement and retrieval and use of this statement indicates acceptance of the appropriate agreement located at https://fidoalliance.org/metadata/metadata-legal-terms/.", + "aaguid": "6e24d385-004a-16a0-7bfe-efd963845b34", + "description": "Ledger Stax FIDO2 Authenticator", + "alternativeDescriptions": {}, + "authenticatorVersion": 1, + "protocolFamily": "fido2", + "schema": 3, + "upv": [{ + "major": 1, + "minor": 0 + }], + "authenticationAlgorithms": [ + "secp256r1_ecdsa_sha256_raw", + "secp256k1_ecdsa_sha256_raw", + "ed25519_eddsa_sha512_raw" + ], + "publicKeyAlgAndEncodings": ["cose"], + "attestationTypes": [ + "basic_full" + ], + "userVerificationDetails": [ + [{ + "userVerificationMethod": "none" + } + ], + [{ + "userVerificationMethod": "presence_internal" + } + ], + [{ + "userVerificationMethod": "passcode_external" + } + ], + [{ + "userVerificationMethod": "presence_internal" + }, + { + "userVerificationMethod": "passcode_external" + } + ], + [{ + "userVerificationMethod": "passcode_internal", + "caDesc": { + "base": 10, + "minLength": 4, + "maxRetries": 3 + } + } + ], + [{ + "userVerificationMethod": "presence_internal" + }, + { + "userVerificationMethod": "passcode_internal", + "caDesc": { + "base": 10, + "minLength": 4, + "maxRetries": 3 + } + } + ] + ], + "keyProtection": ["hardware", "secure_element"], + "matcherProtection": ["on_chip"], + "cryptoStrength": 128, + "attachmentHint": ["external", "wired"], + "tcDisplay": ["any", "hardware"], + "tcDisplayContentType": "text/plain", + "attestationRootCertificates": [ + "MIIBgTCCAScCFBlo0s5QYFdXbfusRdQeoLX6QenlMAoGCCqGSM49BAMCMEMxCzAJBgNVBAYTAkZSMQ8wDQYDVQQKDAZMZWRnZXIxIzAhBgNVBAMMGkxlZGdlciBGSURPIEF0dGVzdGF0aW9uIENBMB4XDTIzMDIyMzEwMzMwOFoXDTMzMDIyMDEwMzMwOFowQzELMAkGA1UEBhMCRlIxDzANBgNVBAoMBkxlZGdlcjEjMCEGA1UEAwwaTGVkZ2VyIEZJRE8gQXR0ZXN0YXRpb24gQ0EwWTATBgcqhkjOPQIBBggqhkjOPQMBBwNCAATK7nXyH4pgN3TMwCWSoMDRe4EV8Jl3XzuhicZ/2gvh+zz3WmW0OZ/EcRYEA8F26ceeuMcd21WQRRKWpjWD+JWiMAoGCCqGSM49BAMCA0gAMEUCIQCwcsHuL8ZFL3FNyU/DOQn3bmx08lnn0O5RktLbOnoPHQIgOEi6ImAZ181q8RJiL0hbw7ZquuniRq6fjWjGoBu1Moo=" + ], + "icon": "", + "authenticatorGetInfo": { + "versions": ["U2F_V2", "FIDO_2_0"], + "extensions": ["hmac-secret", "txAuthSimple"], + "aaguid": "6e24d385004a16a07bfeefd963845b34", + "options": { + "rk": true, + "up": true, + "uv": true, + "clientPin": true + }, + "maxMsgSize": 1024, + "pinUvAuthProtocols": [1] + } +} \ No newline at end of file diff --git a/conformance/test-nanos.json b/conformance/test-nanos.json index 23af85ad..007c9b93 100644 --- a/conformance/test-nanos.json +++ b/conformance/test-nanos.json @@ -73,7 +73,7 @@ "icon": "", "authenticatorGetInfo": { "versions": ["U2F_V2", "FIDO_2_0"], - "extensions": ["hmac-secret", "txAuthSimple"], + "extensions": ["hmac-secret"], "aaguid": "341e4da93c2e81035a9faad887135200", "options": { "rk": true, diff --git a/conformance/test-nanosp.json b/conformance/test-nanosp.json index e05adfbb..ed0fb442 100644 --- a/conformance/test-nanosp.json +++ b/conformance/test-nanosp.json @@ -73,7 +73,7 @@ "icon": "", "authenticatorGetInfo": { "versions": ["U2F_V2", "FIDO_2_0"], - "extensions": ["hmac-secret", "txAuthSimple"], + "extensions": ["hmac-secret"], "aaguid": "58b44d0b0a7cf33afd48f7153c871352", "options": { "rk": true, diff --git a/conformance/test-nanox.json b/conformance/test-nanox.json index fa9556a2..ad35dec6 100644 --- a/conformance/test-nanox.json +++ b/conformance/test-nanox.json @@ -73,7 +73,7 @@ "icon": "", "authenticatorGetInfo": { "versions": ["U2F_V2", "FIDO_2_0"], - "extensions": ["hmac-secret", "txAuthSimple"], + "extensions": ["hmac-secret"], "aaguid": "fcb1bcb4f370078c6993bc24d0ae3fbe", "options": { "rk": true, diff --git a/conformance/test-stax.json b/conformance/test-stax.json new file mode 100644 index 00000000..9dc9efd9 --- /dev/null +++ b/conformance/test-stax.json @@ -0,0 +1,87 @@ +{ + "legalHeader": "Submission of this statement and retrieval and use of this statement indicates acceptance of the appropriate agreement located at https://fidoalliance.org/metadata/metadata-legal-terms/.", + "aaguid": "6e24d385-004a-16a0-7bfe-efd963845b34", + "description": "Ledger Stax FIDO2 Authenticator", + "alternativeDescriptions": {}, + "authenticatorVersion": 1, + "protocolFamily": "fido2", + "schema": 3, + "upv": [{ + "major": 1, + "minor": 0 + }], + "authenticationAlgorithms": [ + "secp256r1_ecdsa_sha256_raw", + "secp256k1_ecdsa_sha256_raw", + "ed25519_eddsa_sha512_raw" + ], + "publicKeyAlgAndEncodings": ["cose"], + "attestationTypes": [ + "basic_full" + ], + "userVerificationDetails": [ + [{ + "userVerificationMethod": "none" + } + ], + [{ + "userVerificationMethod": "presence_internal" + } + ], + [{ + "userVerificationMethod": "passcode_external" + } + ], + [{ + "userVerificationMethod": "presence_internal" + }, + { + "userVerificationMethod": "passcode_external" + } + ], + [{ + "userVerificationMethod": "passcode_internal", + "caDesc": { + "base": 10, + "minLength": 4, + "maxRetries": 3 + } + } + ], + [{ + "userVerificationMethod": "presence_internal" + }, + { + "userVerificationMethod": "passcode_internal", + "caDesc": { + "base": 10, + "minLength": 4, + "maxRetries": 3 + } + } + ] + ], + "keyProtection": ["hardware", "secure_element"], + "matcherProtection": ["on_chip"], + "cryptoStrength": 128, + "attachmentHint": ["external", "wired"], + "tcDisplay": ["any", "hardware"], + "tcDisplayContentType": "text/plain", + "attestationRootCertificates": [ + "MIIBgTCCAScCFCCm+doNCJYpK4mBmyd2xdAMieATMAoGCCqGSM49BAMCMEMxCzAJBgNVBAYTAkZSMQ8wDQYDVQQKDAZMZWRnZXIxIzAhBgNVBAMMGkxlZGdlciBGSURPIEF0dGVzdGF0aW9uIENBMB4XDTIyMDkyNjA4MDUzN1oXDTMyMDkyMzA4MDUzN1owQzELMAkGA1UEBhMCRlIxDzANBgNVBAoMBkxlZGdlcjEjMCEGA1UEAwwaTGVkZ2VyIEZJRE8gQXR0ZXN0YXRpb24gQ0EwWTATBgcqhkjOPQIBBggqhkjOPQMBBwNCAAQBNQxWyIb4hypWXXlxpGG0VMc5xyFsg/A5wq8q7NoNcHmNAkBtn7EOB21c107fcO1DC3OjNxqo6keVIGOj09hJMAoGCCqGSM49BAMCA0gAMEUCIQDirkVesYHsQU+cZiZA3K14MgBl0naL4BfAXdsAYgSBkgIgI9YDiMQGWt9eQPkBAyGv0A2Xx/+jRbkU9c0umz1cSXs=" + ], + "icon": "", + "authenticatorGetInfo": { + "versions": ["U2F_V2", "FIDO_2_0"], + "extensions": ["hmac-secret", "txAuthSimple"], + "aaguid": "6e24d385004a16a07bfeefd963845b34", + "options": { + "rk": true, + "up": true, + "uv": true, + "clientPin": true + }, + "maxMsgSize": 1024, + "pinUvAuthProtocols": [1] + } +} \ No newline at end of file diff --git a/glyphs/icon_security_key_64px.gif b/glyphs/icon_security_key_64px.gif new file mode 100644 index 00000000..5c048cf6 Binary files /dev/null and b/glyphs/icon_security_key_64px.gif differ diff --git a/icons/icon_security_key_stax.gif b/icons/icon_security_key_stax.gif new file mode 100644 index 00000000..7065a24f Binary files /dev/null and b/icons/icon_security_key_stax.gif differ diff --git a/include/crypto.h b/include/crypto.h index 1383bcad..323c42d8 100644 --- a/include/crypto.h +++ b/include/crypto.h @@ -51,7 +51,7 @@ int crypto_generate_public_key(cx_ecfp_private_key_t *private_key, /** * Generate credRandom key from nonce. */ -void crypto_generate_credRandom_key(const uint8_t *nonce, uint8_t *credRandom); +void crypto_generate_credRandom_key(const uint8_t *nonce, uint8_t *credRandom, bool with_uv); /** * Sign data_hash with private_key and store it in signature. diff --git a/include/ctap2.h b/include/ctap2.h index 0bb2ca97..b79db1f8 100644 --- a/include/ctap2.h +++ b/include/ctap2.h @@ -30,7 +30,6 @@ #include "cbip_decode.h" #include "extension_hmac_secret.h" -#include "extension_txAuthSimple.h" #define RP_ID_HASH_SIZE CX_SHA256_SIZE #define CRED_RANDOM_SIZE 32 @@ -86,8 +85,7 @@ #define PIN_PROTOCOL_VERSION_V1 1 -#define FLAG_EXTENSION_HMAC_SECRET 0x01 -#define FLAG_EXTENSION_TX_AUTH_SIMPLE 0x02 +#define FLAG_EXTENSION_HMAC_SECRET 0x01 // Helper to detect if CTAP2_CBOR_CMD command is proxyied over U2F_CMD // - CTAP2 calls that are CTAP2_CMD_CBOR commands: @@ -152,7 +150,6 @@ typedef struct ctap2_assert_data_s { uint8_t clientPinAuthenticated; // set if a standard FIDO client PIN authentication was performed uint8_t userPresenceRequired; // set if up is set - uint8_t singleCredential; // set if a single credential was provided in the allow list uint8_t extensions; // extensions flags as a bitmask uint8_t allowListPresent; @@ -161,11 +158,6 @@ typedef struct ctap2_assert_data_s { // Multiple flow data uint16_t currentCredentialIndex; ctap2_assert_multiple_flow_data_t multipleFlowData; - - char *txAuthMessage; // pointer to the TX Auth message or NULL - uint32_t txAuthLength; // length of the TX Auth message - char - txAuthLast; // last character of the txAuth CBOR field overwritten by a '\0' when displayed } ctap2_assert_data_t; typedef enum ctap2_ux_state_e { @@ -175,7 +167,6 @@ typedef enum ctap2_ux_state_e { CTAP2_UX_STATE_MULTIPLE_ASSERTION, CTAP2_UX_STATE_NO_ASSERTION, CTAP2_UX_STATE_RESET, - CTAP2_UX_STATE_CANCELLED = 0xff } ctap2_ux_state_t; bool ctap2_check_rpid_filter(const char *rpId, uint32_t rpIdLen); diff --git a/include/extension_txAuthSimple.h b/include/extension_txAuthSimple.h deleted file mode 100644 index b953045a..00000000 --- a/include/extension_txAuthSimple.h +++ /dev/null @@ -1,26 +0,0 @@ -/* -******************************************************************************* -* Ledger App Security Key -* (c) 2022 Ledger -* -* Licensed under the Apache License, Version 2.0 (the "License"); -* you may not use this file except in compliance with the License. -* You may obtain a copy of the License at -* -* http://www.apache.org/licenses/LICENSE-2.0 -* -* Unless required by applicable law or agreed to in writing, software -* distributed under the License is distributed on an "AS IS" BASIS, -* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -* See the License for the specific language governing permissions and -* limitations under the License. -********************************************************************************/ - -#ifndef __EXTENSION_TX_AUTH_SIMPLE_H__ -#define __EXTENSION_TX_AUTH_SIMPLE_H__ - -#define MAX_TX_AUTH_SIMPLE_SIZE 200 - -#define EXTENSION_TX_AUTH_SIMPLE "txAuthSimple" - -#endif diff --git a/include/ui_shared.h b/include/ui_shared.h index 728be748..4c6d31d5 100644 --- a/include/ui_shared.h +++ b/include/ui_shared.h @@ -21,4 +21,21 @@ void ui_idle(); +#ifdef HAVE_NBGL + +#include "nbgl_use_case.h" +#include "nbgl_layout.h" + +void app_nbgl_start_review(uint8_t nb_pairs, + const nbgl_layoutTagValue_t *pairs, + const char *confirm_text, + nbgl_choiceCallback_t on_choice, + nbgl_callback_t on_select); + +void app_nbgl_status(const char *message, + bool is_success, + nbgl_callback_t on_quit, + tune_index_e tune); +#endif + #endif diff --git a/src/app_main.c b/src/app_main.c index 38703129..fb28ed96 100644 --- a/src/app_main.c +++ b/src/app_main.c @@ -32,14 +32,7 @@ * Override app_ticker_event_callback io_event() dummy implementation */ void app_ticker_event_callback(void) { - if (ctap2UxState == CTAP2_UX_STATE_CANCELLED) { - ctap2UxState = CTAP2_UX_STATE_NONE; -#ifdef HAVE_BAGL - ux_stack_pop(); - ux_stack_push(); -#endif - ui_idle(); - } else if (ctap2UxState != CTAP2_UX_STATE_NONE) { + if (ctap2UxState != CTAP2_UX_STATE_NONE) { u2f_transport_ctap2_send_keepalive(&G_io_u2f, KEEPALIVE_REASON_TUP_NEEDED); } } diff --git a/src/crypto.c b/src/crypto.c index 442688e4..b550a698 100644 --- a/src/crypto.c +++ b/src/crypto.c @@ -26,8 +26,9 @@ #include "credential.h" #include "ctap2.h" -#define ROLE_PRIVATE_KEY 0 -#define ROLE_CRED_RANDOM_KEY 1 +#define ROLE_PRIVATE_KEY 0 +#define ROLE_CRED_RANDOM_KEY_UV 1 +#define ROLE_CRED_RANDOM_KEY_NO_UV 2 bool crypto_compare(const uint8_t *a, const uint8_t *b, uint16_t length) { uint16_t given_length = length; @@ -104,12 +105,16 @@ int crypto_generate_public_key(cx_ecfp_private_key_t *private_key, return app_public_key.W_len; } -void crypto_generate_credRandom_key(const uint8_t *nonce, uint8_t *credRandom) { +void crypto_generate_credRandom_key(const uint8_t *nonce, uint8_t *credRandom, bool with_uv) { uint8_t extended_nonce[32]; // credRandomKey = SHA256((1 << 128 | nonce) || privateKeySeed) memset(extended_nonce, 0, sizeof(extended_nonce)); - extended_nonce[15] = ROLE_CRED_RANDOM_KEY; + if (with_uv) { + extended_nonce[15] = ROLE_CRED_RANDOM_KEY_UV; + } else { + extended_nonce[15] = ROLE_CRED_RANDOM_KEY_NO_UV; + } memcpy(extended_nonce + 16, nonce, CREDENTIAL_NONCE_SIZE); crypto_compute_sha256(extended_nonce, sizeof(extended_nonce), diff --git a/src/ctap2_get_assertion.c b/src/ctap2_get_assertion.c index 9e10f401..18de8ff0 100644 --- a/src/ctap2_get_assertion.c +++ b/src/ctap2_get_assertion.c @@ -159,7 +159,6 @@ static int process_getAssert_authnr_allowList(cbipDecoder_t *decoder, cbipItem_t arrayLen = tmpItem.value; if ((status == CBIPH_STATUS_FOUND) && (arrayLen > 0)) { ctap2AssertData->allowListPresent = 1; - ctap2AssertData->singleCredential = (arrayLen == 1); for (int i = 0; i < arrayLen; i++) { if (i == 0) { @@ -228,21 +227,6 @@ static int process_getAssert_authnr_extensions(cbipDecoder_t *decoder, cbipItem_ // when building the response ctap2AssertData->extensions |= FLAG_EXTENSION_HMAC_SECRET; } - - // Check txAuthSimple extension - if (cbiph_get_map_key_str_text(decoder, - &extensionsItem, - EXTENSION_TX_AUTH_SIMPLE, - &ctap2AssertData->txAuthMessage, - &ctap2AssertData->txAuthLength) == CBIPH_STATUS_FOUND) { - // Avoid displaying an empty string, just in case - if (ctap2AssertData->txAuthLength == 0) { - PRINTF("Invalid empty txAuthSimple\n"); - return ERROR_INVALID_CBOR; - } - // TODO : check that the text is displayable - ctap2AssertData->extensions |= FLAG_EXTENSION_TX_AUTH_SIMPLE; - } } return 0; } @@ -662,9 +646,6 @@ static int build_getAssert_authData(uint8_t *buffer, uint32_t bufferLength, uint if ((ctap2AssertData->extensions & FLAG_EXTENSION_HMAC_SECRET) != 0) { extensionsSize++; } - if ((ctap2AssertData->extensions & FLAG_EXTENSION_TX_AUTH_SIMPLE) != 0) { - extensionsSize++; - } cbip_encoder_init(&encoder, buffer + offset, bufferLength - offset); cbip_add_map_header(&encoder, extensionsSize); @@ -674,7 +655,9 @@ static int build_getAssert_authData(uint8_t *buffer, uint32_t bufferLength, uint uint8_t *salt = NULL; uint32_t saltLength = 0; - crypto_generate_credRandom_key(ctap2AssertData->nonce, credRandom); + crypto_generate_credRandom_key(ctap2AssertData->nonce, + credRandom, + ctap2AssertData->pinRequired); status = compute_getAssert_hmacSecret_output(&salt, &saltLength, credRandom); if (status != ERROR_NONE) { @@ -683,18 +666,6 @@ static int build_getAssert_authData(uint8_t *buffer, uint32_t bufferLength, uint cbip_add_byte_string(&encoder, salt, saltLength); } - if ((ctap2AssertData->extensions & FLAG_EXTENSION_TX_AUTH_SIMPLE) != 0) { - cbip_add_string(&encoder, - EXTENSION_TX_AUTH_SIMPLE, - sizeof(EXTENSION_TX_AUTH_SIMPLE) - 1); - if (ctap2AssertData->txAuthLength > MAX_TX_AUTH_SIMPLE_SIZE) { - ctap2AssertData->txAuthLength = 0; - } - cbip_add_string(&encoder, - ctap2AssertData->txAuthMessage, - ctap2AssertData->txAuthLength); - } - if (encoder.fault) { PRINTF("Error encoding extensions\n"); return ERROR_OTHER; @@ -760,10 +731,7 @@ static int sign_and_build_getAssert_authData(uint8_t *authData, ctap2_send_keepalive_processing(); - mapSize = 2; - if (ctap2AssertData->singleCredential == 0) { - mapSize++; - } + mapSize = 3; if (credData->residentKey) { mapSize++; } @@ -772,7 +740,7 @@ static int sign_and_build_getAssert_authData(uint8_t *authData, cbip_add_map_header(&encoder, mapSize); - if (ctap2AssertData->singleCredential == 0) { + { // Rewrap credentials then encoded in the CBOR response // This could be optimized but this would means bypassing the // cbip_add_byte_string helper and the encoder. @@ -862,14 +830,10 @@ void ctap2_get_assertion_confirm(uint16_t idx) { uint32_t dataLen; credential_data_t credData; - ctap2UxState = CTAP2_UX_STATE_NONE; - PRINTF("ctap2_get_assertion_confirm %d\n", idx); ctap2_send_keepalive_processing(); - ui_idle(); - // Perform User Verification if required if (ctap2AssertData->pinRequired) { performBuiltInUv(); @@ -883,11 +847,6 @@ void ctap2_get_assertion_confirm(uint16_t idx) { return; } - // Restore the original last char in the CBOR buffer if a TX Auth was displayed - if (ctap2AssertData->txAuthMessage != NULL) { - ctap2AssertData->txAuthMessage[ctap2AssertData->txAuthLength] = ctap2AssertData->txAuthLast; - } - // Retrieve needed data from credential ctap2_get_assertion_credential_idx(idx); status = credential_decode(&credData, @@ -942,7 +901,5 @@ void ctap2_get_assertion_confirm(uint16_t idx) { } void ctap2_get_assertion_user_cancel() { - ctap2UxState = CTAP2_UX_STATE_NONE; send_cbor_error(&G_io_u2f, ERROR_OPERATION_DENIED); - ui_idle(); } diff --git a/src/ctap2_get_assertion_flow.c b/src/ctap2_get_assertion_flow.c index 2f5d0460..38b07cce 100644 --- a/src/ctap2_get_assertion_flow.c +++ b/src/ctap2_get_assertion_flow.c @@ -20,39 +20,59 @@ #include "os.h" #include "ux.h" +#include "format.h" #include "ctap2.h" #include "credential.h" #include "globals.h" +#include "ui_shared.h" -static void ctap2_ux_display_user_assertion(void) { +static void ctap2_ux_display_user_assertion(char buffer[static 36]) { ctap2_assert_data_t *ctap2AssertData = globals_get_ctap2_assert_data(); credential_data_t credData; - uint8_t nameLength = 0; - // TODO show that user.id is truncated if necessary if (credential_decode(&credData, ctap2AssertData->credential, ctap2AssertData->credentialLen, true) != 0) { // This should never happen, but keep a consistent state if it ever does - verifyHash[0] = '\0'; + buffer[0] = '\0'; } else if (credData.userStr != NULL) { - nameLength = MIN(credData.userStrLen, sizeof(verifyHash) - 1); - memcpy(verifyHash, credData.userStr, nameLength); - verifyHash[nameLength] = '\0'; + nameLength = MIN(credData.userStrLen, 36 - 1); + memcpy(buffer, credData.userStr, nameLength); + buffer[nameLength] = '\0'; } else { - snprintf(verifyHash, sizeof(verifyHash), "%.*H", credData.userIdLen, credData.userId); - verifyHash[sizeof(verifyHash) - 1] = '\0'; - nameLength = MIN(credData.userIdLen * 2, sizeof(verifyHash) - 1); + nameLength = MIN(credData.userIdLen, (36 / 2) - 1); + format_hex(credData.userId, nameLength, buffer, 36); + nameLength = nameLength * 2; } if (nameLength > 32) { - memcpy(verifyHash + 32, "...", sizeof("...")); + memcpy(buffer + 32, "...", sizeof("...")); } - PRINTF("name %s\n", verifyHash); + PRINTF("name %s\n", buffer); +} + +static void ctap_ux_on_user_choice(bool confirm, uint16_t idx) { + ctap2UxState = CTAP2_UX_STATE_NONE; + + if (confirm) { + ctap2_get_assertion_confirm(idx); +#ifdef HAVE_NBGL + app_nbgl_status("Login request signed", true, ui_idle, TUNE_SUCCESS); +#else + ui_idle(); +#endif + } else { + ctap2_get_assertion_user_cancel(); +#ifdef HAVE_NBGL + app_nbgl_status("Login cancelled", false, ui_idle, NBGL_NO_TUNE); +#else + ui_idle(); +#endif + } } #if defined(HAVE_BAGL) @@ -81,12 +101,12 @@ UX_STEP_NOCB(ux_ctap2_get_assertion_flow_user_step, UX_STEP_CB(ux_ctap2_get_assertion_flow_accept_step, pb, - ctap2_get_assertion_confirm(1), + ctap_ux_on_user_choice(true, 1), {&C_icon_validate_14, "Log in"}); UX_STEP_CB(ux_ctap2_get_assertion_flow_refuse_step, pbb, - ctap2_get_assertion_user_cancel(), + ctap_ux_on_user_choice(false, 0), {&C_icon_crossmark, "Reject", "login request"}); UX_FLOW(ux_ctap2_get_assertion_flow, @@ -122,7 +142,7 @@ static void display_next_multiple_flow_state(uint16_t idx) { "Log in user %d/%d", ctap2AssertData->currentCredentialIndex, ctap2AssertData->availableCredentials); - ctap2_ux_display_user_assertion(); + ctap2_ux_display_user_assertion(verifyHash); } static void display_next_state(uint8_t state) { @@ -160,7 +180,7 @@ UX_STEP_INIT(ux_ctap2_get_assertion_multiple_left_border, NULL, NULL, { UX_STEP_CB_INIT(ux_ctap2_get_assertion_multiple_user_border, bnnn_paging, { display_next_state(STATE_VARIABLE); }, - ctap2_get_assertion_confirm(ux_step), + ctap_ux_on_user_choice(true, ux_step), { .title = verifyName, .text = verifyHash, @@ -178,46 +198,6 @@ UX_FLOW(ux_ctap2_get_assertion_multiple_flow, &ux_ctap2_get_assertion_multiple_right_border, &ux_ctap2_get_assertion_flow_refuse_step); -// Extra steps if a text is associated to the TX for single assertion - -UX_STEP_NOCB(ux_ctap2_get_assertion_text_flow_first_step, - pnn, - { - &C_icon_certificate, - "Review login", - "request with text", - }); - -UX_STEP_NOCB(ux_ctap2_get_assertion_text_flow_text_step, - bnnn_paging, - {.title = "Message", .text = NULL}); - -UX_FLOW(ux_ctap2_get_assertion_text_flow, - &ux_ctap2_get_assertion_text_flow_first_step, - &ux_ctap2_get_assertion_text_flow_text_step, - &ux_ctap2_get_assertion_flow_domain_step, - &ux_ctap2_get_assertion_flow_user_step, - &ux_ctap2_get_assertion_flow_accept_step, - &ux_ctap2_get_assertion_flow_refuse_step); - -// Extra steps if a text is associated to the TX for multiple assertion -UX_STEP_NOCB(ux_ctap2_get_assertion_multiple_text_flow_first_step, - pnn, - { - &C_icon_certificate, - "Select user and", - "log in with text", - }); - -UX_FLOW(ux_ctap2_get_assertion_multiple_text_flow, - &ux_ctap2_get_assertion_multiple_text_flow_first_step, - &ux_ctap2_get_assertion_text_flow_text_step, - &ux_ctap2_get_assertion_flow_domain_step, - &ux_ctap2_get_assertion_multiple_left_border, - &ux_ctap2_get_assertion_multiple_user_border, - &ux_ctap2_get_assertion_multiple_right_border, - &ux_ctap2_get_assertion_flow_refuse_step); - // Dedicated flow to get user presence confirmation if no account is registered UX_STEP_NOCB(ux_ctap2_no_assertion_flow_0_step, pnn, @@ -237,7 +217,7 @@ UX_STEP_NOCB(ux_ctap2_no_assertion_flow_1_step, UX_STEP_CB(ux_ctap2_no_assertion_flow_3_step, pb, - ctap2_get_assertion_confirm(0), + ctap_ux_on_user_choice(true, 0), {&C_icon_back_x, "Close"}); UX_FLOW(ux_ctap2_no_assertion_flow, @@ -246,6 +226,129 @@ UX_FLOW(ux_ctap2_no_assertion_flow, &ux_ctap2_get_assertion_flow_domain_step, &ux_ctap2_no_assertion_flow_3_step); +#elif defined(HAVE_NBGL) + +#include "nbgl_use_case.h" +#include "nbgl_layout.h" + +static nbgl_page_t *pageContext; +#define NB_OF_PAIRS 2 +static const nbgl_layoutTagValue_t pairs[NB_OF_PAIRS] = {{.item = "Website", .value = rpID}, + {.item = "User ID", .value = verifyHash}}; + +#define SELECT_MAX_ID_NB 5 +#define SELECT_ID_BUFFER_SIZE 36 +static char user_id_list[SELECT_MAX_ID_NB][SELECT_ID_BUFFER_SIZE]; +static const char *const bar_texts[SELECT_MAX_ID_NB] = {user_id_list[0], + user_id_list[1], + user_id_list[2], + user_id_list[3], + user_id_list[4]}; +static uint8_t token_list[SELECT_MAX_ID_NB]; +uint8_t available_credentials; +uint8_t selected_credential; + +static void on_user_choice(bool confirm) { + ctap_ux_on_user_choice(confirm, selected_credential); +} + +static void on_user_select(void); + +static void on_user_select_exit() { + // Relauch without changing previously shown user id + ctap2_get_assertion_credential_idx(selected_credential); + app_nbgl_start_review(NB_OF_PAIRS, pairs, "Log in", on_user_choice, on_user_select); +} + +static bool on_user_select_navigation_callback(uint8_t page, nbgl_pageContent_t *content) { + if (page > available_credentials / SELECT_MAX_ID_NB) { + return false; + } + int first_page_index = page * SELECT_MAX_ID_NB + 1; + int i = 0; + while ((i < SELECT_MAX_ID_NB) && ((i + first_page_index) <= available_credentials)) { + ctap2_get_assertion_credential_idx(first_page_index + i); + ctap2_ux_display_user_assertion(user_id_list[i]); + token_list[i] = FIRST_USER_TOKEN + first_page_index + i; + i++; + } + content->tuneId = NBGL_NO_TUNE; + content->type = BARS_LIST; + content->barsList.barTexts = bar_texts; + content->barsList.tokens = token_list; + content->barsList.nbBars = i; + content->barsList.tuneId = TUNE_TAP_CASUAL; + return true; +} + +static void on_user_select_callback(int token, uint8_t index) { + UNUSED(index); + + if (token <= FIRST_USER_TOKEN) { + PRINTF("Should not happen!"); + return; + } + + int idx = token - FIRST_USER_TOKEN; + if (idx > available_credentials) { + PRINTF("Should not happen!"); + return; + } + + PRINTF("User selected %d\n", idx); + + // change the current credential idx and relaunch the review + selected_credential = idx; + ctap2_get_assertion_credential_idx(selected_credential); + ctap2_ux_display_user_assertion(verifyHash); + app_nbgl_start_review(NB_OF_PAIRS, pairs, "Log in", on_user_choice, on_user_select); +} + +static void on_user_select(void) { + // Reuse useCaseSettings which fit our needs + nbgl_useCaseSettings("User IDs", + 0, + (available_credentials - 1 / SELECT_MAX_ID_NB) + 1, + false, + on_user_select_exit, + on_user_select_navigation_callback, + on_user_select_callback); +} + +static void on_no_assertion_user_choice(int token, uint8_t index) { + UNUSED(token); + UNUSED(index); + + nbgl_pageRelease(pageContext); + + ctap2UxState = CTAP2_UX_STATE_NONE; + + ctap2_get_assertion_confirm(0); + ui_idle(); +} + +static void app_nbgl_no_assertion(void) { + snprintf(verifyHash, sizeof(verifyHash), "Login details not found\nfor %s", rpID); + nbgl_pageInfoDescription_t info = { + .bottomButtonStyle = NO_BUTTON_STYLE, + .footerText = NULL, + .centeredInfo.icon = &C_icon_security_key_64px, + .centeredInfo.offsetY = 0, + .centeredInfo.onTop = false, + .centeredInfo.style = LARGE_CASE_INFO, + .centeredInfo.text1 = verifyHash, + .centeredInfo.text2 = "Make sure to log in\nusing the same Ledger\nyou registered with.", + .centeredInfo.text3 = NULL, + .tapActionText = "Tap to dismiss", + .tapActionToken = FIRST_USER_TOKEN, + .topRightStyle = NO_BUTTON_STYLE, + .actionButtonText = NULL, + .tuneId = TUNE_TAP_CASUAL}; + + pageContext = nbgl_pageDrawInfo(&on_no_assertion_user_choice, NULL, &info); + nbgl_refresh(); +} + #endif void ctap2_get_assertion_ux(ctap2_ux_state_t state) { @@ -257,44 +360,45 @@ void ctap2_get_assertion_ux(ctap2_ux_state_t state) { rpID[len] = '\0'; PRINTF("rpId %s\n", rpID); - ctap2_ux_display_user_assertion(); - ctap2UxState = state; #if defined(HAVE_BAGL) - ux_step = 0; - ux_step_count = ctap2AssertData->availableCredentials; - - if (ctap2AssertData->txAuthMessage != NULL) { - ctap2AssertData->txAuthLast = ctap2AssertData->txAuthMessage[ctap2AssertData->txAuthLength]; - ctap2AssertData->txAuthMessage[ctap2AssertData->txAuthLength] = '\0'; - G_ux.externalText = ctap2AssertData->txAuthMessage; - } else { - G_ux.externalText = NULL; - } +#elif defined(HAVE_NBGL) + selected_credential = 1; + io_seproxyhal_play_tune(TUNE_LOOK_AT_ME); +#endif switch (state) { case CTAP2_UX_STATE_GET_ASSERTION: { - if (ctap2AssertData->txAuthMessage != NULL) { - ux_flow_init(0, ux_ctap2_get_assertion_text_flow, NULL); - } else { - ux_flow_init(0, ux_ctap2_get_assertion_flow, NULL); - } + ctap2_ux_display_user_assertion(verifyHash); +#if defined(HAVE_BAGL) + ux_flow_init(0, ux_ctap2_get_assertion_flow, NULL); + break; +#elif defined(HAVE_NBGL) + app_nbgl_start_review(NB_OF_PAIRS, pairs, "Log in", on_user_choice, NULL); +#endif break; } case CTAP2_UX_STATE_MULTIPLE_ASSERTION: { - if (ctap2AssertData->txAuthMessage != NULL) { - ux_flow_init(0, ux_ctap2_get_assertion_multiple_text_flow, NULL); - } else { - ux_flow_init(0, ux_ctap2_get_assertion_multiple_flow, NULL); - } +#if defined(HAVE_BAGL) + ux_step_count = ctap2AssertData->availableCredentials; + ux_flow_init(0, ux_ctap2_get_assertion_multiple_flow, NULL); +#elif defined(HAVE_NBGL) + available_credentials = ctap2AssertData->availableCredentials; + ctap2_get_assertion_credential_idx(selected_credential); + ctap2_ux_display_user_assertion(verifyHash); + app_nbgl_start_review(NB_OF_PAIRS, pairs, "Log in", on_user_choice, on_user_select); +#endif break; } default: { +#if defined(HAVE_BAGL) ux_flow_init(0, ux_ctap2_no_assertion_flow, NULL); +#elif defined(HAVE_NBGL) + app_nbgl_no_assertion(); +#endif break; } } -#endif } diff --git a/src/ctap2_get_info.c b/src/ctap2_get_info.c index b3d06184..a5c9385f 100644 --- a/src/ctap2_get_info.c +++ b/src/ctap2_get_info.c @@ -59,9 +59,8 @@ void ctap2_get_info_handle(u2f_service_t *service, uint8_t *buffer, uint16_t len // extensions (0x02) cbip_add_int(&encoder, TAG_EXTENSIONS); - cbip_add_array_header(&encoder, 2); + cbip_add_array_header(&encoder, 1); cbip_add_string(&encoder, EXTENSION_HMAC_SECRET, sizeof(EXTENSION_HMAC_SECRET) - 1); - cbip_add_string(&encoder, EXTENSION_TX_AUTH_SIMPLE, sizeof(EXTENSION_TX_AUTH_SIMPLE) - 1); // AAGUID (0x03) diff --git a/src/ctap2_make_credential.c b/src/ctap2_make_credential.c index d5e968f4..035004b9 100644 --- a/src/ctap2_make_credential.c +++ b/src/ctap2_make_credential.c @@ -658,14 +658,10 @@ void ctap2_make_credential_confirm() { uint8_t nonce[CREDENTIAL_NONCE_SIZE]; int status; - ctap2UxState = CTAP2_UX_STATE_NONE; - PRINTF("ctap2_make_credential_confirm\n"); ctap2_send_keepalive_processing(); - ui_idle(); - // Perform User Verification if required if (ctap2RegisterData->pinRequired) { performBuiltInUv(); @@ -716,7 +712,5 @@ void ctap2_make_credential_confirm() { } void ctap2_make_credential_user_cancel() { - ctap2UxState = CTAP2_UX_STATE_NONE; send_cbor_error(&G_io_u2f, ERROR_OPERATION_DENIED); - ui_idle(); } diff --git a/src/ctap2_make_credential_flow.c b/src/ctap2_make_credential_flow.c index a2f42fc6..b9a282dd 100644 --- a/src/ctap2_make_credential_flow.c +++ b/src/ctap2_make_credential_flow.c @@ -20,25 +20,45 @@ #include "os.h" #include "ux.h" +#include "format.h" #include "ctap2.h" #include "globals.h" +#include "ui_shared.h" static void ctap2_ux_get_display_user(void) { ctap2_register_data_t *ctap2RegisterData = globals_get_ctap2_register_data(); + // TODO show that user.id is truncated if necessary if (ctap2RegisterData->userStr) { uint8_t nameLength = MIN(ctap2RegisterData->userStrLen, sizeof(verifyHash) - 1); memcpy(verifyHash, ctap2RegisterData->userStr, nameLength); verifyHash[nameLength] = '\0'; } else { - snprintf(verifyHash, - sizeof(verifyHash), - "%.*H", - ctap2RegisterData->userIdLen, - ctap2RegisterData->userId); - verifyHash[sizeof(verifyHash) - 1] = '\0'; + uint8_t nameLength = MIN(ctap2RegisterData->userIdLen, (sizeof(verifyHash) - 1) / 2); + + format_hex(ctap2RegisterData->userId, nameLength, verifyHash, sizeof(verifyHash)); + } +} + +static void ctap_ux_on_user_choice(bool confirm) { + ctap2UxState = CTAP2_UX_STATE_NONE; + + if (confirm) { + ctap2_make_credential_confirm(); +#ifdef HAVE_NBGL + app_nbgl_status("Registration details\nsent", true, ui_idle, TUNE_SUCCESS); +#else + ui_idle(); +#endif + } else { + ctap2_make_credential_user_cancel(); +#ifdef HAVE_NBGL + app_nbgl_status("Registration cancelled", false, ui_idle, NBGL_NO_TUNE); +#else + ui_idle(); +#endif } } @@ -68,7 +88,7 @@ UX_STEP_NOCB(ux_ctap2_make_cred_flow_user_step, UX_STEP_CB(ux_ctap2_make_cred_flow_accept_step, pb, - ctap2_make_credential_confirm(), + ctap_ux_on_user_choice(true), { &C_icon_validate_14, "Register", @@ -76,7 +96,7 @@ UX_STEP_CB(ux_ctap2_make_cred_flow_accept_step, UX_STEP_CB(ux_ctap2_make_cred_flow_refuse_step, pb, - ctap2_make_credential_user_cancel(), + ctap_ux_on_user_choice(false), { &C_icon_crossmark, "Don't register", @@ -84,7 +104,7 @@ UX_STEP_CB(ux_ctap2_make_cred_flow_refuse_step, UX_STEP_CB(ux_ctap2_make_cred_resident_flow_accept_step, pbb, - ctap2_make_credential_confirm(), + ctap_ux_on_user_choice(true), { &C_icon_validate_14, "Register", @@ -105,6 +125,21 @@ UX_FLOW(ux_ctap2_make_cred_resident_flow, &ux_ctap2_make_cred_resident_flow_accept_step, &ux_ctap2_make_cred_flow_refuse_step); +#elif defined(HAVE_NBGL) + +#include "nbgl_use_case.h" +#include "nbgl_layout.h" + +#define NB_OF_PAIRS 2 +static const nbgl_layoutTagValue_t pairs[NB_OF_PAIRS] = {{ + .item = "Website", + .value = rpID, + }, + { + .item = "User ID", + .value = verifyHash, + }}; + #endif void ctap2_make_credential_ux(void) { @@ -121,10 +156,21 @@ void ctap2_make_credential_ux(void) { ctap2_ux_get_display_user(); #if defined(HAVE_BAGL) - G_ux.externalText = NULL; ux_flow_init(0, (ctap2RegisterData->residentKey ? ux_ctap2_make_cred_resident_flow : ux_ctap2_make_cred_flow), NULL); +#elif defined(HAVE_NBGL) + io_seproxyhal_play_tune(TUNE_LOOK_AT_ME); + + if (ctap2RegisterData->residentKey) { + app_nbgl_start_review(NB_OF_PAIRS, + pairs, + "Register resident key", + ctap_ux_on_user_choice, + NULL); + } else { + app_nbgl_start_review(NB_OF_PAIRS, pairs, "Register", ctap_ux_on_user_choice, NULL); + } #endif } diff --git a/src/ctap2_processing.c b/src/ctap2_processing.c index ae7c6bdb..cc9085d2 100644 --- a/src/ctap2_processing.c +++ b/src/ctap2_processing.c @@ -132,11 +132,20 @@ void ctap2_handle_cmd_cancel(u2f_service_t *service, uint8_t *buffer, uint16_t l UNUSED(length); PRINTF("ctap2_cmd_cancel %d\n", ctap2UxState); - if ((ctap2UxState != CTAP2_UX_STATE_NONE) && (ctap2UxState != CTAP2_UX_STATE_CANCELLED)) { + if (ctap2UxState != CTAP2_UX_STATE_NONE) { PRINTF("Cancel pending UI\n"); - ctap2UxState = CTAP2_UX_STATE_CANCELLED; - ui_idle(); + ctap2UxState = CTAP2_UX_STATE_NONE; + + // Answer as fast as possible as Chrome expect a fast answer and in case + // it didn't comes fast enough, it won't be sent back if the user + // eventually choose again this authenticator. send_cbor_error(service, ERROR_KEEPALIVE_CANCEL); + +#ifdef HAVE_BAGL + ux_stack_pop(); + ux_stack_push(); +#endif + ui_idle(); } } diff --git a/src/ctap2_reset.c b/src/ctap2_reset.c index d59dad88..d0789e6d 100644 --- a/src/ctap2_reset.c +++ b/src/ctap2_reset.c @@ -33,17 +33,12 @@ void ctap2_reset_handle(u2f_service_t *service, uint8_t *buffer, uint16_t length } void ctap2_reset_confirm() { - ctap2UxState = CTAP2_UX_STATE_NONE; - config_process_ctap2_reset(); G_io_apdu_buffer[0] = ERROR_NONE; send_cbor_response(&G_io_u2f, 1); - ui_idle(); } void ctap2_reset_cancel() { - ctap2UxState = CTAP2_UX_STATE_NONE; send_cbor_error(&G_io_u2f, ERROR_OPERATION_DENIED); - ui_idle(); } diff --git a/src/ctap2_reset_flow.c b/src/ctap2_reset_flow.c index 6ffb0893..81eaf4bc 100644 --- a/src/ctap2_reset_flow.c +++ b/src/ctap2_reset_flow.c @@ -21,6 +21,19 @@ #include "ctap2.h" #include "globals.h" +#include "ui_shared.h" + +static void ctap_ux_on_user_action(bool confirm) { + ctap2UxState = CTAP2_UX_STATE_NONE; + + if (confirm) { + ctap2_reset_confirm(); + ui_idle(); + } else { + ctap2_reset_cancel(); + ui_idle(); + } +} #if defined(HAVE_BAGL) @@ -34,12 +47,12 @@ UX_STEP_NOCB(ux_ctap2_reset_flow_0_step, UX_STEP_CB(ux_ctap2_reset_flow_1_step, pb, - ctap2_reset_confirm(), + ctap_ux_on_user_action(true), {&C_icon_validate_14, "Yes, delete"}); UX_STEP_CB(ux_ctap2_reset_flow_2_step, pb, - ctap2_reset_cancel(), + ctap_ux_on_user_action(false), { &C_icon_crossmark, "No, don't delete", @@ -53,8 +66,25 @@ UX_FLOW(ux_ctap2_reset_flow, void ctap2_reset_ux(void) { ctap2UxState = CTAP2_UX_STATE_RESET; - G_ux.externalText = NULL; ux_flow_init(0, ux_ctap2_reset_flow, NULL); } +#elif defined(HAVE_NBGL) +#include "nbgl_use_case.h" + +void ctap2_reset_ux(void) { + ctap2UxState = CTAP2_UX_STATE_RESET; + + io_seproxyhal_play_tune(TUNE_LOOK_AT_ME); + + nbgl_useCaseChoice(&C_warning64px, + "Delete saved login\n" + "details for all\n" + "websites?\n", + NULL, + "Yes, delete", + "No, don't delete", + ctap_ux_on_user_action); +} + #endif \ No newline at end of file diff --git a/src/u2f_processing.c b/src/u2f_processing.c index be4844ca..a8f3d629 100644 --- a/src/u2f_processing.c +++ b/src/u2f_processing.c @@ -23,6 +23,7 @@ #include "cx.h" #include "ux.h" #include "io.h" +#include "format.h" #include "u2f_processing.h" #include "u2f_service.h" @@ -399,6 +400,8 @@ static int u2f_process_user_presence_confirmed(void) { /* U2F UX Flows */ /******************************************/ +#if defined(HAVE_BAGL) + static unsigned int u2f_callback_cancel(void) { io_send_sw(SW_PROPRIETARY_INTERNAL); ui_idle(); @@ -471,10 +474,46 @@ UX_FLOW(ux_login_flow, &ux_login_flow_2_step, FLOW_LOOP); +#elif defined(HAVE_NBGL) + +#include "nbgl_use_case.h" + +#define NB_OF_PAIRS 2 +static const nbgl_layoutTagValue_t pairs[NB_OF_PAIRS] = {{ + .item = "Website", + .value = verifyName, + }, + { + .item = "Website ID", + .value = verifyHash, + }}; + +static void on_register_choice(bool confirm) { + if (confirm) { + u2f_process_user_presence_confirmed(); + app_nbgl_status("Registration details\nsent", true, ui_idle, TUNE_SUCCESS); + } else { + io_send_sw(SW_PROPRIETARY_INTERNAL); + app_nbgl_status("Registration cancelled", false, ui_idle, NBGL_NO_TUNE); + } +} + +static void on_login_choice(bool confirm) { + if (confirm) { + u2f_process_user_presence_confirmed(); + app_nbgl_status("Login request signed", true, ui_idle, TUNE_SUCCESS); + } else { + io_send_sw(SW_PROPRIETARY_INTERNAL); + app_nbgl_status("Log in cancelled", false, ui_idle, NBGL_NO_TUNE); + } +} + +#endif + static void u2f_prompt_user_presence(bool enroll, uint8_t *applicationParameter) { UX_WAKE_UP(); - snprintf(verifyHash, sizeof(verifyHash), "%.*H", 32, applicationParameter); + format_hex(applicationParameter, 32, verifyHash, sizeof(verifyHash)); strcpy(verifyName, "Unknown"); const char *name = fido_match_known_appid(applicationParameter); @@ -482,12 +521,19 @@ static void u2f_prompt_user_presence(bool enroll, uint8_t *applicationParameter) strlcpy(verifyName, name, sizeof(verifyName)); } - G_ux.externalText = NULL; +#if defined(HAVE_BAGL) if (enroll) { ux_flow_init(0, ux_register_flow, NULL); } else { ux_flow_init(0, ux_login_flow, NULL); } +#elif defined(HAVE_NBGL) + if (enroll) { + app_nbgl_start_review(NB_OF_PAIRS, pairs, "Register", on_register_choice, NULL); + } else { + app_nbgl_start_review(NB_OF_PAIRS, pairs, "Login", on_login_choice, NULL); + } +#endif } /******************************************/ diff --git a/src/ui_shared.c b/src/ui_shared.c index 82599207..cc3a2a8a 100644 --- a/src/ui_shared.c +++ b/src/ui_shared.c @@ -31,7 +31,7 @@ static void app_quit(void) { static void display_warning(); static void display_settings(); -static void toogle_settings(); +static void toggle_settings(); UX_STEP_NOCB(ux_settings_enabling_flow_warning_step, bn_paging, @@ -59,7 +59,7 @@ UX_STEP_CB(ux_settings_warning_flow_cancel_step, UX_STEP_CB(ux_settings_enabling_flow_confirm_step, pbb, - toogle_settings(), + toggle_settings(), { &C_icon_validate_14, "Enable", @@ -75,7 +75,7 @@ static void display_warning() { ux_flow_init(0, ux_settings_enabling_flow, NULL); } -static void toogle_settings() { +static void toggle_settings() { if (config_get_rk_enabled()) { config_set_rk_enabled(false); } else { @@ -84,7 +84,7 @@ static void toogle_settings() { display_settings(); } -UX_STEP_CB(ux_settings_flow_1_enabled_step, bn, toogle_settings(), {"Resident keys", "Enabled"}); +UX_STEP_CB(ux_settings_flow_1_enabled_step, bn, toggle_settings(), {"Resident keys", "Enabled"}); UX_STEP_CB(ux_settings_flow_1_disabled_step, bn, display_warning(), {"Resident keys", "Disabled"}); @@ -151,8 +151,259 @@ void ui_idle(void) { if (G_ux.stack_count == 0) { ux_stack_push(); } - G_ux.externalText = NULL; ux_flow_init(0, ux_idle_flow, NULL); } -#endif \ No newline at end of file +#elif defined(HAVE_NBGL) + +#include "nbgl_use_case.h" +#include "nbgl_page.h" +#include "nbgl_layout.h" + +// 'About' menu + +static const char *const INFO_TYPES[] = {"Version", "Developer", "Copyright"}; +static const char *const INFO_CONTENTS[] = {APPVERSION, "Ledger", "(c) 2023 Ledger"}; + +#ifdef HAVE_RK_SUPPORT_SETTING + +static nbgl_layoutSwitch_t toggle; +static void ui_menu_settings_page(void); + +static void warning_choice(bool accept) { + if (accept) { + config_set_rk_enabled(true); + app_nbgl_status("Resident keys enabled", true, ui_menu_settings_page, NBGL_NO_TUNE); + } else { + ui_menu_settings_page(); + } +} + +static void settings_callback(int token, uint8_t index) { + UNUSED(index); + switch (token) { + case FIRST_USER_TOKEN: + if (config_get_rk_enabled()) { + config_set_rk_enabled(false); + } else { + nbgl_useCaseChoice(&C_warning64px, + "Enable resident keys?", + "Updating the OS or this app\n" + "will delete login info stored on\n" + "this device, causing login\n" + "issues.", + "Enable", + "Cancel", + warning_choice); + } + break; + default: + PRINTF("Should not happen!"); + break; + } +} +#endif // HAVE_RK_SUPPORT_SETTING + +static bool nav_callback(uint8_t page, nbgl_pageContent_t *content) { + if (page == 0) { + content->type = INFOS_LIST; + content->infosList.nbInfos = 3; + content->infosList.infoTypes = INFO_TYPES; + content->infosList.infoContents = INFO_CONTENTS; + } +#ifdef HAVE_RK_SUPPORT_SETTING + else if (page == 1) { + toggle.text = "Resident keys"; + toggle.subText = + "Stores login info on this\n" + "device's memory and lets you\n" + "login without username.\n\n" + "Caution: Updating the OS or\n" + "this app will delete the stored\n" + "login info, causing login issues\n" + "for connected accounts"; + toggle.token = FIRST_USER_TOKEN; + toggle.tuneId = TUNE_TAP_CASUAL; + toggle.initState = config_get_rk_enabled(); + content->type = SWITCHES_LIST; + content->switchesList.nbSwitches = 1; + content->switchesList.switches = &toggle; + } else { +#endif // HAVE_RK_SUPPORT_SETTING + return false; + } + return true; +} + +static void ui_menu_settings(uint8_t init_page) { + nbgl_useCaseSettings(APPNAME, + init_page, +#ifdef HAVE_RK_SUPPORT_SETTING + 2, +#else + 1, +#endif + false, + ui_idle, + nav_callback, + settings_callback); +} + +static void ui_menu_settings_home(void) { + ui_menu_settings(0); +} + +static void ui_menu_settings_page(void) { + ui_menu_settings(1); +} + +void ui_idle(void) { + nbgl_useCaseHome(APPNAME, + &C_icon_security_key_64px, + "Use this app for two-factor\nauthentication and\npassword-less log ins.", +#ifdef HAVE_RK_SUPPORT_SETTING + true, +#else + false, +#endif + ui_menu_settings_home, + app_quit); +} + +static nbgl_layout_t *layout; +static nbgl_page_t *pageContext; +static nbgl_choiceCallback_t onChoice; +static nbgl_callback_t onSelect; +static nbgl_callback_t onQuit; + +enum { TITLE_TOKEN = FIRST_USER_TOKEN, CHOICE_TOKEN, SELECT_TOKEN, QUIT_TOKEN }; + +static void onActionCallback(int token, uint8_t index) { + if (token == CHOICE_TOKEN && onChoice != NULL) { + // Release the review layout. + nbgl_layoutRelease(layout); + + onChoice(index == 0); + } else if (token == SELECT_TOKEN && onSelect != NULL) { + // Release the review layout. + nbgl_layoutRelease(layout); + + onSelect(); + } else if (token == QUIT_TOKEN && onQuit != NULL) { + // Release the review layout. + nbgl_layoutRelease(layout); + + onQuit(); + } +} + +void app_nbgl_start_review(uint8_t nb_pairs, + const nbgl_layoutTagValue_t *pairs, + const char *confirm_text, + nbgl_choiceCallback_t on_choice, + nbgl_callback_t on_select) { + nbgl_layoutDescription_t layoutDescription; + onChoice = on_choice; + onSelect = on_select; + + layoutDescription.modal = false; + layoutDescription.withLeftBorder = true; + layoutDescription.onActionCallback = onActionCallback; + layoutDescription.tapActionText = NULL; + layoutDescription.ticker.tickerCallback = NULL; + + layout = nbgl_layoutGet(&layoutDescription); + + nbgl_layoutBar_t bar; + bar.text = APPNAME; + bar.subText = NULL; + bar.iconRight = NULL; + bar.iconLeft = NULL; + bar.token = TITLE_TOKEN; + bar.centered = true; + bar.inactive = false; + bar.tuneId = NBGL_NO_TUNE; + nbgl_layoutAddTouchableBar(layout, &bar); + nbgl_layoutAddSeparationLine(layout); + + const nbgl_layoutTagValueList_t tagValueList = {.nbPairs = nb_pairs, + .pairs = pairs, + .smallCaseForValue = false, + .nbMaxLinesForValue = 0, + .wrapping = false}; + + nbgl_layoutAddTagValueList(layout, &tagValueList); + + if (onSelect) { + nbgl_layoutButton_t select_button_info = {.text = "Select another ID", + .icon = NULL, + .token = SELECT_TOKEN, + .style = WHITE_BACKGROUND, + .fittingContent = true, + .onBottom = false, + .tuneId = TUNE_TAP_CASUAL}; + + nbgl_layoutAddButton(layout, &select_button_info); + } + + nbgl_layoutChoiceButtons_t choice_buttons_info = {.bottomText = "Cancel", + .token = CHOICE_TOKEN, + .topText = confirm_text, + .style = ROUNDED_AND_FOOTER_STYLE, + .tuneId = TUNE_TAP_CASUAL}; + nbgl_layoutAddChoiceButtons(layout, &choice_buttons_info); + + nbgl_layoutDraw(layout); + + nbgl_refresh(); +} + +static void tickerCallback(void) { + nbgl_pageRelease(pageContext); + if (onQuit != NULL) { + onQuit(); + } +} + +void app_nbgl_status(const char *message, + bool is_success, + nbgl_callback_t on_quit, + tune_index_e tune) { + if (tune != NBGL_NO_TUNE) { + io_seproxyhal_play_tune(tune); + } + + nbgl_screenTickerConfiguration_t ticker = { + .tickerCallback = &tickerCallback, + .tickerIntervale = 0, // not periodic + .tickerValue = 3000 // 3 seconds + }; + onQuit = on_quit; + + nbgl_pageInfoDescription_t info = {.bottomButtonStyle = NO_BUTTON_STYLE, + .footerText = NULL, + .centeredInfo.icon = &C_round_cross_64px, + .centeredInfo.offsetY = 0, + .centeredInfo.onTop = false, + .centeredInfo.style = LARGE_CASE_INFO, + .centeredInfo.text1 = message, + .centeredInfo.text2 = NULL, + .centeredInfo.text3 = NULL, + .tapActionText = "", + .tapActionToken = QUIT_TOKEN, + .topRightStyle = NO_BUTTON_STYLE, + .actionButtonText = NULL, + .tuneId = TUNE_TAP_CASUAL}; + + if (is_success) { + info.centeredInfo.icon = &C_round_check_64px; + } else { + info.centeredInfo.icon = &C_round_warning_64px; + } + + pageContext = nbgl_pageDrawInfo(&onActionCallback, &ticker, &info); + + nbgl_refresh(); +} + +#endif diff --git a/tests/speculos/conftest.py b/tests/speculos/conftest.py index fde6e084..34ab150f 100644 --- a/tests/speculos/conftest.py +++ b/tests/speculos/conftest.py @@ -32,7 +32,7 @@ BACKENDS = ["speculos"] -DEVICES = ["nanox", "nanosp", "all"] +DEVICES = ["nanox", "nanosp", "stax", "all"] def pytest_addoption(parser): diff --git a/tests/speculos/ctap1_client.py b/tests/speculos/ctap1_client.py index 435cb481..53295a63 100644 --- a/tests/speculos/ctap1_client.py +++ b/tests/speculos/ctap1_client.py @@ -51,11 +51,17 @@ def __init__(self, device, model, navigator, debug=False): self.debug = debug def confirm(self): - instructions = [NavInsID.BOTH_CLICK] + if self.model == "stax": + instructions = [NavInsID.USE_CASE_CHOICE_CONFIRM] + else: + instructions = [NavInsID.BOTH_CLICK] self.navigator.navigate(instructions, screen_change_after_last_instruction=False) def wait_for_return_on_dashboard(self): + if self.model == "stax": + # On Stax tap on the center to dismiss the status message faster + self.navigator.navigate([NavInsID.USE_CASE_STATUS_DISMISS]) self.navigator._backend.wait_for_home_screen() def parse_response(self, response): @@ -102,6 +108,12 @@ def register(self, client_param, app_param, user_accept=True, text = "Register" else: text = "Abort" + elif self.model == "stax": + if user_accept is not None: + if not user_accept: + val_ins = [NavInsID.USE_CASE_CHOICE_REJECT] + else: + val_ins = [NavInsID.USE_CASE_CHOICE_CONFIRM] navigate(self.navigator, user_accept, @@ -158,6 +170,12 @@ def authenticate(self, client_param, app_param, key_handle, text = "Login" else: text = "Abort" + elif self.model == "stax": + if user_accept is not None: + if not user_accept: + val_ins = [NavInsID.USE_CASE_CHOICE_REJECT] + else: + val_ins = [NavInsID.USE_CASE_CHOICE_CONFIRM] navigate(self.navigator, user_accept, diff --git a/tests/speculos/ctap2_client.py b/tests/speculos/ctap2_client.py index f3e3f559..e0fc4bb3 100644 --- a/tests/speculos/ctap2_client.py +++ b/tests/speculos/ctap2_client.py @@ -2,7 +2,7 @@ from typing import Mapping -from ragger.navigator import NavInsID +from ragger.navigator import NavInsID, NavIns from fido2 import cbor from fido2.ctap import CtapError @@ -35,11 +35,21 @@ def __init__(self, device, model, navigator, ctap2_u2f_proxy, debug=False): super().__init__(device) def confirm(self): - instructions = [NavInsID.BOTH_CLICK] + if self.model == "stax": + instructions = [NavInsID.USE_CASE_CHOICE_CONFIRM] + else: + instructions = [NavInsID.BOTH_CLICK] self.navigator.navigate(instructions, screen_change_after_last_instruction=False) def wait_for_return_on_dashboard(self): + if self.model == "stax": + # On Stax tap on the center to dismiss the status message faster + # Ignore if there is nothing that happen (probably already on home screen), + # which is expected for flow without status (reset) + self.navigator.navigate([NavInsID.USE_CASE_STATUS_DISMISS], + screen_change_after_last_instruction=False) + self.navigator._backend.wait_for_home_screen() def send_cbor_nowait(self, cmd, data=None, *, event=None, on_keepalive=None): @@ -134,6 +144,12 @@ def make_credential(self, client_data_hash, rp, user, key_params, text = "Don't register" else: text = "Register$" + elif self.model == "stax": + if user_accept is not None: + if not user_accept: + val_ins = [NavInsID.USE_CASE_CHOICE_REJECT] + else: + val_ins = [NavInsID.USE_CASE_CHOICE_CONFIRM] navigate(self.navigator, user_accept, @@ -168,8 +184,6 @@ def get_assertion(self, rp_id, client_data_hash, allow_list=None, assert login_type in ["simple", "multi", "none"] - TAG_RESP_CREDENTIAL = 0x01 - cmd = Ctap2.CMD.GET_ASSERTION data = args(rp_id, client_data_hash, @@ -210,6 +224,21 @@ def get_assertion(self, rp_id, client_data_hash, allow_list=None, text = "Log in" else: text = "Reject" + elif self.model == "stax": + if user_accept is not None: + if login_type == "none": + val_ins = [NavInsID.TAPPABLE_CENTER_TAP] + + if not user_accept: + val_ins = [NavInsID.USE_CASE_CHOICE_REJECT] + else: + if login_type == "multi" and select_user_idx != 1: + assert select_user_idx <= 5 + val_ins = [NavIns(NavInsID.TOUCH, (200, 350)), + NavIns(NavInsID.TOUCH, (200, 40 + 90 * select_user_idx)), + NavInsID.USE_CASE_CHOICE_CONFIRM] + else: + val_ins = [NavInsID.USE_CASE_CHOICE_CONFIRM] navigate(self.navigator, user_accept, @@ -231,90 +260,6 @@ def get_assertion(self, rp_id, client_data_hash, allow_list=None, response = self.parse_response(response) - if allow_list and len(allow_list) == 1 and TAG_RESP_CREDENTIAL not in response: - # Credential may be omitted if the allowList has exactly one Credential. - # But AssertionResponse() class doesn't support it. - # So we are patching it here by adding the credential in the response. - response[1] = allow_list[0] - - return AssertionResponse.from_dict(response) - - def get_assertion_with_txSimpleAuth(self, rp_id, client_data_hash, allow_list=None, - extensions=None, options=None, pin_uv_param=None, - pin_uv_protocol=None, *, event=None, on_keepalive=None, - login_type="simple", user_accept=True, check_users=None, - compare_args=None, select_user_idx=1): - # Refresh navigator screen content reference - self.navigator._backend.get_current_screen_content() - - assert login_type in ["simple", "multi"] - """ - Copy of get_assertion() to keep it simpler as the ux is different with txSimpleAuth - and we don't want to merge he two function as this will create a mess for a somehow - deprecated extension. - """ - TAG_RESP_CREDENTIAL = 0x01 - - cmd = Ctap2.CMD.GET_ASSERTION - data = args(rp_id, - client_data_hash, - allow_list, - extensions, - options, - pin_uv_param, - pin_uv_protocol) - - ctap_hid_cmd = self.send_cbor_nowait(cmd, data, event=event, - on_keepalive=on_keepalive) - - text = None - nav_ins = None - val_ins = None - - if self.model.startswith("nano"): - nav_ins = NavInsID.RIGHT_CLICK - val_ins = [NavInsID.BOTH_CLICK] - if user_accept is not None: - if login_type == "multi": - if check_users and len(check_users) == 1: - raise ValueError("Found 1 user while expecting multiple") - - if user_accept: - text = f"Log in user {select_user_idx}/" - else: - text = "Reject" - - else: - if check_users and len(check_users) != 1: - raise ValueError("Found multiple users while expecting 1") - - if user_accept: - text = "Log in" - else: - text = "Reject" - - navigate(self.navigator, - user_accept, - True, # Always check screen - False, # Never check cancel - compare_args, - text, - nav_ins, - val_ins) - - response = self.device.recv(ctap_hid_cmd) - - if user_accept is not None: - self.wait_for_return_on_dashboard() - - response = self.parse_response(response) - - if allow_list and len(allow_list) == 1 and TAG_RESP_CREDENTIAL not in response: - # Credential may be omitted if the allowList has exactly one Credential. - # But AssertionResponse() class doesn't support it. - # So we are patching it here by adding the credential in the response. - response[1] = allow_list[0] - return AssertionResponse.from_dict(response) def reset(self, *, event=None, on_keepalive=None, user_accept=True, @@ -337,6 +282,12 @@ def reset(self, *, event=None, on_keepalive=None, user_accept=True, text = "Yes, delete" else: text = "No, don't delete" + elif self.model == "stax": + if user_accept is not None: + if not user_accept: + val_ins = [NavInsID.USE_CASE_CHOICE_REJECT] + else: + val_ins = [NavInsID.USE_CASE_CHOICE_CONFIRM] navigate(self.navigator, user_accept, diff --git a/tests/speculos/fido2/test_extension_hmac_secret.py b/tests/speculos/fido2/test_extension_hmac_secret.py index 118ae84a..429225af 100644 --- a/tests/speculos/fido2/test_extension_hmac_secret.py +++ b/tests/speculos/fido2/test_extension_hmac_secret.py @@ -39,6 +39,18 @@ def test_extensions_hmac_secret(client): hmac_secret1 = hmac_ext.process_get_output(assertion)["hmacGetSecret"] + # Retrieve another assertion with same salt but with UV + options = {"uv": True} + assertion = client.ctap2.get_assertion(rp["id"], client_data_hash, + allow_list, + options=options, + extensions=extensions) + + hmac_secret2 = hmac_ext.process_get_output(assertion)["hmacGetSecret"] + + # Compare the outputs + assert hmac_secret1["output1"] != hmac_secret2["output1"] + # Retrieve another assertion with two salts salt2 = generate_random_bytes(32) hmac_ext_data = hmac_ext.process_get_input({ diff --git a/tests/speculos/fido2/test_extension_txAuthSimple.py b/tests/speculos/fido2/test_extension_txAuthSimple.py deleted file mode 100644 index 359e596a..00000000 --- a/tests/speculos/fido2/test_extension_txAuthSimple.py +++ /dev/null @@ -1,132 +0,0 @@ -import pytest -import sys - -from fido2.ctap import CtapError -from fido2.webauthn import AuthenticatorData - -from client import TESTS_SPECULOS_DIR -from utils import generate_random_bytes, generate_get_assertion_params - -MAX_TX_AUTH_SIMPLE_SIZE = 200 - - -@pytest.mark.skipif( - "--fast" in sys.argv, - reason="running in fast mode", -) -def test_extensions_txAuthSimple(client, test_name): - info = client.ctap2.info - assert "txAuthSimple" in info.extensions - - # Create a credential - rp, credential_data, user = generate_get_assertion_params(client, ref=0) - - client_data_hash = generate_random_bytes(32) - message_list = ["Pretty short message", "Pretty short message" * 5, "Pretty short message" * 20] - for idx, message in enumerate(message_list): - compare_args = (TESTS_SPECULOS_DIR, test_name + "/" + str(idx)) - allow_list = [{"id": credential_data.credential_id, "type": "public-key"}] - extensions = {"txAuthSimple": message} - assertion = client.ctap2.get_assertion_with_txSimpleAuth(rp["id"], client_data_hash, - allow_list, extensions, - check_users=[user], - compare_args=compare_args) - - assertion.verify(client_data_hash, credential_data.public_key) - - expected_flags = AuthenticatorData.FLAG.USER_PRESENT - expected_flags |= AuthenticatorData.FLAG.EXTENSION_DATA - assert assertion.auth_data.flags == expected_flags - - if len(message) < MAX_TX_AUTH_SIMPLE_SIZE: - assert assertion.auth_data.extensions["txAuthSimple"] == message - else: - assert assertion.auth_data.extensions["txAuthSimple"] == '' - - # We are doing dangerous things with txAuthSimple: - # - adding an '\0' at the end in the CBOR buffer - # So check that we still can process the rest of the buffer correctly.. - - # check parsing of uv - client_data_hash = generate_random_bytes(32) - message = "Pretty short message" - allow_list = [{"id": credential_data.credential_id, "type": "public-key"}] - extensions = {"txAuthSimple": message} - options = {"uv": True} - compare_args = (TESTS_SPECULOS_DIR, test_name + "/" + str(len(message_list))) - assertion = client.ctap2.get_assertion_with_txSimpleAuth(rp["id"], client_data_hash, - allow_list, extensions, options, - check_users=[user], - compare_args=compare_args) - - assertion.verify(client_data_hash, credential_data.public_key) - expected_flags = AuthenticatorData.FLAG.USER_PRESENT - expected_flags |= AuthenticatorData.FLAG.USER_VERIFIED - expected_flags |= AuthenticatorData.FLAG.EXTENSION_DATA - assert assertion.auth_data.flags == expected_flags - assert assertion.auth_data.extensions["txAuthSimple"] == message - - -def test_extensions_txAuthSimple_user_refused(client, test_name): - compare_args = (TESTS_SPECULOS_DIR, test_name) - # Create a credential - rp, credential_data, user = generate_get_assertion_params(client, ref=0) - - client_data_hash = generate_random_bytes(32) - message = "Pretty short message" - - allow_list = [{"id": credential_data.credential_id, "type": "public-key"}] - extensions = {"txAuthSimple": message} - - with pytest.raises(CtapError) as e: - client.ctap2.get_assertion_with_txSimpleAuth(rp["id"], client_data_hash, - allow_list, extensions, - user_accept=False, - check_users=[user], - compare_args=compare_args) - assert e.value.code == CtapError.ERR.OPERATION_DENIED - - -def test_extensions_txAuthSimple_multiple(client, test_name): - allow_list = [] - registered_users = [] - users_credential_data = [] - rp = None - - # Register 3 users for the same RP - for idx in range(3): - rp, credential_data, user = generate_get_assertion_params(client, rp=rp, ref=idx) - allow_list.append({"id": credential_data.credential_id, "type": "public-key"}) - users_credential_data.append(credential_data) - registered_users.append(user) - - client_data_hash = generate_random_bytes(32) - message = "Pretty short message" - extensions = {"txAuthSimple": message} - - # Try without user accept - compare_args = (TESTS_SPECULOS_DIR, test_name + "/refused") - with pytest.raises(CtapError) as e: - client.ctap2.get_assertion_with_txSimpleAuth(rp["id"], client_data_hash, - allow_list, extensions, - login_type="multi", - user_accept=False, - check_users=registered_users, - compare_args=compare_args) - assert e.value.code == CtapError.ERR.OPERATION_DENIED - - # Try with user accept - compare_args = (TESTS_SPECULOS_DIR, test_name + "/accepted") - assertion = client.ctap2.get_assertion_with_txSimpleAuth(rp["id"], client_data_hash, - allow_list, extensions, - login_type="multi", - check_users=registered_users, - compare_args=compare_args) - - assertion.verify(client_data_hash, users_credential_data[0].public_key) - - assert assertion.auth_data.extensions["txAuthSimple"] == message - -# Todo add tests with -# - Validation of request: -# - CBOR fields errors: missing / bad type / bad length... diff --git a/tests/speculos/fido2/test_fido2_screens.py b/tests/speculos/fido2/test_fido2_screens.py index a8bcac8a..8091848e 100644 --- a/tests/speculos/fido2/test_fido2_screens.py +++ b/tests/speculos/fido2/test_fido2_screens.py @@ -8,45 +8,64 @@ from utils import generate_make_credentials_params from utils import HAVE_RK_SUPPORT_SETTING -from ragger.navigator import NavInsID +from ragger.navigator import NavInsID, NavIns @pytest.mark.skipif(not HAVE_RK_SUPPORT_SETTING, reason="settings not enable") def test_fido_screens_settings(client, test_name): - instructions = [] - # Screen 0 -> 1 - instructions.append(NavInsID.RIGHT_CLICK) - # Screen 1 -> 2 - instructions.append(NavInsID.RIGHT_CLICK) - # Screen 2 -> 3 - instructions.append(NavInsID.RIGHT_CLICK) - # enter settings - instructions.append(NavInsID.BOTH_CLICK) - - # Enable and check "Enabling" warning message - instructions.append(NavInsID.BOTH_CLICK) - # Screen 0 -> 1 - instructions.append(NavInsID.RIGHT_CLICK) - # Screen 1 -> 2 - instructions.append(NavInsID.RIGHT_CLICK) - # Screen 2 -> 3 - instructions.append(NavInsID.RIGHT_CLICK) - # Screen 3 -> 4 - instructions.append(NavInsID.RIGHT_CLICK) - # Screen 4 -> 5 - instructions.append(NavInsID.RIGHT_CLICK) - # Confirm - instructions.append(NavInsID.BOTH_CLICK) - - # Disable - instructions.append(NavInsID.BOTH_CLICK) - - # leave settings - # Screen 0 -> 1 - instructions.append(NavInsID.RIGHT_CLICK) - # confirm - instructions.append(NavInsID.BOTH_CLICK) + + if client.model.startswith("nano"): + instructions = [] + # Screen 0 -> 1 + instructions.append(NavInsID.RIGHT_CLICK) + # Screen 1 -> 2 + instructions.append(NavInsID.RIGHT_CLICK) + # Screen 2 -> 3 + instructions.append(NavInsID.RIGHT_CLICK) + # enter settings + instructions.append(NavInsID.BOTH_CLICK) + + # Enable and check "Enabling" warning message + instructions.append(NavInsID.BOTH_CLICK) + # Screen 0 -> 1 + instructions.append(NavInsID.RIGHT_CLICK) + # Screen 1 -> 2 + instructions.append(NavInsID.RIGHT_CLICK) + # Screen 2 -> 3 + instructions.append(NavInsID.RIGHT_CLICK) + # Screen 3 -> 4 + instructions.append(NavInsID.RIGHT_CLICK) + # Screen 4 -> 5 + instructions.append(NavInsID.RIGHT_CLICK) + # Confirm + instructions.append(NavInsID.BOTH_CLICK) + + # Disable + instructions.append(NavInsID.BOTH_CLICK) + + # leave settings + # Screen 0 -> 1 + instructions.append(NavInsID.RIGHT_CLICK) + # confirm + instructions.append(NavInsID.BOTH_CLICK) + else: + instructions = [ + # Enter in the settings + NavInsID.USE_CASE_HOME_SETTINGS, + NavInsID.USE_CASE_SETTINGS_NEXT, + + # Enable and skip "Enabling" message + NavIns(NavInsID.CHOICE_CHOOSE, (1,)), + NavInsID.USE_CASE_CHOICE_CONFIRM, + NavInsID.USE_CASE_STATUS_DISMISS, + + # Disable + NavIns(NavInsID.CHOICE_CHOOSE, (1,)), + + # Leave settings + NavInsID.USE_CASE_SETTINGS_MULTI_PAGE_EXIT, + ] client.navigator.navigate_and_compare(TESTS_SPECULOS_DIR, test_name, instructions, screen_change_before_first_instruction=False) diff --git a/tests/speculos/fido2/test_get_info.py b/tests/speculos/fido2/test_get_info.py index c8bb75be..9c3230d5 100644 --- a/tests/speculos/fido2/test_get_info.py +++ b/tests/speculos/fido2/test_get_info.py @@ -23,8 +23,7 @@ def test_get_info_extensions(client): info = client.ctap2.info assert "hmac-secret" in info.extensions - assert "txAuthSimple" in info.extensions - assert len(info.extensions) == 2 + assert len(info.extensions) == 1 def test_get_info_aaguid(client): @@ -42,6 +41,10 @@ def test_get_info_aaguid(client): hs = sha256("Ledger FIDO 2 1.0 NanoS+".encode('utf-8')).hexdigest() hs = hs[:32] # Keep only the 16 first bytes assert hs == info.aaguid.hex() + elif client.model == "stax": + hs = sha256("Ledger FIDO 2 1.0 Stax".encode('utf-8')).hexdigest() + hs = hs[:32] # Keep only the 16 first bytes + assert hs == info.aaguid.hex() else: raise ValueError("Unhandled model") diff --git a/tests/speculos/fido2/test_option_rk.py b/tests/speculos/fido2/test_option_rk.py index df903c85..a84123ef 100644 --- a/tests/speculos/fido2/test_option_rk.py +++ b/tests/speculos/fido2/test_option_rk.py @@ -9,7 +9,7 @@ from utils import generate_get_assertion_params from utils import HAVE_RK_SUPPORT_SETTING -from ragger.navigator import NavInsID +from ragger.navigator import NavInsID, NavIns @pytest.mark.skipif(not HAVE_RK_SUPPORT_SETTING, @@ -39,26 +39,42 @@ def enable_rk_option(client): if not HAVE_RK_SUPPORT_SETTING: raise ValueError("rk and setting not enabled") - instructions = [ - # Enter in the settings - NavInsID.RIGHT_CLICK, - NavInsID.RIGHT_CLICK, - NavInsID.RIGHT_CLICK, - NavInsID.BOTH_CLICK, - - # Enable and skip "Enabling" message - NavInsID.BOTH_CLICK, - NavInsID.RIGHT_CLICK, - NavInsID.RIGHT_CLICK, - NavInsID.RIGHT_CLICK, - NavInsID.RIGHT_CLICK, - NavInsID.RIGHT_CLICK, - NavInsID.BOTH_CLICK, - - # Leave settings - NavInsID.RIGHT_CLICK, - NavInsID.BOTH_CLICK - ] + if client.model.startswith("nano"): + instructions = [ + # Enter in the settings + NavInsID.RIGHT_CLICK, + NavInsID.RIGHT_CLICK, + NavInsID.RIGHT_CLICK, + NavInsID.BOTH_CLICK, + + # Enable and skip "Enabling" message + NavInsID.BOTH_CLICK, + NavInsID.RIGHT_CLICK, + NavInsID.RIGHT_CLICK, + NavInsID.RIGHT_CLICK, + NavInsID.RIGHT_CLICK, + NavInsID.RIGHT_CLICK, + NavInsID.BOTH_CLICK, + + # Leave settings + NavInsID.RIGHT_CLICK, + NavInsID.BOTH_CLICK + ] + else: + instructions = [ + # Enter in the settings + NavInsID.USE_CASE_HOME_SETTINGS, + NavInsID.USE_CASE_SETTINGS_NEXT, + + # Enable and skip "Enabling" message + NavIns(NavInsID.CHOICE_CHOOSE, (1,)), + NavInsID.USE_CASE_CHOICE_CONFIRM, + NavInsID.USE_CASE_STATUS_DISMISS, + + # Leave settings + NavInsID.USE_CASE_SETTINGS_MULTI_PAGE_EXIT, + ] + client.navigator.navigate(instructions, screen_change_before_first_instruction=False) diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/0/00000.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/0/00000.png deleted file mode 100644 index c497c8d9..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/0/00000.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/0/00001.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/0/00001.png deleted file mode 100644 index 01f7f83f..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/0/00001.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/0/00002.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/0/00002.png deleted file mode 100644 index 5055d3f2..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/0/00002.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/0/00003.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/0/00003.png deleted file mode 100644 index 64faf5be..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/0/00003.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/0/00004.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/0/00004.png deleted file mode 100644 index a7bb5944..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/0/00004.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/1/00000.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/1/00000.png deleted file mode 100644 index c497c8d9..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/1/00000.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/1/00001.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/1/00001.png deleted file mode 100644 index e6f2764f..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/1/00001.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/1/00002.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/1/00002.png deleted file mode 100644 index 22a3cd72..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/1/00002.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/1/00003.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/1/00003.png deleted file mode 100644 index 5055d3f2..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/1/00003.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/1/00004.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/1/00004.png deleted file mode 100644 index 64faf5be..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/1/00004.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/1/00005.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/1/00005.png deleted file mode 100644 index a7bb5944..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/1/00005.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/2/00000.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/2/00000.png deleted file mode 100644 index c497c8d9..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/2/00000.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/2/00001.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/2/00001.png deleted file mode 100644 index a70d53f6..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/2/00001.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/2/00002.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/2/00002.png deleted file mode 100644 index f6281e07..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/2/00002.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/2/00003.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/2/00003.png deleted file mode 100644 index cf491279..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/2/00003.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/2/00004.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/2/00004.png deleted file mode 100644 index 808241d7..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/2/00004.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/2/00005.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/2/00005.png deleted file mode 100644 index cf2bf14c..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/2/00005.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/2/00006.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/2/00006.png deleted file mode 100644 index 8f0a8a72..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/2/00006.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/2/00007.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/2/00007.png deleted file mode 100644 index fc71cff8..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/2/00007.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/2/00008.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/2/00008.png deleted file mode 100644 index 5055d3f2..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/2/00008.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/2/00009.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/2/00009.png deleted file mode 100644 index 64faf5be..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/2/00009.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/2/00010.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/2/00010.png deleted file mode 100644 index a7bb5944..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/2/00010.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/3/00000.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/3/00000.png deleted file mode 100644 index c497c8d9..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/3/00000.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/3/00001.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/3/00001.png deleted file mode 100644 index 01f7f83f..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/3/00001.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/3/00002.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/3/00002.png deleted file mode 100644 index 5055d3f2..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/3/00002.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/3/00003.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/3/00003.png deleted file mode 100644 index 64faf5be..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/3/00003.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/3/00004.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/3/00004.png deleted file mode 100644 index a7bb5944..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple/3/00004.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_multiple/accepted/00000.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_multiple/accepted/00000.png deleted file mode 100644 index d12afcf3..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_multiple/accepted/00000.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_multiple/accepted/00001.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_multiple/accepted/00001.png deleted file mode 100644 index 01f7f83f..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_multiple/accepted/00001.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_multiple/accepted/00002.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_multiple/accepted/00002.png deleted file mode 100644 index 5055d3f2..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_multiple/accepted/00002.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_multiple/accepted/00003.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_multiple/accepted/00003.png deleted file mode 100644 index cd65a598..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_multiple/accepted/00003.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_multiple/refused/00000.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_multiple/refused/00000.png deleted file mode 100644 index d12afcf3..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_multiple/refused/00000.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_multiple/refused/00001.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_multiple/refused/00001.png deleted file mode 100644 index 01f7f83f..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_multiple/refused/00001.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_multiple/refused/00002.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_multiple/refused/00002.png deleted file mode 100644 index 5055d3f2..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_multiple/refused/00002.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_multiple/refused/00003.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_multiple/refused/00003.png deleted file mode 100644 index cd65a598..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_multiple/refused/00003.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_multiple/refused/00004.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_multiple/refused/00004.png deleted file mode 100644 index 35f02b5e..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_multiple/refused/00004.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_multiple/refused/00005.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_multiple/refused/00005.png deleted file mode 100644 index 4c08ccc9..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_multiple/refused/00005.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_multiple/refused/00006.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_multiple/refused/00006.png deleted file mode 100644 index 7a8040fa..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_multiple/refused/00006.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_user_refused/00000.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_user_refused/00000.png deleted file mode 100644 index c497c8d9..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_user_refused/00000.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_user_refused/00001.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_user_refused/00001.png deleted file mode 100644 index 01f7f83f..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_user_refused/00001.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_user_refused/00002.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_user_refused/00002.png deleted file mode 100644 index 5055d3f2..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_user_refused/00002.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_user_refused/00003.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_user_refused/00003.png deleted file mode 100644 index 64faf5be..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_user_refused/00003.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_user_refused/00004.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_user_refused/00004.png deleted file mode 100644 index a7bb5944..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_user_refused/00004.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_user_refused/00005.png b/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_user_refused/00005.png deleted file mode 100644 index 7a8040fa..00000000 Binary files a/tests/speculos/snapshots/nanosp/test_extensions_txAuthSimple_user_refused/00005.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanosp/test_fido_screens_settings/00002.png b/tests/speculos/snapshots/nanosp/test_fido_screens_settings/00002.png index 9328353a..12341d7d 100644 Binary files a/tests/speculos/snapshots/nanosp/test_fido_screens_settings/00002.png and b/tests/speculos/snapshots/nanosp/test_fido_screens_settings/00002.png differ diff --git a/tests/speculos/snapshots/nanosp/test_u2f_screens_idle/00002.png b/tests/speculos/snapshots/nanosp/test_u2f_screens_idle/00002.png index 9328353a..12341d7d 100644 Binary files a/tests/speculos/snapshots/nanosp/test_u2f_screens_idle/00002.png and b/tests/speculos/snapshots/nanosp/test_u2f_screens_idle/00002.png differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/0/00000.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/0/00000.png deleted file mode 100644 index ac0e630e..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/0/00000.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/0/00001.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/0/00001.png deleted file mode 100644 index 3707371c..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/0/00001.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/0/00002.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/0/00002.png deleted file mode 100644 index 5055d3f2..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/0/00002.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/0/00003.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/0/00003.png deleted file mode 100644 index 64faf5be..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/0/00003.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/0/00004.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/0/00004.png deleted file mode 100644 index 2751bcf4..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/0/00004.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/1/00000.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/1/00000.png deleted file mode 100644 index ac0e630e..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/1/00000.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/1/00001.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/1/00001.png deleted file mode 100644 index f17e74c9..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/1/00001.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/1/00002.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/1/00002.png deleted file mode 100644 index ceeeaf72..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/1/00002.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/1/00003.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/1/00003.png deleted file mode 100644 index 5055d3f2..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/1/00003.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/1/00004.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/1/00004.png deleted file mode 100644 index 64faf5be..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/1/00004.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/1/00005.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/1/00005.png deleted file mode 100644 index 2751bcf4..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/1/00005.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/2/00000.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/2/00000.png deleted file mode 100644 index ac0e630e..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/2/00000.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/2/00001.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/2/00001.png deleted file mode 100644 index b42dcc1c..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/2/00001.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/2/00002.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/2/00002.png deleted file mode 100644 index c7c6e965..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/2/00002.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/2/00003.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/2/00003.png deleted file mode 100644 index 95d1da19..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/2/00003.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/2/00004.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/2/00004.png deleted file mode 100644 index 6667040d..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/2/00004.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/2/00005.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/2/00005.png deleted file mode 100644 index 07a4b75e..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/2/00005.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/2/00006.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/2/00006.png deleted file mode 100644 index 7d7a0a08..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/2/00006.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/2/00007.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/2/00007.png deleted file mode 100644 index d49a307c..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/2/00007.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/2/00008.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/2/00008.png deleted file mode 100644 index 5055d3f2..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/2/00008.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/2/00009.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/2/00009.png deleted file mode 100644 index 64faf5be..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/2/00009.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/2/00010.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/2/00010.png deleted file mode 100644 index 2751bcf4..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/2/00010.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/3/00000.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/3/00000.png deleted file mode 100644 index ac0e630e..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/3/00000.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/3/00001.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/3/00001.png deleted file mode 100644 index 3707371c..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/3/00001.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/3/00002.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/3/00002.png deleted file mode 100644 index 5055d3f2..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/3/00002.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/3/00003.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/3/00003.png deleted file mode 100644 index 64faf5be..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/3/00003.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/3/00004.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/3/00004.png deleted file mode 100644 index 2751bcf4..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple/3/00004.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_multiple/accepted/00000.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_multiple/accepted/00000.png deleted file mode 100644 index 76c6f028..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_multiple/accepted/00000.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_multiple/accepted/00001.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_multiple/accepted/00001.png deleted file mode 100644 index 3707371c..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_multiple/accepted/00001.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_multiple/accepted/00002.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_multiple/accepted/00002.png deleted file mode 100644 index 5055d3f2..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_multiple/accepted/00002.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_multiple/accepted/00003.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_multiple/accepted/00003.png deleted file mode 100644 index 3c1486dc..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_multiple/accepted/00003.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_multiple/refused/00000.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_multiple/refused/00000.png deleted file mode 100644 index 76c6f028..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_multiple/refused/00000.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_multiple/refused/00001.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_multiple/refused/00001.png deleted file mode 100644 index 3707371c..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_multiple/refused/00001.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_multiple/refused/00002.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_multiple/refused/00002.png deleted file mode 100644 index 5055d3f2..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_multiple/refused/00002.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_multiple/refused/00003.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_multiple/refused/00003.png deleted file mode 100644 index 3c1486dc..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_multiple/refused/00003.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_multiple/refused/00004.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_multiple/refused/00004.png deleted file mode 100644 index faaef233..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_multiple/refused/00004.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_multiple/refused/00005.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_multiple/refused/00005.png deleted file mode 100644 index 1df24028..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_multiple/refused/00005.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_multiple/refused/00006.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_multiple/refused/00006.png deleted file mode 100644 index 503abd20..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_multiple/refused/00006.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_user_refused/00000.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_user_refused/00000.png deleted file mode 100644 index ac0e630e..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_user_refused/00000.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_user_refused/00001.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_user_refused/00001.png deleted file mode 100644 index 3707371c..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_user_refused/00001.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_user_refused/00002.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_user_refused/00002.png deleted file mode 100644 index 5055d3f2..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_user_refused/00002.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_user_refused/00003.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_user_refused/00003.png deleted file mode 100644 index 64faf5be..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_user_refused/00003.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_user_refused/00004.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_user_refused/00004.png deleted file mode 100644 index 2751bcf4..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_user_refused/00004.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_user_refused/00005.png b/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_user_refused/00005.png deleted file mode 100644 index 503abd20..00000000 Binary files a/tests/speculos/snapshots/nanox/test_extensions_txAuthSimple_user_refused/00005.png and /dev/null differ diff --git a/tests/speculos/snapshots/nanox/test_fido_screens_settings/00002.png b/tests/speculos/snapshots/nanox/test_fido_screens_settings/00002.png index 9328353a..12341d7d 100644 Binary files a/tests/speculos/snapshots/nanox/test_fido_screens_settings/00002.png and b/tests/speculos/snapshots/nanox/test_fido_screens_settings/00002.png differ diff --git a/tests/speculos/snapshots/nanox/test_u2f_screens_idle/00002.png b/tests/speculos/snapshots/nanox/test_u2f_screens_idle/00002.png index 9328353a..12341d7d 100644 Binary files a/tests/speculos/snapshots/nanox/test_u2f_screens_idle/00002.png and b/tests/speculos/snapshots/nanox/test_u2f_screens_idle/00002.png differ diff --git a/tests/speculos/snapshots/stax/test_authenticate_ok/00000.png b/tests/speculos/snapshots/stax/test_authenticate_ok/00000.png new file mode 100644 index 00000000..4f831132 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_authenticate_ok/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_authenticate_user_refused/00000.png b/tests/speculos/snapshots/stax/test_authenticate_user_refused/00000.png new file mode 100644 index 00000000..4f831132 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_authenticate_user_refused/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_fido2_screens_short_id/get/00000.png b/tests/speculos/snapshots/stax/test_fido2_screens_short_id/get/00000.png new file mode 100644 index 00000000..c9459bee Binary files /dev/null and b/tests/speculos/snapshots/stax/test_fido2_screens_short_id/get/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_fido2_screens_short_id/make/00000.png b/tests/speculos/snapshots/stax/test_fido2_screens_short_id/make/00000.png new file mode 100644 index 00000000..90972db4 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_fido2_screens_short_id/make/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_fido2_screens_user_display_name/get/00000.png b/tests/speculos/snapshots/stax/test_fido2_screens_user_display_name/get/00000.png new file mode 100644 index 00000000..774b948a Binary files /dev/null and b/tests/speculos/snapshots/stax/test_fido2_screens_user_display_name/get/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_fido2_screens_user_display_name/make/00000.png b/tests/speculos/snapshots/stax/test_fido2_screens_user_display_name/make/00000.png new file mode 100644 index 00000000..9fa51473 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_fido2_screens_user_display_name/make/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_fido2_screens_user_icon/get/00000.png b/tests/speculos/snapshots/stax/test_fido2_screens_user_icon/get/00000.png new file mode 100644 index 00000000..c9459bee Binary files /dev/null and b/tests/speculos/snapshots/stax/test_fido2_screens_user_icon/get/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_fido2_screens_user_icon/make/00000.png b/tests/speculos/snapshots/stax/test_fido2_screens_user_icon/make/00000.png new file mode 100644 index 00000000..90972db4 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_fido2_screens_user_icon/make/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_fido2_screens_user_name/get/00000.png b/tests/speculos/snapshots/stax/test_fido2_screens_user_name/get/00000.png new file mode 100644 index 00000000..43d97aad Binary files /dev/null and b/tests/speculos/snapshots/stax/test_fido2_screens_user_name/get/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_fido2_screens_user_name/make/00000.png b/tests/speculos/snapshots/stax/test_fido2_screens_user_name/make/00000.png new file mode 100644 index 00000000..eee3ae51 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_fido2_screens_user_name/make/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_fido2_screens_user_name_and_display_name/get/00000.png b/tests/speculos/snapshots/stax/test_fido2_screens_user_name_and_display_name/get/00000.png new file mode 100644 index 00000000..774b948a Binary files /dev/null and b/tests/speculos/snapshots/stax/test_fido2_screens_user_name_and_display_name/get/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_fido2_screens_user_name_and_display_name/make/00000.png b/tests/speculos/snapshots/stax/test_fido2_screens_user_name_and_display_name/make/00000.png new file mode 100644 index 00000000..9fa51473 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_fido2_screens_user_name_and_display_name/make/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_fido_screens_settings/00000.png b/tests/speculos/snapshots/stax/test_fido_screens_settings/00000.png new file mode 100644 index 00000000..3b0c2b11 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_fido_screens_settings/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_fido_screens_settings/00001.png b/tests/speculos/snapshots/stax/test_fido_screens_settings/00001.png new file mode 100644 index 00000000..79892f2f Binary files /dev/null and b/tests/speculos/snapshots/stax/test_fido_screens_settings/00001.png differ diff --git a/tests/speculos/snapshots/stax/test_fido_screens_settings/00002.png b/tests/speculos/snapshots/stax/test_fido_screens_settings/00002.png new file mode 100644 index 00000000..422f08df Binary files /dev/null and b/tests/speculos/snapshots/stax/test_fido_screens_settings/00002.png differ diff --git a/tests/speculos/snapshots/stax/test_fido_screens_settings/00003.png b/tests/speculos/snapshots/stax/test_fido_screens_settings/00003.png new file mode 100644 index 00000000..a70b2888 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_fido_screens_settings/00003.png differ diff --git a/tests/speculos/snapshots/stax/test_fido_screens_settings/00004.png b/tests/speculos/snapshots/stax/test_fido_screens_settings/00004.png new file mode 100644 index 00000000..643f1da5 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_fido_screens_settings/00004.png differ diff --git a/tests/speculos/snapshots/stax/test_fido_screens_settings/00005.png b/tests/speculos/snapshots/stax/test_fido_screens_settings/00005.png new file mode 100644 index 00000000..3da7058c Binary files /dev/null and b/tests/speculos/snapshots/stax/test_fido_screens_settings/00005.png differ diff --git a/tests/speculos/snapshots/stax/test_fido_screens_settings/00006.png b/tests/speculos/snapshots/stax/test_fido_screens_settings/00006.png new file mode 100644 index 00000000..422f08df Binary files /dev/null and b/tests/speculos/snapshots/stax/test_fido_screens_settings/00006.png differ diff --git a/tests/speculos/snapshots/stax/test_fido_screens_settings/00007.png b/tests/speculos/snapshots/stax/test_fido_screens_settings/00007.png new file mode 100644 index 00000000..3b0c2b11 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_fido_screens_settings/00007.png differ diff --git a/tests/speculos/snapshots/stax/test_get_assertion/00000.png b/tests/speculos/snapshots/stax/test_get_assertion/00000.png new file mode 100644 index 00000000..e422a642 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_get_assertion/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_get_assertion_allow_list/00000.png b/tests/speculos/snapshots/stax/test_get_assertion_allow_list/00000.png new file mode 100644 index 00000000..fec1390c Binary files /dev/null and b/tests/speculos/snapshots/stax/test_get_assertion_allow_list/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_get_assertion_allow_list/00001.png b/tests/speculos/snapshots/stax/test_get_assertion_allow_list/00001.png new file mode 100644 index 00000000..ec92e9bb Binary files /dev/null and b/tests/speculos/snapshots/stax/test_get_assertion_allow_list/00001.png differ diff --git a/tests/speculos/snapshots/stax/test_get_assertion_allow_list/00002.png b/tests/speculos/snapshots/stax/test_get_assertion_allow_list/00002.png new file mode 100644 index 00000000..3a7b8238 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_get_assertion_allow_list/00002.png differ diff --git a/tests/speculos/snapshots/stax/test_get_assertion_duplicate_allow_list_entries/00000.png b/tests/speculos/snapshots/stax/test_get_assertion_duplicate_allow_list_entries/00000.png new file mode 100644 index 00000000..e422a642 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_get_assertion_duplicate_allow_list_entries/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_get_assertion_no_credentials/00000.png b/tests/speculos/snapshots/stax/test_get_assertion_no_credentials/00000.png new file mode 100644 index 00000000..649cbc24 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_get_assertion_no_credentials/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_get_assertion_user_refused/00000.png b/tests/speculos/snapshots/stax/test_get_assertion_user_refused/00000.png new file mode 100644 index 00000000..e422a642 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_get_assertion_user_refused/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_get_assertion_uv/00000.png b/tests/speculos/snapshots/stax/test_get_assertion_uv/00000.png new file mode 100644 index 00000000..e422a642 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_get_assertion_uv/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_get_assertion_wrong_id/00000.png b/tests/speculos/snapshots/stax/test_get_assertion_wrong_id/00000.png new file mode 100644 index 00000000..649cbc24 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_get_assertion_wrong_id/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_get_assertion_wrong_rp/00000.png b/tests/speculos/snapshots/stax/test_get_assertion_wrong_rp/00000.png new file mode 100644 index 00000000..a39b5049 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_get_assertion_wrong_rp/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_make_credential/00000.png b/tests/speculos/snapshots/stax/test_make_credential/00000.png new file mode 100644 index 00000000..e987a9ff Binary files /dev/null and b/tests/speculos/snapshots/stax/test_make_credential/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_make_credential_certificate/00000.png b/tests/speculos/snapshots/stax/test_make_credential_certificate/00000.png new file mode 100644 index 00000000..e987a9ff Binary files /dev/null and b/tests/speculos/snapshots/stax/test_make_credential_certificate/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_make_credential_exclude_list/00000.png b/tests/speculos/snapshots/stax/test_make_credential_exclude_list/00000.png new file mode 100644 index 00000000..e987a9ff Binary files /dev/null and b/tests/speculos/snapshots/stax/test_make_credential_exclude_list/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_make_credential_up/00000.png b/tests/speculos/snapshots/stax/test_make_credential_up/00000.png new file mode 100644 index 00000000..e987a9ff Binary files /dev/null and b/tests/speculos/snapshots/stax/test_make_credential_up/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_make_credential_user_refused/00000.png b/tests/speculos/snapshots/stax/test_make_credential_user_refused/00000.png new file mode 100644 index 00000000..e987a9ff Binary files /dev/null and b/tests/speculos/snapshots/stax/test_make_credential_user_refused/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_make_credential_uv/00000.png b/tests/speculos/snapshots/stax/test_make_credential_uv/00000.png new file mode 100644 index 00000000..e987a9ff Binary files /dev/null and b/tests/speculos/snapshots/stax/test_make_credential_uv/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_option_rk_get_assertion/0/get_allow_list/00000.png b/tests/speculos/snapshots/stax/test_option_rk_get_assertion/0/get_allow_list/00000.png new file mode 100644 index 00000000..ba399e6b Binary files /dev/null and b/tests/speculos/snapshots/stax/test_option_rk_get_assertion/0/get_allow_list/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_option_rk_get_assertion/0/get_rk/00000.png b/tests/speculos/snapshots/stax/test_option_rk_get_assertion/0/get_rk/00000.png new file mode 100644 index 00000000..ba399e6b Binary files /dev/null and b/tests/speculos/snapshots/stax/test_option_rk_get_assertion/0/get_rk/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_option_rk_get_assertion/0/make/00000.png b/tests/speculos/snapshots/stax/test_option_rk_get_assertion/0/make/00000.png new file mode 100644 index 00000000..95d136f3 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_option_rk_get_assertion/0/make/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_option_rk_get_assertion/1/get_allow_list/00000.png b/tests/speculos/snapshots/stax/test_option_rk_get_assertion/1/get_allow_list/00000.png new file mode 100644 index 00000000..62620c05 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_option_rk_get_assertion/1/get_allow_list/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_option_rk_get_assertion/1/get_allow_list/00001.png b/tests/speculos/snapshots/stax/test_option_rk_get_assertion/1/get_allow_list/00001.png new file mode 100644 index 00000000..4ccfab54 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_option_rk_get_assertion/1/get_allow_list/00001.png differ diff --git a/tests/speculos/snapshots/stax/test_option_rk_get_assertion/1/get_allow_list/00002.png b/tests/speculos/snapshots/stax/test_option_rk_get_assertion/1/get_allow_list/00002.png new file mode 100644 index 00000000..62620c05 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_option_rk_get_assertion/1/get_allow_list/00002.png differ diff --git a/tests/speculos/snapshots/stax/test_option_rk_get_assertion/1/get_rk/00000.png b/tests/speculos/snapshots/stax/test_option_rk_get_assertion/1/get_rk/00000.png new file mode 100644 index 00000000..62620c05 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_option_rk_get_assertion/1/get_rk/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_option_rk_get_assertion/1/get_rk/00001.png b/tests/speculos/snapshots/stax/test_option_rk_get_assertion/1/get_rk/00001.png new file mode 100644 index 00000000..4ccfab54 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_option_rk_get_assertion/1/get_rk/00001.png differ diff --git a/tests/speculos/snapshots/stax/test_option_rk_get_assertion/1/get_rk/00002.png b/tests/speculos/snapshots/stax/test_option_rk_get_assertion/1/get_rk/00002.png new file mode 100644 index 00000000..62620c05 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_option_rk_get_assertion/1/get_rk/00002.png differ diff --git a/tests/speculos/snapshots/stax/test_option_rk_get_assertion/1/make/00000.png b/tests/speculos/snapshots/stax/test_option_rk_get_assertion/1/make/00000.png new file mode 100644 index 00000000..9b0d559e Binary files /dev/null and b/tests/speculos/snapshots/stax/test_option_rk_get_assertion/1/make/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_option_rk_get_assertion/2/get_allow_list/00000.png b/tests/speculos/snapshots/stax/test_option_rk_get_assertion/2/get_allow_list/00000.png new file mode 100644 index 00000000..4e39ed60 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_option_rk_get_assertion/2/get_allow_list/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_option_rk_get_assertion/2/get_allow_list/00001.png b/tests/speculos/snapshots/stax/test_option_rk_get_assertion/2/get_allow_list/00001.png new file mode 100644 index 00000000..c438d024 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_option_rk_get_assertion/2/get_allow_list/00001.png differ diff --git a/tests/speculos/snapshots/stax/test_option_rk_get_assertion/2/get_allow_list/00002.png b/tests/speculos/snapshots/stax/test_option_rk_get_assertion/2/get_allow_list/00002.png new file mode 100644 index 00000000..4e39ed60 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_option_rk_get_assertion/2/get_allow_list/00002.png differ diff --git a/tests/speculos/snapshots/stax/test_option_rk_get_assertion/2/get_rk/00000.png b/tests/speculos/snapshots/stax/test_option_rk_get_assertion/2/get_rk/00000.png new file mode 100644 index 00000000..4e39ed60 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_option_rk_get_assertion/2/get_rk/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_option_rk_get_assertion/2/get_rk/00001.png b/tests/speculos/snapshots/stax/test_option_rk_get_assertion/2/get_rk/00001.png new file mode 100644 index 00000000..c438d024 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_option_rk_get_assertion/2/get_rk/00001.png differ diff --git a/tests/speculos/snapshots/stax/test_option_rk_get_assertion/2/get_rk/00002.png b/tests/speculos/snapshots/stax/test_option_rk_get_assertion/2/get_rk/00002.png new file mode 100644 index 00000000..4e39ed60 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_option_rk_get_assertion/2/get_rk/00002.png differ diff --git a/tests/speculos/snapshots/stax/test_option_rk_get_assertion/2/make/00000.png b/tests/speculos/snapshots/stax/test_option_rk_get_assertion/2/make/00000.png new file mode 100644 index 00000000..88155763 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_option_rk_get_assertion/2/make/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_option_rk_make_cred_exclude_refused/00000.png b/tests/speculos/snapshots/stax/test_option_rk_make_cred_exclude_refused/00000.png new file mode 100644 index 00000000..e987a9ff Binary files /dev/null and b/tests/speculos/snapshots/stax/test_option_rk_make_cred_exclude_refused/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_option_rk_overwrite_get_assertion/1/get_assertion/00000.png b/tests/speculos/snapshots/stax/test_option_rk_overwrite_get_assertion/1/get_assertion/00000.png new file mode 100644 index 00000000..37106758 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_option_rk_overwrite_get_assertion/1/get_assertion/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_option_rk_overwrite_get_assertion/1/make/00000.png b/tests/speculos/snapshots/stax/test_option_rk_overwrite_get_assertion/1/make/00000.png new file mode 100644 index 00000000..27ee25b5 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_option_rk_overwrite_get_assertion/1/make/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_option_rk_overwrite_get_assertion/2/get_assertion/00000.png b/tests/speculos/snapshots/stax/test_option_rk_overwrite_get_assertion/2/get_assertion/00000.png new file mode 100644 index 00000000..567b6ee3 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_option_rk_overwrite_get_assertion/2/get_assertion/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_option_rk_overwrite_get_assertion/2/make/00000.png b/tests/speculos/snapshots/stax/test_option_rk_overwrite_get_assertion/2/make/00000.png new file mode 100644 index 00000000..55e66354 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_option_rk_overwrite_get_assertion/2/make/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_register_ok/00000.png b/tests/speculos/snapshots/stax/test_register_ok/00000.png new file mode 100644 index 00000000..8d190f17 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_register_ok/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_register_user_refused/00000.png b/tests/speculos/snapshots/stax/test_register_user_refused/00000.png new file mode 100644 index 00000000..8d190f17 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_register_user_refused/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_reset/False/00000.png b/tests/speculos/snapshots/stax/test_reset/False/00000.png new file mode 100644 index 00000000..bf508056 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_reset/False/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_reset/True/00000.png b/tests/speculos/snapshots/stax/test_reset/True/00000.png new file mode 100644 index 00000000..bf508056 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_reset/True/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/0_Binance/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/0_Binance/00000.png new file mode 100644 index 00000000..26953251 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/0_Binance/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/10_Fedora/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/10_Fedora/00000.png new file mode 100644 index 00000000..45d1635d Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/10_Fedora/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/11_Gandi/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/11_Gandi/00000.png new file mode 100644 index 00000000..e2a1f529 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/11_Gandi/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/12_GitHub/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/12_GitHub/00000.png new file mode 100644 index 00000000..f895eae1 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/12_GitHub/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/13_GitLab/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/13_GitLab/00000.png new file mode 100644 index 00000000..88ce6c14 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/13_GitLab/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/14_Google/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/14_Google/00000.png new file mode 100644 index 00000000..b93b3405 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/14_Google/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/15_Keeper/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/15_Keeper/00000.png new file mode 100644 index 00000000..e5bfaf2b Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/15_Keeper/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/16_LastPass/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/16_LastPass/00000.png new file mode 100644 index 00000000..f7520db2 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/16_LastPass/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/17_Slush Pool/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/17_Slush Pool/00000.png new file mode 100644 index 00000000..b26be14d Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/17_Slush Pool/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/18_Stripe/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/18_Stripe/00000.png new file mode 100644 index 00000000..be1b674e Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/18_Stripe/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/19_u2f.bin.coffee/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/19_u2f.bin.coffee/00000.png new file mode 100644 index 00000000..1416629d Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/19_u2f.bin.coffee/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/1_Bitbucket/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/1_Bitbucket/00000.png new file mode 100644 index 00000000..acde8ed5 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/1_Bitbucket/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/20_webauthn.bin.coffee/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/20_webauthn.bin.coffee/00000.png new file mode 100644 index 00000000..7af877bd Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/20_webauthn.bin.coffee/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/21_WebAuthn.io/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/21_WebAuthn.io/00000.png new file mode 100644 index 00000000..23f7aa08 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/21_WebAuthn.io/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/22_WebAuthn.me/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/22_WebAuthn.me/00000.png new file mode 100644 index 00000000..42d8c9b9 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/22_WebAuthn.me/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/23_demo.yubico.com/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/23_demo.yubico.com/00000.png new file mode 100644 index 00000000..733bcc81 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/23_demo.yubico.com/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/2_Bitfinex/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/2_Bitfinex/00000.png new file mode 100644 index 00000000..c3fb8983 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/2_Bitfinex/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/3_Bitwarden/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/3_Bitwarden/00000.png new file mode 100644 index 00000000..f89d7b3d Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/3_Bitwarden/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/4_Coinbase/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/4_Coinbase/00000.png new file mode 100644 index 00000000..a707f335 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/4_Coinbase/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/5_Dashlane/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/5_Dashlane/00000.png new file mode 100644 index 00000000..f929e484 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/5_Dashlane/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/6_Dropbox/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/6_Dropbox/00000.png new file mode 100644 index 00000000..15c2ca8a Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/6_Dropbox/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/7_Dropbox/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/7_Dropbox/00000.png new file mode 100644 index 00000000..ac24a0a7 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/7_Dropbox/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/8_Duo/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/8_Duo/00000.png new file mode 100644 index 00000000..6fef6aec Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/8_Duo/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/9_FastMail/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/9_FastMail/00000.png new file mode 100644 index 00000000..ead1a1f3 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/log/9_FastMail/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/0_Binance/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/0_Binance/00000.png new file mode 100644 index 00000000..f2f92145 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/0_Binance/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/10_Fedora/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/10_Fedora/00000.png new file mode 100644 index 00000000..2c4321f5 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/10_Fedora/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/11_Gandi/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/11_Gandi/00000.png new file mode 100644 index 00000000..fddc2c99 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/11_Gandi/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/12_GitHub/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/12_GitHub/00000.png new file mode 100644 index 00000000..b8c22b2c Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/12_GitHub/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/13_GitLab/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/13_GitLab/00000.png new file mode 100644 index 00000000..715858c1 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/13_GitLab/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/14_Google/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/14_Google/00000.png new file mode 100644 index 00000000..4acef417 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/14_Google/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/15_Keeper/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/15_Keeper/00000.png new file mode 100644 index 00000000..2e5aabb9 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/15_Keeper/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/16_LastPass/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/16_LastPass/00000.png new file mode 100644 index 00000000..011901f0 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/16_LastPass/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/17_Slush Pool/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/17_Slush Pool/00000.png new file mode 100644 index 00000000..c4ec1fff Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/17_Slush Pool/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/18_Stripe/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/18_Stripe/00000.png new file mode 100644 index 00000000..7a4d8a94 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/18_Stripe/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/19_u2f.bin.coffee/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/19_u2f.bin.coffee/00000.png new file mode 100644 index 00000000..4ff13e7f Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/19_u2f.bin.coffee/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/1_Bitbucket/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/1_Bitbucket/00000.png new file mode 100644 index 00000000..423cfd3a Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/1_Bitbucket/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/20_webauthn.bin.coffee/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/20_webauthn.bin.coffee/00000.png new file mode 100644 index 00000000..733a1ccf Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/20_webauthn.bin.coffee/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/21_WebAuthn.io/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/21_WebAuthn.io/00000.png new file mode 100644 index 00000000..fdc9675f Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/21_WebAuthn.io/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/22_WebAuthn.me/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/22_WebAuthn.me/00000.png new file mode 100644 index 00000000..0dcccf36 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/22_WebAuthn.me/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/23_demo.yubico.com/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/23_demo.yubico.com/00000.png new file mode 100644 index 00000000..316ea048 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/23_demo.yubico.com/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/2_Bitfinex/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/2_Bitfinex/00000.png new file mode 100644 index 00000000..f63cd614 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/2_Bitfinex/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/3_Bitwarden/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/3_Bitwarden/00000.png new file mode 100644 index 00000000..5cd86976 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/3_Bitwarden/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/4_Coinbase/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/4_Coinbase/00000.png new file mode 100644 index 00000000..fa8f1405 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/4_Coinbase/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/5_Dashlane/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/5_Dashlane/00000.png new file mode 100644 index 00000000..8840a0d4 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/5_Dashlane/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/6_Dropbox/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/6_Dropbox/00000.png new file mode 100644 index 00000000..009d565d Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/6_Dropbox/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/7_Dropbox/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/7_Dropbox/00000.png new file mode 100644 index 00000000..da4622d5 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/7_Dropbox/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/8_Duo/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/8_Duo/00000.png new file mode 100644 index 00000000..86819f52 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/8_Duo/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/9_FastMail/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/9_FastMail/00000.png new file mode 100644 index 00000000..965d26a3 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_fido_known_list/reg/9_FastMail/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_idle/00000.png b/tests/speculos/snapshots/stax/test_u2f_screens_idle/00000.png new file mode 100644 index 00000000..3b0c2b11 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_idle/00000.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_idle/00001.png b/tests/speculos/snapshots/stax/test_u2f_screens_idle/00001.png new file mode 100644 index 00000000..79892f2f Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_idle/00001.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_idle/00002.png b/tests/speculos/snapshots/stax/test_u2f_screens_idle/00002.png new file mode 100644 index 00000000..422f08df Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_idle/00002.png differ diff --git a/tests/speculos/snapshots/stax/test_u2f_screens_idle/00003.png b/tests/speculos/snapshots/stax/test_u2f_screens_idle/00003.png new file mode 100644 index 00000000..3b0c2b11 Binary files /dev/null and b/tests/speculos/snapshots/stax/test_u2f_screens_idle/00003.png differ diff --git a/tests/speculos/u2f/test_u2f_screens.py b/tests/speculos/u2f/test_u2f_screens.py index 01ca8a95..44303170 100644 --- a/tests/speculos/u2f/test_u2f_screens.py +++ b/tests/speculos/u2f/test_u2f_screens.py @@ -9,22 +9,29 @@ from utils import HAVE_RK_SUPPORT_SETTING -def test_u2f_screens_idle(client, test_name): +def test_u2f_screens_idle(client, test_name, firmware): # Refresh navigator screen content reference time.sleep(0.1) client.navigator._backend.get_current_screen_content() instructions = [] - # Screen 0 -> 1 - instructions.append(NavInsID.RIGHT_CLICK) - # Screen 1 -> 2 - instructions.append(NavInsID.RIGHT_CLICK) - # Screen 2 -> 3 - instructions.append(NavInsID.RIGHT_CLICK) - - if HAVE_RK_SUPPORT_SETTING: - # Screen 3 -> 4 + if firmware.device.startswith("nano"): + # Screen 0 -> 1 instructions.append(NavInsID.RIGHT_CLICK) + # Screen 1 -> 2 + instructions.append(NavInsID.RIGHT_CLICK) + # Screen 2 -> 3 + instructions.append(NavInsID.RIGHT_CLICK) + + if HAVE_RK_SUPPORT_SETTING: + # Screen 3 -> 4 + instructions.append(NavInsID.RIGHT_CLICK) + else: + instructions = [ + NavInsID.USE_CASE_HOME_SETTINGS, + NavInsID.USE_CASE_SETTINGS_NEXT, + NavInsID.USE_CASE_SETTINGS_MULTI_PAGE_EXIT + ] client.navigator.navigate_and_compare(TESTS_SPECULOS_DIR, test_name, instructions, screen_change_before_first_instruction=False) diff --git a/tests/speculos/utils.py b/tests/speculos/utils.py index b2932f1f..d9c752b3 100644 --- a/tests/speculos/utils.py +++ b/tests/speculos/utils.py @@ -7,6 +7,8 @@ from fido2.utils import sha256 from fido2.webauthn import AttestedCredentialData +from ragger.navigator import NavIns, NavInsID + # Application build configuration HAVE_NO_RESET_GENERATION_INCREMENT = True HAVE_RK_SUPPORT_SETTING = True @@ -16,26 +18,6 @@ "101112131415161718191a1b1c1d1e1f") -CRED_PARAMS = [ - ("webctap.myservice.com", - bytes.fromhex("000102030405060708090a0b0c0d0e0f" - "101112131415161718191a1b1c1d1e1f"), - "My user name"), - ("webctap.myservice_1.com", - bytes.fromhex("00000000000000000000000000000000" - "00000000000000000000000000000001"), - "My user 1 name"), - ("webctap.myservice_2.com", - bytes.fromhex("00000000000000000000000000000000" - "00000000000000000000000000000002"), - "My user 2 name"), - ("webctap.myservice_3.com", - bytes.fromhex("00000000000000000000000000000000" - "00000000000000000000000000000003"), - "My user 3 name") -] - - def prepare_apdu(cla=0, ins=0, p1=0, p2=0, data=b""): size = len(data) size_h = size >> 16 & 0xFF @@ -60,7 +42,16 @@ def generate_make_credentials_params(ref=None): user_id = generate_random_bytes(64) user_name = None else: - rp_id, user_id, user_name = CRED_PARAMS[ref] + if ref == 0: + rp_id = "webctap.myservice.com" + user_id = bytes.fromhex("000102030405060708090a0b0c0d0e0f" + "101112131415161718191a1b1c1d1e1f") + user_name = "My user name" + else: + rp_id = f"webctap.myservice_{ref}.com" + user_id = bytes.fromhex("00000000000000000000000000000000" + f"0000000000000000000000000000000{ref}") + user_name = f"My user {ref} name" client_data_hash = generate_random_bytes(32) rp = {"id": rp_id} @@ -163,14 +154,22 @@ def navigate(navigator, if user_accept is not None: # Over U2F endpoint (but not over HID) the device needs the # response to be retrieved before continuing the UX flow. - navigator.navigate_until_text_and_compare( - nav_ins, - val_ins, - text, - root, - test_name, - screen_change_after_last_instruction=False) + + if text: + navigator.navigate_until_text_and_compare( + nav_ins, + val_ins, + text, + root, + test_name, + screen_change_after_last_instruction=False) + else: + navigator.navigate_and_compare( + root, + test_name, + val_ins, + screen_change_after_last_instruction=False) elif check_cancel: - navigator.navigate([nav_ins], + navigator.navigate([NavIns(NavInsID.WAIT, (0.1,))], screen_change_after_last_instruction=False)