AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • 主页
  • 系统&网络
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • 主页
  • 系统&网络
    • 最新
    • 热门
    • 标签
  • Ubuntu
    • 最新
    • 热门
    • 标签
  • Unix
    • 最新
    • 标签
  • DBA
    • 最新
    • 标签
  • Computer
    • 最新
    • 标签
  • Coding
    • 最新
    • 标签
主页 / coding / 问题 / 79200791
Accepted
mohamed amine salah
mohamed amine salah
Asked: 2024-11-19 01:06:11 +0800 CST2024-11-19 01:06:11 +0800 CST 2024-11-19 01:06:11 +0800 CST

响应中的 id_token 缺少 nonce

  • 772

我使用 Keycloak 21 进行身份验证,但遇到了一个问题,即调用 /token 后返回的 id_token 中不包含 nonce 值。我在 /auth 请求中传递了 nonce,

keycloak
  • 1 1 个回答
  • 30 Views

1 个回答

  • Voted
  1. Best Answer
    Bench Vue
    2024-11-19T11:06:50+08:002024-11-19T11:06:50+08:00

    你可以通过这个方法

    URL

    GET ${KEYCLOAK_URL}/realms/${REALM}/protocol/openid-connect/auth?
    

    query parameters

            client_id: CLIENT_ID,
            redirect_uri: REDIRECT_URI,
            response_type: 'code',
            scope: 'openid',
            nonce: nonce,
    

    然后在回调URL获取ID token

    步骤

    步骤 1:通过 docker compose 启动 Keycloak v21

    version: '3.9'
    
    services:
      keycloak:
        image: quay.io/keycloak/keycloak:21.1.2
        container_name: keycloak
        command:
          - start-dev
        environment:
          KEYCLOAK_ADMIN: admin
          KEYCLOAK_ADMIN_PASSWORD: admin
          KC_DB: postgres
          KC_DB_URL_HOST: keycloak-db
          KC_DB_URL_DATABASE: keycloak
          KC_DB_USERNAME: keycloak
          KC_DB_PASSWORD: keycloakpassword
          KC_HOSTNAME: localhost
        ports:
          - "8080:8080"
        depends_on:
          - keycloak-db
        volumes:
          - keycloak-data:/opt/keycloak/data
    
      keycloak-db:
        image: postgres:15
        container_name: keycloak-db
        environment:
          POSTGRES_USER: keycloak
          POSTGRES_PASSWORD: keycloakpassword
          POSTGRES_DB: keycloak
        ports:
          - "5432:5432"
        volumes:
          - postgres-data:/var/lib/postgresql/data
    
    volumes:
      keycloak-data:
      postgres-data:
    

    更多详细信息请参阅此处

    步骤 2. 创建 my-realm、my-cleint 和 user1

    在此处输入图片描述

    回调网址

    http://localhost:3000/callback
    

    在此处输入图片描述

    从 UI 复制客户端机密

    在此处输入图片描述

    username是 User1,password是 1234

    在此处输入图片描述

    在此处输入图片描述

    步骤 3:运行 Express 服务器

    install dependencies

    npm install express axios querystring [email protected] crypto
    

    demo.js

    const express = require('express');
    const axios = require('axios');
    const crypto = require('crypto');
    const querystring = require('querystring');
    const jwtDecode = require('jwt-decode');
    
    const app = express();
    const PORT = 3000; //replace your Redirect PORT
    
    // Keycloak Configuration
    const KEYCLOAK_URL = 'http://localhost:8080'; // Replace with your Keycloak server URL
    const REALM = 'my-realm'; // Replace with your realm name
    const CLIENT_ID = 'my-client'; // Replace with your client ID
    const CLIENT_SECRET = '[your my-client password]'; // Replace if your client is confidential
    const REDIRECT_URI = 'http://localhost:3000/callback'; // Replace with your app's redirect URI
    
    // Generate a random nonce
    const generateNonce = () => crypto.randomBytes(16).toString('base64');
    
    // Store the nonce temporarily (use a proper session store in production)
    let storedNonce;
    
    // Routes
    
    // 1. Start Authorization
    app.get('/login', (req, res) => {
        const nonce = generateNonce();
        storedNonce = nonce; // Save the nonce for validation later
    
        const authUrl = `${KEYCLOAK_URL}/realms/${REALM}/protocol/openid-connect/auth?` + querystring.stringify({
            client_id: CLIENT_ID,
            redirect_uri: REDIRECT_URI,
            response_type: 'code',
            scope: 'openid',
            nonce: nonce,
        });
    
        res.redirect(authUrl); // Redirect user to Keycloak login
    });
    
    // Step 2. Handle Callback for get tokens
    app.get('/callback', async (req, res) => {
        const { code } = req.query;
    
        if (!code) {
            return res.status(400).send('Authorization code not found!');
        }
    
        try {
            const tokenResponse = await axios.post(
                `${KEYCLOAK_URL}/realms/${REALM}/protocol/openid-connect/token`,
                querystring.stringify({
                    grant_type: 'authorization_code',
                    code: code,
                    redirect_uri: REDIRECT_URI,
                    client_id: CLIENT_ID,
                    client_secret: CLIENT_SECRET,
                }),
                { headers: { 'Content-Type': 'application/x-www-form-urlencoded' } }
            );
    
            const { id_token } = tokenResponse.data;
    
            // Decode and validate the ID token
            const decodedToken = jwtDecode(id_token);
    
            if (decodedToken.nonce !== storedNonce) {
                return res.status(401).send('Invalid nonce!');
            }
    
            res.send({
                message: 'Login successful!',
                id_token: decodedToken,
            });
        } catch (error) {
            console.error('Error exchanging code for tokens:', error);
            res.status(500).send('Error exchanging code for tokens.');
        }
    });
    
    // Start the Express server
    app.listen(PORT, () => {
        console.log(`Server running at http://localhost:${PORT}`);
    });
    

    run it

    node demo.js
    

    在此处输入图片描述

    步骤4 通过浏览器获取token

    http://localhost:3000/login
    

    结果

    在此处输入图片描述

    • 1

