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 / 问题 / 79127939
Accepted
Brandon Ros
Brandon Ros
Asked: 2024-10-26 14:28:02 +0800 CST2024-10-26 14:28:02 +0800 CST 2024-10-26 14:28:02 +0800 CST

Windows Edge/Chrome/Firefox 可以进行 Kerberos 协商(不是 NTLM)吗?

  • 772

我能够kinit + klist在 Mac + Windows 上成功验证票据。我甚至将我的 Kerberos 配置转移到“KerberosForWindows”。

似乎 Windows 上的任何浏览器都无法执行 Kerberos 样式协商,而不是 Windows 样式的 NTLM。是这样的吗?

我已尝试了所有本地站点 / 内部网 / 受信任域 / 浏览器配置传播域以进行协商身份验证。我只是想确认 Windows 浏览器是否可以进行 GSSAPI 样式协商。

// Kerberos authentication middleware
async function kerberosAuth(req, res, next) {
  // Check for the Authorization header and extract the token
  const authHeader = req.headers['authorization'];
  if (!authHeader || !authHeader.startsWith('Negotiate ')) {
      res.setHeader('WWW-Authenticate', 'Negotiate');
      return res.status(401).send('Kerberos authentication required');
  }
  const token = authHeader.slice('Negotiate '.length);

  // Base64 decode the token
  const decodedToken = Buffer.from(token, 'base64');

  // Check if it's NTLM
  if (decodedToken.toString('hex').startsWith('4e544c4d')) {
    // always hit on Windows
    return res.status(500).send('NTLM is not supported. Please use Kerberos authentication.');
  } else {
    // only ever hit on Mac/Linux
  }
google-chrome
  • 1 1 个回答
  • 27 Views

1 个回答

  • Voted
  1. Best Answer
    grawity_u1686
    2024-10-27T00:26:17+08:002024-10-27T00:26:17+08:00

    HTTP 上的 KerberosNegotiate也非常具有“Windows 风格”,因为 HTTP 机制是由 Microsoft 发明的,并且定义为使用 MS-SPNEGO(因此称为“Negotiate”)而不是原始 Kerberos 令牌。

    所有流行的浏览器都支持 Windows 上通过 HTTP 实现的 Kerberos Negotiate,但 MIT 的“Kerberos for Windows”包(gssapi.dll)仅受 Mozilla/Firefox/Thunderbird 支持,而 IE 和 Chrome/Edge仅通过 Windows 内置的 SSPI 接口(也是 Firefox 中的默认接口)支持 Kerberos。

    因此,我通常避免在工作站上安装 KfW,而是配置 Windows SSPI,因为这样可以为更长的软件列表提供 GSSAPI 支持。

    1. (如果该领域不是 AD 域)将该领域标记为“MIT 领域”,这样 Windows 就不会期望其 KDC 响应 Netlogon ping。对于 Active Directory 领域,请跳过此步骤。

      sudo ksetup /SetRealmFlags EXAMPLE.COM TcpSupported
      

      实际标志是可选的;领域条目本身的存在表明领域类型为非 AD。

    2. (如果该领域不是 AD 域,并且有人懒得添加 DNS SRV 记录)手动定义 KDC 主机名。

      sudo ksetup /AddKdc EXAMPLE.COM foo.example.com
      

      正确管理的领域(MIT 和 AD)应该具有_kerberos._udp.<domain>指向 KDC 的 SRV 记录,并且不需要此配置。请注意,此条目的存在将其标记为 MIT/非 AD 领域,如上所述。

    3. (如果客户端使用本地帐户进行 Windows 登录)将 Kerberos 凭据存储在 Windows 凭据管理器中:

      cmdkey /add:*.example.com /user:[email protected] /pass
      

      对于非域登录,这是 SSPI 最接近对应的kinit,当程序尝试获取服务票证时,它将使用存储的凭据自动获取 TGT。

      如果 TGT 的凭据不可用,IE 和 Edge/Chrome 将使用 Basic 样式的对话框提示他们,但 Firefox 不支持该功能(因为这通常不是 GSSAPI 应用程序所做的事情)。

    附注:当前版本的 Firefox 存在一个错误,即当浏览器中启用 DoH 时,它们无法通过 SSPI 使用 Kerberos。请确保在 Firefox 设置中将其禁用。


    • 确保为正确的主机名存储了您的凭据。与 MIT Kerberos/KfW 不同,本机 Windows Kerberos 不执行规范化(既不执行 rDNS 也不执行 CNAME),因此如果您访问http://foo,它将变为 SPN HTTP/foo,而不是HTTP/foo.example.com- 因此您需要专门为 存储凭据foo;仅为 存储它们是不够的*.example.com。

    • 出于同样的原因,首先确保该领域具有正确的 SPN。使用 Wireshark 或查看 KDC 的日志以查看请求的内容。

    • 不要使用保存的凭据,而是尝试使用浏览器启动runas它们,使其处于“登录会话”中(就像是域登录一样):

      runas /u:[email protected] /netonly cmd
      

      然后从此 Cmd 窗口启动 Firefox 或任何其他支持 Kerberos 的客户端(但我不认为这适用于 Chrome/Edge)。

    • 运行klist get <spn>并检查该主体的票证是否出现在结果列表中。(确保您运行的是 Windows 内置 klist,而不是 KfW klist 或 Java klist!)

      PS C:\>  where.exe klist
      C:\Program Files\Eclipse Adoptium\jre-21.0.3.9-hotspot\bin\klist.exe
      C:\Windows\System32\klist.exe
      PS C:\>  C:\Windows\System32\klist get HTTP/foo.example.com
      
    • 对于 Firefox,请访问about:logging并启用negotiateauth:5模块的日志记录。

      (或者,将其放入NSPR_LOG_MODULES环境变量中,并进行设置NSPR_LOG_FILE,或者使用 运行 Firefox -console。)

    • 对于 IE 和 Edge/Chrome,手动添加您的主机名或添加*.example.com到“本地 Intranet”区域,并确保选择“仅在 Intranet 区域自动登录”。

    • 直接调用 SSPI,绕过所有特定于浏览器的逻辑:

      1. 从 MS Store 获取 Python 3.12
      2. pip install pyspnego或者pip install sspilib
      3. 打开交互式 Python REPL 并:
        import spnego
        ctx = spnego.client(hostname="foo.example.com",
                            service="HTTP",
                            protocol="negotiate")
        out_token = ctx.step(None)
        
        或者
        import sspilib
        creds = sspilib.UserCredential(usage="initiate",
                                       protocol="Negotiate",
                                       protocol_list=["!ntlm"])
        ctx = sspilib.ClientSecurityContext(credential=creds,
                                            target_name="HTTP/foo.example.com")
        out_token = ctx.step(none)
        
        或者
        import sspi
        ctx = sspi.ClientAuth("Negotiate", targetspn="HTTP/foo.example.com")
        err, bufs = ctx.authorize(None)
        out_token = bufs[0].Buffer
        
        ...并验证它是否返回 Kerberos 令牌out_token。
    • 1

相关问题

  • 在两个 chrome 后台脚本之间发送消息

  • Google Chrome - 开发工具 - 网络选项卡:设计更新

  • Chrome 重复控制台条目

  • 自更新 120 以来的 Chrome 调试器:使用“debugger”关键字时断点不起作用

  • Chrome 扩展:注入页面的 iframe 返回 null

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