- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
背景:我正在摆弄一个无法工作的 XMPP 服务器。
因此,有关此特定响应 token 的文档称为 rspauth
实际上在任何地方都没有很好的记录。有些人似乎只是跳过它并使用如下所示的静态字符串:
cnNwYXV0aD1lYTQwZjYwMzM1YzQyN2I1NTI3Yjg0ZGJhYmNkZmZmZA==
b64 解码为:
rspauth=ea40f60335c427b5527b84dbabcdfffd
但是在 MD5-DIGEST
的最后阶段身份验证,据说我应该发送以下内容:
<challenge xmlns='urn:ietf:params:xml:ns:xmpp-sasl'>
cnNwYXV0aD1lYTQwZjYwMzM1YzQyN2I1NTI3Yjg0ZGJhYmNkZmZmZA==
</challenge>
同样,我不确定为什么每个错误报告中的每个人都使用这个静态字符串。但任何 XMPP 客户端都会响应:
jabber: Error is -10 : SASL(-10): server failed mutual authentication step: DIGEST-MD5: This server wants us to believe that he knows shared secret
我尝试尽可能遵循 RFC,并且 this is what the RFC says关于rspauth=
:
The server receives and validates the "digest-response". The server
checks that the nonce-count is "00000001". If it supports subsequent
authentication (see section 2.2), it saves the value of the nonce and the nonce-count. It sends a message formatted as follows:response-auth = "rspauth" "=" response-value
where response-value is calculated as above, using the values sent in step two, except that if qop is "auth", then A2 is
A2 = { ":", digest-uri-value }
基于此,这就是我构建我的 rspauth
的方式:
rspauth=b64enc(ByteConv('rspauth=:'+md5_this("xmpp/example.com")))
归结为:
# md5_this == 6dae15e9021a0103e8e09ce86956a659 (obv not with example.com)
respauth = 'cnNwYXV0aD02ZGFlMTVlOTAyMWEwMTAzZThlMDljZTg2OTU2YTY1OQ=='
cli.respond('<challenge xmlns='urn:ietf:params:xml:ns:xmpp-sasl'>cnNwYXV0aD02ZGFlMTVlOTAyMWEwMTAzZThlMDljZTg2OTU2YTY1OQ==</challenge>')
根据this thread关于此事,我发送的最后一条内容是错误的,我应该发送:
<success xmlns='urn:ietf:params:xml:ns:xmpp-sasl'>cnNwYXV0aD02ZGFlMTVlOTAyMWEwMTAzZThlMDljZTg2OTU2YTY1OQ==</success>
这也失败了,然后客户端发送 </stream:stream>
连接中断。
随着<challenge>...
现在这就是我迷失的地方,我假设我正在构建 rspauth=...
token 错误,但我不知道它应该是什么。
这是 Pidgin
之间的完整通信跟踪和 XMPP 服务器:
client connected
<< <?xml version='1.0' ?>
<< <stream:stream to='example.com' xmlns='jabber:client' xmlns:stream='http://etherx.jabber.org/streams' version='1.0'>
>> Sending: <?xml version='1.0'?>
<stream:stream xmlns:stream='http://etherx.jabber.org/streams' version='1.0' from='example.com' id='d86961dc-bfb5-4578-aa45-116d5f14ef54' xml:lang='en' xmlns='jabber:client'>
<stream:features>
<starttls xmlns='urn:ietf:params:xml:ns:xmpp-tls'><required/></starttls>
<register xmlns='http://jabber.org/features/iq-register'/></stream:features>
<< <starttls xmlns='urn:ietf:params:xml:ns:xmpp-tls'/>
>> Sending: <proceed xmlns='urn:ietf:params:xml:ns:xmpp-tls'/>
- Secure connection established [TLS]
<< <stream:stream to='example.com' xmlns='jabber:client' xmlns:stream='http://etherx.jabber.org/streams' version='1.0'>
>> sending: <?xml version='1.0'?>
<stream:stream xmlns:stream='http://etherx.jabber.org/streams' version='1.0' from='example.com' id='516f7395-4112-4892-87f1-2e9f7f3a96e1' xml:lang='en' xmlns='jabber:client'>
<stream:features>
<mechanisms xmlns='urn:ietf:params:xml:ns:xmpp-sasl'>
<mechanism>DIGEST-MD5</mechanism>
</mechanisms>
<auth xmlns='http://jabber.org/features/iq-auth'/>
</stream:features>
<< <auth xmlns='urn:ietf:params:xml:ns:xmpp-sasl' mechanism='DIGEST-MD5' xmlns:ga='http://www.google.com/talk/protocol/auth' ga:client-uses-full-bind-result='true'/>
>> Sending: <challenge xmlns='urn:ietf:params:xml:ns:xmpp-sasl'>cmVhbG09ImV4YW1wbGUuY29tIixub25jZT0iMTE2Iixxb3A9ImF1dGgiLGNoYXJzZXQ9dXRmLTgsYWxnb3JpdGhtPW1kNS1zZXNz</challenge>
<< <response xmlns='urn:ietf:params:xml:ns:xmpp-sasl'>dXNlcm5hbWU9InRvcnhlZCIscmVhbG09ImV4YW1wbGUuY29tIixub25jZT0iMTE2Iixjbm9uY2U9IjZGUDF5RUtBRk1TN2lHSnRBNlNiME5oQ1JBcmhGU0t3OHRMa2xJVEJPZGs9IixuYz0wMDAwMDAwMSxxb3A9YXV0aCxkaWdlc3QtdXJpPSJ4bXBwL2V4YW1wbGUuY29tIixyZXNwb25zZT1jODhmNTRiMjJlMmFiZGI4ZThlMTljOWVjZDliYjAxOCxjaGFyc2V0PXV0Zi04</response>
>> Sending: <challenge xmlns='urn:ietf:params:xml:ns:xmpp-sasl'>cnNwYXV0aD1lYTQwZjYwMzM1YzQyN2I1NTI3Yjg0ZGJhYmNkZmZmZA==</challenge>
DEBUG: rspauth=ea40f60335c427b5527b84dbabcdfffd
<< </stream:stream>
我遵循了这些 RFC 指南:
并查看了这些源代码:
基本上剩下的就是对成功的连接进行逆向工程,并进一步分解响应,但 md5 哈希值很难及时逆向,所以我这次寻求帮助。
检查了一些旧的源代码并发现了以下内容:
respauth = step_4 + ':' + nonce + ':' + cresp['nc'] + ':' + cnonce + ':' + cresp['qop'] + ':' + step_5
rspauth = 'rspauth=' + md5_this(rspauth)
现在它仍然会生成相同的错误,但它不是静态字符串,所以我现在可以使用它。
最佳答案
我知道这已经是 4 个月前的事了,但我也遇到了完全相同的问题,而且我似乎已经解决了。
sasl 摘要的 rfc ( https://www.rfc-editor.org/rfc/rfc2831#section-2.1.3 )
我用 C# 编写了代码,但它应该很容易熟练,我使用了 RFC 中使用的函数名称。
我知道代码功能不完整或没有错误,但这对我来说适用于 pidgin、gajim 和 psi(psi 不会检查从服务器返回的哈希值,我确信这与此有关)。我希望这段代码对您或其他需要查找此内容的人有所帮助。
!注意! MD5-Digest 被视为不安全,新标准是 SCRAM SASL ( https://www.rfc-editor.org/rfc/rfc5802 )
检查客户端Hash的方法
public string GenerateClientHash(string user, string realm, string password, string nonce, string cnonce, string authzid, string digesturi, string qop, string nc)
{
byte[] A2 = (qop.Equals("auth") ? Encoder.GetBytes($"AUTHENTICATE:{digesturi}"): Encoder.GetBytes($"AUTHENTICATE:{digesturi}:00000000000000000000000000000000"));
byte[] h1 = H(Encoder.GetBytes($"{user}:{realm}:{password}"));
byte[] h2 = Encoder.GetBytes($":{nonce}:{cnonce}");
byte[] A1 = Merge(h1,h2);
byte[] response_value = HEX(KD(HEX(H(A1)),
Merge(Encoder.GetBytes($"{nonce}:{nc}:{cnonce}:{qop}:"), HEX(H(A2)))
));
byte[] h3 = Merge(h1, h2);
return Encoder.GetString(response_value);
}
生成服务器哈希(您遇到问题的那个)
public string GenerateServerHash(string user, string realm, string password, string nonce, string cnonce, string authzid, string digesturi, string qop, string nc)
{
byte[] A2 = (qop.Equals("auth") ? Encoder.GetBytes($":{digesturi}") : Encoder.GetBytes($":{digesturi}:00000000000000000000000000000000"));
byte[] h1 = H(Encoder.GetBytes($"{user}:{realm}:{password}"));
byte[] h2 = Encoder.GetBytes($":{nonce}:{cnonce}");
byte[] A1 = Merge(h1, h2);
byte[] response_value = HEX(KD(HEX(H(A1)),
Merge(Encoder.GetBytes($"{nonce}:{nc}:{cnonce}:{qop}:"), HEX(H(A2)))
));
byte[] h3 = Merge(h1, h2);
return Encoder.GetString(response_value);
}
RFC 提到的功能。
/// <summary>
/// Let H(s) be the 16 octet MD5 hash [RFC 1321] of the octet string s.
/// </summary>
/// <param name="s">Byte Array Af streng for at sikre encodning</param>
/// <returns>s -> Base16(s) = s16 -> MD5Hash(s16)</returns>
public byte[] H(byte[] s)
{
MD5CryptoServiceProvider md5 = new MD5CryptoServiceProvider();
return md5.ComputeHash(s);
}
/// <summary>
/// Let KD(k, s) be H({k, ":", s}), i.e., the 16 octet hash of the string k, a colon and the string s.
/// </summary>
/// <param name="k">Byte Array Af streng for at sikre encodning</param>
/// <param name="s">Byte Array Af streng for at sikre encodning</param>
/// <returns>k + ":" + s = ks -> H(ks)</returns>
public byte[] KD(byte[] k, byte[] s)
{
byte[] colon = Encoder.GetBytes(":");
byte[] ks = Merge(k, colon, s);
return H(ks);
}
/// <summary>
/// Let HEX(n) be the representation of the 16 octet MD5 hash n as a string of 32 hex digits(with alphabetic characters always in lower case, since MD5 is case sensitive).
/// </summary>
/// <param name="n">Byte Array Af streng for at sikre encodning</param>
/// <returns>lowercase of n</returns>
public byte[] HEX(byte[] n)
{
byte[] b = Encoder.GetBytes(toBase16(n).ToLower());
return b;
}
关于python - XMPP rspauth token 应包含 DIGEST-MD5 身份验证中的内容?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34826516/
在 Smack API 中,有一个用于连接的配置类,在此页面中描述 ConnectionConfiguration 我对服务名称和服务器名称之间的区别感到困惑。 假设我有一台名为“mybox.mydo
我想阻止常见的 xmpp 客户端(例如 Pidgin、PSI+...)登录我们的 xmpp 服务器(ejabberd)。我怎样才能做到这一点? 提前致谢。 最佳答案 我认为这不可能真正做到。当然,您可
我想阻止常见的 xmpp 客户端(例如 Pidgin、PSI+...)登录我们的 xmpp 服务器(ejabberd)。我怎样才能做到这一点? 提前致谢。 最佳答案 我不认为这真的可以做到。当然,您可
我正在尝试使用 XEP-0048 - 书签 (http://xmpp.org/extensions/xep-0048.html) 自动加入房间。 我正在使用 RobbieHanson XMPPFram
我正在使用 XMPP(通过 ejabberd)。据此XEP standard ,我发现XMPP服务器可以存储离线消息,并且可以在离线用户上线时传递它们(分享他的存在)。 我的问题是: 1> 这些离线消
我试图通过用低级 Python 编码来学习 XMPP 规范( RFC 3920 )。但是我在 6.5 的第 4 步被挂断了一个多小时。 ,选择身份验证机制。我发送: ,并得到:而不是 base64 编
我搜索过但没有找到 XMPP 使用哪些端口。我需要实现XMPP服务器和客户端并使用XML传输、文件传输和流媒体。他们使用不同的端口吗?有没有办法让它们都一样使用,这样我就不需要打扰网络管理员?谢谢 最
我正在编写一个通过 XMPP 与客户端通信的应用程序。我希望能够使用用户现有的 xmpp 帐户(他们都有 google ID),但我不希望我的消息出现在他们的常规 IM 流中。 我在想,当我的客户端与
如果他们有 jabber 帐户,我想高效地检查所有本地地址簿联系人。用户使用手机号在XMPP服务器上注册。我目前将以下 XEP-0055 节发送到 ejabberd 服务器并评估结果。
是否可以将消息传递到服务器时包含到 XMPP 消息时间? 现在我从 OpenFire 服务器收到如下消息: test 但我需要知道消息何时发送(传递到服务器),例如: test2012-10-12 1
我想知道 xmpp 发布-订阅功能是同时向所有订阅者发送通知(广播)还是在队列中发送通知,即一个一个地发送通知。 我想知道我正在使用的 pub-sub 和 jaxl 类之间的区别,因为 jaxl 在队
关闭。这个问题是off-topic .它目前不接受答案。 想改善这个问题吗? Update the question所以它是 on-topic对于堆栈溢出。 8年前关闭。 Improve this q
XMPP 允许用户使用同一个帐户同时从多个客户端连接到服务器。我构建了一个执行此操作的应用程序,但如果启用了桌面客户端,我不希望用户能够使用移动客户端进行连接。这是一个游戏,连接到这两个会导致问题。
我正在为我的 XMPP 连接使用 libjingle。我可以在没有服务器的情况下连接两个 XMPP 客户端吗?如果是,我该怎么做,如果不是,那为什么不可能呢? XMPP client1 XMPP c
我目前正在研究 XMPP,我想知道是否有办法创建动态 XMPP 花名册。我希望服务器/组件/插件自动生成任何用户的联系人列表。 组件可以访问和修改名册吗? 我知道有些服务器(如 OpenFire)使用
我的网络上有两台机器。我在其中一台机器(机器 A)上安装了 ejabberd,在那里注册了两个用户。我在两台机器上都运行了 Pidgin。我在机器 A 上登录到 Pidgin,并且能够登录。当我尝试在
已结束。此问题正在寻求书籍、工具、软件库等的推荐。它不满足Stack Overflow guidelines 。目前不接受答案。 我们不允许提出寻求书籍、工具、软件库等推荐的问题。您可以编辑问题,以便
默认情况下,XMPP 状态会发布给所有订阅该人的人。是否可以发送诸如 iq 调用之类的内容来获取不在我的名册中的 ID 的存在标签? 最佳答案 如果您想知道 XMPP 实体是否已连接,可以使用 XMP
我希望使用 xmpp 在我的应用程序中包含两个功能。第一个是几乎完成的一对一聊天(使用 strophe),第二个是实时通知,就像它在 facebook 中的工作方式一样。我试图了解 xmpp 中的 p
引用this question ,XMPP 被认为是 IM 互操作性的开放标准。 对于我的应用程序来说,如果我使用 XMPP 进行内部客户端-服务器通信,或者开发自己的内部协议(protocol)但在
我是一名优秀的程序员,十分优秀!