相关问题

  • 客户端凭据是 unsupported_grant_type

  • 使用 Keycloak 作为登录提供商的 StackOverflow 返回空的电子邮件地址

  • Keycloak 令牌交换错误 - 客户端不在令牌受众范围内

  • 如何在keycloak中列出服务帐户用户

  • 在没有 admin-cli 的情况下通过 KeyCloak 中的 API 创建用户?

Sidebar

Stats

  • 问题 205573
  • 回答 270741
  • 最佳答案 135370
  • 用户 68524
  • 热门
  • 回答
  • Marko Smith

    Vue 3:创建时出错“预期标识符但发现‘导入’”[重复]

    • 1 个回答
  • Marko Smith

    为什么这个简单而小的 Java 代码在所有 Graal JVM 上的运行速度都快 30 倍,但在任何 Oracle JVM 上却不行?

    • 1 个回答
  • Marko Smith

    具有指定基础类型但没有枚举器的“枚举类”的用途是什么?

    • 1 个回答
  • Marko Smith

    如何修复未手动导入的模块的 MODULE_NOT_FOUND 错误?

    • 6 个回答
  • Marko Smith

    `(表达式,左值) = 右值` 在 C 或 C++ 中是有效的赋值吗?为什么有些编译器会接受/拒绝它?

    • 3 个回答
  • Marko Smith

    何时应使用 std::inplace_vector 而不是 std::vector?

    • 3 个回答
  • Marko Smith

    在 C++ 中,一个不执行任何操作的空程序需要 204KB 的堆,但在 C 中则不需要

    • 1 个回答
  • Marko Smith

    PowerBI 目前与 BigQuery 不兼容:Simba 驱动程序与 Windows 更新有关

    • 2 个回答
  • Marko Smith

    AdMob:MobileAds.initialize() - 对于某些设备,“java.lang.Integer 无法转换为 java.lang.String”

    • 1 个回答
  • Marko Smith

    我正在尝试仅使用海龟随机和数学模块来制作吃豆人游戏

    • 1 个回答
  • Martin Hope
    Aleksandr Dubinsky 为什么 InetAddress 上的 switch 模式匹配会失败,并出现“未涵盖所有可能的输入值”? 2024-12-23 06:56:21 +0800 CST
  • Martin Hope
    Phillip Borge 为什么这个简单而小的 Java 代码在所有 Graal JVM 上的运行速度都快 30 倍,但在任何 Oracle JVM 上却不行? 2024-12-12 20:46:46 +0800 CST
  • Martin Hope
    Oodini 具有指定基础类型但没有枚举器的“枚举类”的用途是什么? 2024-12-12 06:27:11 +0800 CST
  • Martin Hope
    sleeptightAnsiC `(表达式,左值) = 右值` 在 C 或 C++ 中是有效的赋值吗?为什么有些编译器会接受/拒绝它? 2024-11-09 07:18:53 +0800 CST
  • Martin Hope
    The Mad Gamer 何时应使用 std::inplace_vector 而不是 std::vector? 2024-10-29 23:01:00 +0800 CST
  • Martin Hope
    Chad Feller 在 5.2 版中,bash 条件语句中的 [[ .. ]] 中的分号现在是可选的吗? 2024-10-21 05:50:33 +0800 CST
  • Martin Hope
    Wrench 为什么双破折号 (--) 会导致此 MariaDB 子句评估为 true? 2024-05-05 13:37:20 +0800 CST
  • Martin Hope
    Waket Zheng 为什么 `dict(id=1, **{'id': 2})` 有时会引发 `KeyError: 'id'` 而不是 TypeError? 2024-05-04 14:19:19 +0800 CST
  • Martin Hope
    user924 AdMob:MobileAds.initialize() - 对于某些设备,“java.lang.Integer 无法转换为 java.lang.String” 2024-03-20 03:12:31 +0800 CST
  • Martin Hope
    MarkB 为什么 GCC 生成有条件执行 SIMD 实现的代码? 2024-02-17 06:17:14 +0800 CST

热门标签

python javascript c++ c# java typescript sql reactjs html

Explore

  • 主页
  • 问题
    • 最新
    • 热门
  • 标签
  • 帮助

Footer

AskOverflow.Dev

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve