Logo

dev-resources.site

for different kinds of informations.

How to Integrate Passkeys in Python (FastAPI)

Published at
8/31/2024
Categories
python
fastapi
cybersecurity
passkeys
Author
vdelitz
Author
7 person written this
vdelitz
open
How to Integrate Passkeys in Python (FastAPI)

Introduction

In this guide, we will walk you through the process of integrating passkey authentication into a Python web application using the FastAPI framework. This implementation leverages Corbado's passkey-first web-js package, which streamlines the integration with a passkeys backend. By the end of this tutorial, you will have a working FastAPI app with passkey-based authentication.

Read the full original tutorial here

Prerequisites for FastAPI Passkey Integration

To follow this tutorial, you should have a basic understanding of Python, FastAPI, HTML, and JavaScript. 
Additionally, you will need a Corbado account to use passkey services. Let's get started!

FastAPI Passkey Project Structure

Your FastAPI project will contain several key files. The essential ones include:



├── .env                 # Contains all environment variables
├── main.py              # Contains our webapplication (Handles routes)
└── templates
 ├── index.html       # Login page
 └── profile.html     # Profile page


Enter fullscreen mode Exit fullscreen mode

Setting Up Your Corbado Account

Before you begin coding, set up a Corbado account. This will allow you to access their passkey services. Follow these steps:

  1. Sign Up: Register on the Corbado developer panel.
  2. Create a Project: Name your project and select "Corbado Complete" during setup.
  3. Environment Setup: Choose "DEV" as your environment and "Web app" for the application type.
  4. Session Management: Opt for "Corbado session management" to handle both passkey authentication and session management.
  5. Frontend Framework: Select "Vanilla JS" as your frontend framework.
  6. Configuration: Set your Application URL (e.g., http://localhost:8000) and Relying Party ID (e.g., localhost).

After setting up, you will receive HTML/JavaScript snippets that you'll integrate into your FastAPI project.

Initializing the FastAPI Project

Start by creating a main.py file if you haven't already. Install FastAPI and other necessary packages with:



pip install fastapi python-dotenv passkeys


Enter fullscreen mode Exit fullscreen mode

Your main.py will handle the application logic, including setting up routes and managing sessions.

Configuring Environment Variables

In the root of your project, create a .env file to store your environment variables:



PROJECT_ID=your_project_id
API_SECRET=your_api_secret


Enter fullscreen mode Exit fullscreen mode

Load these variables into your application using python-dotenv:



from dotenv import load_dotenv
import os

load_dotenv()
PROJECT_ID = os.getenv("PROJECT_ID")
API_SECRET = os.getenv("API_SECRET")


Enter fullscreen mode Exit fullscreen mode

Creating HTML Templates with Session Management

Next, set up your HTML templates. In the templates/ directory, create login.html and profile.html. These will include the necessary JavaScript for integrating Corbado's authentication components.
Here is the basic structure for login.html:



<!DOCTYPE html>
<html>
<head>
    <link rel="stylesheet" href="https://unpkg.com/@corbado/web-js@latest/dist/bundle/index.css" />
    <script src="https://unpkg.com/@corbado/web-js@latest/dist/bundle/index.js"></script>
</head>
<body>
    <script>
        (async () => {
            await Corbado.load({
                projectId: "{{ PROJECT_ID }}",
                darkMode: "off",
                setShortSessionCookie: "true",
            });
            Corbado.mountAuthUI(document.getElementById('corbado-auth'), {
                onLoggedIn: () => window.location.href = '/profile',
            });
        })();
    </script>
    <div id="corbado-auth"></div>
</body>
</html>


Enter fullscreen mode Exit fullscreen mode

For the profile.html, include elements to display user data and a logout button:



<!DOCTYPE html>
<html>
  <head>
    <link
      rel="stylesheet"
      href="https://unpkg.com/@corbado/web-js@latest/dist/bundle/index.css"
      />
      <script src="https://unpkg.com/@corbado/web-js@latest/dist/bundle/index.js"></script>
    </head>
    <body>

      <!-- Define passkey-list div and logout button -->
    <h2>:/protected 🔒</h2>
    <p>User ID: {{USER_ID}}</p>
    <p>Name: {{USER_NAME}}</p>
    <p>Email: {{USER_EMAIL}}</p>
    <div id="passkey-list"></div>
    <button id="logoutButton">Logout</button>


    <!-- Script to load Corbado and mount PasskeyList UI -->
    <script>
 (async () => {
            await Corbado.load({
                projectId: "{{ PROJECT_ID }}",
                darkMode: "off",
                setShortSessionCookie: "true" // set short session cookie automatically
 }); 

            // Get and mount PasskeyList UI
            const passkeyListElement = document.getElementById("passkey-list"); // Element where you want to render PasskeyList UI
            Corbado.mountPasskeyListUI(passkeyListElement);

            // Get the logout button
            const logoutButton = document.getElementById('logoutButton');
            // Add event listener to logout button
 logoutButton.addEventListener('click', function() {
              Corbado.logout()
 .then(() => {
                        window.location.replace("/");
 })
 .catch(err => {
                        console.error(err);
 });
 });
 })();
    </script>


</body>
</html>


Enter fullscreen mode Exit fullscreen mode

Setting Up FastAPI Controller

Your controller logic will reside in the main.py file. This file will manage routes for both the login and profile pages. The login route will simply inject the PROJECT_ID into the template, while the profile route will validate the session and fetch user data using Corbado's Python SDK.

Example main.py:



from typing import List
from corbado_python_sdk.entities.session_validation_result import (
    SessionValidationResult,
)
from corbado_python_sdk.generated.models.identifier import Identifier

from fastapi import FastAPI, Request, Response
from fastapi.responses import HTMLResponse
from fastapi.templating import Jinja2Templates
from dotenv import load_dotenv
import os
from corbado_python_sdk import (
    Config,
    CorbadoSDK,
    IdentifierInterface,
    SessionInterface,
)

load_dotenv()

app = FastAPI()

templates = Jinja2Templates(directory="templates")

PROJECT_ID: str = os.getenv("PROJECT_ID", "pro-xxx")
API_SECRET: str = os.getenv("API_SECRET", "corbado1_xxx")

# Session config
short_session_cookie_name = "cbo_short_session"

# Config has a default values for 'short_session_cookie_name' and 'BACKEND_API'
config: Config = Config(
    api_secret=API_SECRET,
    project_id=PROJECT_ID,
)

# Initialize SDK
sdk: CorbadoSDK = CorbadoSDK(config=config)
sessions: SessionInterface = sdk.sessions
identifiers: IdentifierInterface = sdk.identifiers


@app.get("/", response_class=HTMLResponse)
async def get_login(request: Request):
    return templates.TemplateResponse(
        "login.html", {"request": request, "PROJECT_ID": PROJECT_ID}
    )


@app.get("/profile", response_class=HTMLResponse)
async def get_profile(request: Request):
    # Acquire cookies with your preferred method
    token: str = request.cookies.get(config.short_session_cookie_name) or ""
    validation_result: SessionValidationResult = (
        sessions.get_and_validate_short_session_value(short_session=token)
    )
    if validation_result.authenticated:

        emailList: List[Identifier] = identifiers.list_all_emails_by_user_id(
            user_id=validation_result.user_id
            or ""  # at this point user_id should be non empty string since user was authenticated
        )

        context = {
            "request": request,
            "PROJECT_ID": PROJECT_ID,
            "USER_ID": validation_result.user_id,
            "USER_NAME": validation_result.full_name,
            "USER_EMAIL": emailList[0].value,
        }
        return templates.TemplateResponse("profile.html", context)

    else:
        return Response(
            content="You are not authenticated or have not yet confirmed your email.",
            status_code=401,
        )


if __name__ == "__main__":
    import uvicorn

    uvicorn.run(app, host="127.0.0.1", port=8000)


Enter fullscreen mode Exit fullscreen mode

Running the FastAPI Application

Finally, to run your FastAPI application, install Uvicorn:



pip install 'uvicorn[standard]'


Enter fullscreen mode Exit fullscreen mode

Then, start your server:



uvicorn main:app --reload


Enter fullscreen mode Exit fullscreen mode

Visit http://localhost:8000 in your browser to see the UI component in action.

Python Passkeys FastAPI Login

Conclusion

This tutorial demonstrated how to integrate passkey authentication into a FastAPI application using Corbado's web-js package. This setup provides a secure and modern authentication method, while also managing user sessions seamlessly. For more details on extending this implementation or integrating with existing apps, refer to the Corbado documentation.

passkeys Article's
30 articles in total
Favicon
Ensuring Successful Passkey Deployment: Testing Strategies for Enterprises
Favicon
How to integrate Passkeys into Enterprise Stacks?
Favicon
Initial Planning & Technical Assessment for Passkeys
Favicon
How to build a Passkey Product, a Strategy and Design for it?
Favicon
How to Engage with Stakeholders in Passkey Projects?
Favicon
Update / Delete Passkeys via WebAuthn Signal API
Favicon
Introduction to Smart Wallets and Passkeys authentication
Favicon
Passkeys, Are passwords obsolete now?!
Favicon
Guide: How to Add Passkeys to Enterprise Systems
Favicon
No Matching Passkeys Available: Troubleshooting Your Login Issue
Favicon
How to Activate Apple Passkeys on MacBooks
Favicon
How to Activate Microsoft Passkeys on Windows
Favicon
How to Use Passkeys with Google Password Manager
Favicon
How to Verify User Accounts in Passkey-Based Systems
Favicon
Passkey One-Tap Login: The Most Efficient Login
Favicon
User Presence & Verification in WebAuthn: Detailed Guide
Favicon
Let's Make the Internet a Safer Place! State of Passkeys is Now Live on Product Hunt 🚀
Favicon
Payment Passkeys @ Mastercard: Revolution for Payment Security
Favicon
How to Integrate Passkeys in Python (FastAPI)
Favicon
Tutorial: Integrate Passkeys into Django (Python)
Favicon
Tutorial: How to Integrate Passkeys into Next.js
Favicon
Passkeys / WebAuthn Library v2.0 is there! 🎉
Favicon
WebAuthn PRF Extension, Related Origins & Passkey Upgrades
Favicon
How To Activate Apple Passkeys on iPhones
Favicon
Passkeys in Australia: Download Free Whitepaper
Favicon
Why B2C Auth is Fundamentally Broken
Favicon
Why B2C Auth is Fundamentally Broken
Favicon
WebAuthn & iframes Integration for Cross-Origin Authentication
Favicon
How Passkeys Protect Against Phishing
Favicon
Are Device-Bound Passkeys AAL2- or AAL3-Compliant?

Featured ones: