This is a sample implementation of a Flutter app with integration to Corbado API to use passkeys (based on FIDO2 / WebAuthn).
android/
: this directory contains the Android-specific files for the app. It includes theapp/
directory, which contains the source code and other files needed to build the Android app.ios/
: this directory contains the iOS-specific files for the app. It includes theRunner/
directory, which contains the source code and other files needed to build the iOS app.lib/
: this directory contains the Dart source code for the app. It includes themain.dart
file, which is the entry point for the app, as well as other Dart source files that make up the app.
Please follow the steps in Getting started to
create a project in our developer panel. In the root folder of this repo, create an env.json
file and copy the contents from env.json.scel
. Here, fill in your project ID.
Inside the developer panel, go to Settings -> Credentials -> Native Apps and click on 'Add new'. There you need to enter the package name (this sample application uses com.corbado.passkeys
as default package name) as well as the SHA-256 fingerprint of your signing key (e.g. 6H:A7:BC:9A:...). It can be obtained by executing gradlew signingReport
in the android/
directory. If you just want your local debug-key which is used when developing the app in Android Studio, use keytool -list -v -keystore ~/.android/debug.keystore -alias androiddebugkey -storepass android -keypass android
. Alternatively, you can look into the console when executing the app:
If you've successfully entered the package name and fingerprint in the developer panel, Corbado deploys a required /.well-known/assetlinks.json file to {project ID}.auth.corbado.com
. This makes the assetlinks.json
file publicly reachable and also binds your passkeys to {project ID}.auth.corbado.com
(the WebAuthn relying party). To use your own domain and bind the passkeys to it, please see the next step 3.2.
If you want to bind the passkeys to your own domain (e.g. your-domain.com
), you need to manually change the WebAuthn relying party and host the assetlinks.json
file on this domain (so on your-domain.com/.well-known/assetlinks.json
). To make it work with Corbado, you need to change the CNAME in the developer panel to a subdomain of your domain, e.g. auth.your-domain.com
. Corbado will then automatically bind your passkeys to the root / top-level domain of the provided CNAME unless it is on the public suffix list, here to your-domain.com
.
⚠️ Passkeys binding and sub-subdomain as CNAME: If the CNAME you enter in Corbado developer panel is a sub-subdomain, e.g.staging.auth.your-domain.com
, you still need to host theassetlinks.json
file on the root / top-level domain: hostassetlinks.json
atyour-domain.com/.well-known/assetlinks.json
, as your relying party ID is the root / top-level domainyour-domain.com
.
Moreover, you need to associate your native app in the assetlinks.json
file. Use the following JSON template and host it under
your-domain.com/.well-known/assetlinks.json
:
[
{
"relation": [
"delegate_permission/common.handle_all_urls",
"delegate_permission/common.get_login_creds"
],
"target": {
"namespace": "android_app",
"package_name": "{PACKAGE-NAME}",
"sha256_cert_fingerprints": ["{FINGERPRINT-OF-YOUR-SIGNING-KEY}"]
}
}
]
Variables:
- PACKAGE-NAME: The Android package name (this sample app uses com.corbado.passkeys)
- FINGERPRINT-OF-YOUR-SIGNING-KEY: The SHA-256 fingerprint obtained in step 3.1 (e.g. 7H:AC:4C:...).
You can use Google's tool to
verify that your assetlinks.json
file is set up and hosted correctly.
To let your Android app securely communicate with Corbado, you need to add the Android app's origin to the authorized origins in the developer panel. The Android app's origin is in the form of android:apk-key-hash:xxx
. You can obtain it by having a look at the "Base64URL encoded" output from step 3.1. That's your Android app origin.
Run the app by executing flutter run --dart-define-from-file=env.json
In case you run the app inside Android studio, edit the run configuration and
add --dart-define-from-file=env.json
to the
additional run args.
If the application says your device does not support biometrics yet, you have to properly setup biometrics on the phone.
Open the settings and add a PIN as well as a fingerprint as shown below (PIN is required for fingerprint).
Inside the developer panel, go to Settings -> Credentials -> Native Apps and click on 'Add new'. There you need to enter the app identifier prefix of your iOS application as well as the bundle identifier (this sample application uses com.corbado.passkeys
as default bundle identifier). The app identifier prefix can be obtained by going to your Apple Developer Certificates, Identifier & Profiles associated with your Apple Developer account, and finding the corresponding app identifier prefix.
If you've successfully entered the app identifier prefix and the bundle identifier in the developer panel, Corbado deploys a required /.well-known/apple-app-site-association file to {project ID}.auth.corbado.com
. This makes the apple-app-site-association.json
file publicly reachable and also binds your passkeys to {project ID}.auth.corbado.com
(the WebAuthn relying party). To use your own domain and bind the passkeys to it, please see the next step 4.2.
If you want to bind the passkeys to your own domain (e.g. your-domain.com
), you need to manually change the WebAuthn relying party and host the apple-app-site-association.json
file on this domain (so on your-domain.com/.well-known/apple-app-site-association
). To make it work with Corbado, you need to change the CNAME in the developer panel to a subdomain of your domain, e.g. auth.your-domain.com
. Corbado will then automatically bind your passkeys to the root / top-level domain of the provided CNAME unless it is on the public suffix list, here to your-domain.com
.
⚠️ Passkeys binding and sub-subdomain as CNAME: If the CNAME you enter in Corbado developer panel is a sub-subdomain, e.g.staging.auth.your-domain.com
, you still need to host theapple-app-site-association.json
file on the root / top-level domain: hostapple-app-site-association.json
atyour-domain.com/.well-known/apple-app-site-association
, as your relying party ID is the root / top-level domainyour-domain.com
.
Moreover, you need to associate your native app in the apple-app-site-association.json
file. Use the following JSON template and store it under
your-domain.com/.well-known/apple-app-site-association.json
:
{
"appclips": {
"apps": []
},
"applinks": {
"apps": []
},
"webcredentials": {
"apps": ["{ APP_IDENTIFIER_PREFIX }.{ BUNDLE_IDENTIFIER }"]
}
}
Variables:
- APP_IDENTIFIER_PREFIX: The iOS app identifier prefix associated with your development team in your Apple Developer account.
- BUNDLE_IDENTIFIER: The bundle identifier associated with your iOS application. Can be found in Xcode development environment.
To let your iOS app securely communicate with Corbado, you need to add the iOS app's origin to the authorized origins in the developer panel. The iOS app's origin is in the form of {PROJECT_ID}.auth.corbado.com
. You can obtain the corresponding project ID from developer pannel.
Run the app by executing flutter run --dart-define-from-file=env.json
In case you run the app inside Xcode, edit the run configuration and
add --dart-define-from-file=env.json
to the
additional run args.
This is a basic sample application that shows how passkeys can be integrated in a Flutter app using Corbado. This repository will be continuously updated and improved. We already have the following features on our roadmap that will be pushed within the next weeks:
- Automatic whitelisting of Android app's origin based on given package name and fingerprint
- Opt-out for email confirmation via email magic link
- Email magic links open the native Flutter app if openend on mobile device
- Better cross-device & cross-platform showcase with a web app
If you have questions, feedback or wishes regarding features, please reacht out to us via email or join our passkeys community on Slack. We're also happy to receive pull requests if you have suggestions for improvement.