在 Delphi XE 中使用 Indy IdHttp Digest 身份验证的 401
Posted
技术标签:
【中文标题】在 Delphi XE 中使用 Indy IdHttp Digest 身份验证的 401【英文标题】:401 With Indy IdHttp Digest Authentication in Delphi XE 【发布时间】:2012-08-21 14:21:09 【问题描述】:尝试使用 Digest 对合作伙伴的使用 Delphi XE 的 Web 服务执行 get()。
我已将IdAuthenticationDigest
包含在uses 子句中,该子句应该会自动根据我所阅读的内容起作用-但我一定遗漏了一些东西,因为我得到了401 Unauthorized。
代码:
begin
// Init request:
IdHttp := TIdHttp.Create(nil);
try
idHttp.Request.ContentType := self.inputType; // 'application/xml'
idHttp.Request.Accept := self.outputType; //'application/json';
// Set request method:
idHttp.Request.Method := Method; // 'Get'
// Set username and password:
idHttp.Request.BasicAuthentication := False;
// IdHttp.Request.Username/Password also fails
IdHttp.Request.Authentication.Username := 'xx';
IdHttp.Request.Authentication.password := 'xx';
IdHttp.Request.ContentLength := Length(Body);
// Send request:
if Method = 'GET' then
Result := idHttp.Get(self.ServiceHost + URI)
else
if Method = 'POST' then
Result := idHttp.Post(self.ServiceHost + URI, SendStream);
finally
idHttp.Free;
end;
end;
【问题讨论】:
【参考方案1】:您需要设置Request.Username
和Request.Password
属性,而不是使用Request.Authentication
属性。此外,根本不要设置Request.Method
或Request.ContentLength
属性。所有这三个属性都由TIdHTTP
内部管理。
// Init request:
IdHttp := TIdHttp.Create(nil);
try
idHttp.Request.ContentType := self.inputType; // 'application/xml'
idHttp.Request.Accept := self.outputType; //'application/json';
// Set username and password:
idHttp.Request.BasicAuthentication := False;
IdHttp.Request.Username := 'xx';
IdHttp.Request.Password := 'xx';
// Send request:
if Method = 'GET' then
Result := IdHttp.Get(self.ServiceHost + URI)
else
if Method = 'POST' then
Result := IdHttp.Post(self.ServiceHost + URI, SendStream);
finally
IdHttp.Free;
end;
【讨论】:
谢谢,试过了(Request.Username 是我开始的方式),仍然失败,出现 401 - 资源需要身份验证。 那么请提供来回的实际 HTTP 流量的跟踪日志。TIdHTTP
正在发送错误的摘要请求,或者根本没有发送摘要请求。使用数据包嗅探器(如 Wireshark)或 Indy 自己的 TIdLog...
组件之一来捕获该流量。
根本不发送摘要请求。
在TIdHTTP.OnSelectAuthorization
事件中,AuthenticationClass
参数是否在输入时设置为TIdDigestAuthentication
?如果不是,那么要么服务器一开始就没有请求摘要(您可以在同一事件中使用 AuthInfo
参数进行验证),或者 TIdDigestAuthentication
实际上并未向 TIdHTTP
注册。
我刚刚确认AuthenticationClass
在OnSelectAuthorization
事件中被设置为TIdDigestAuthentication
。我尝试联系的 API 合作伙伴声称缺少构成摘要请求的两个请求中的第二个。【参考方案2】:
添加类似这样的 OnAuthorization 事件:
procedure TForm1.IdHTTP1Authorization(Sender: TObject;
Authentication: TIdAuthentication; var Handled: Boolean);
begin
Authentication.Username:='user';
Authentication.Password:='passs';
if Authentication is TIdDigestAuthentication then
begin
showmessage('onAuthorization: '+Authentication.Authentication);
TIdDigestAuthentication(IdHTTP1.Request.Authentication).Uri:=IdHTTP1.Request.URL;
TIdDigestAuthentication(Authentication).Method := 'GET';
end;
Handled:=true;
end;
有时 indy 会遗漏一些必要的信息。就我而言,我正在连接到 Tomcat 服务器,但这需要在发送摘要参数身份验证信息时使用 Get 方法。
【讨论】:
这个事件没有被触发。 确保 IdHttp 的 basicAuthentication 设置为 false 并且 IdHttp.Request.Authentication.Username 和 IdHttp.Request.Authentication.password 为空。【参考方案3】:在执行 GET 之前,您还需要设置 hoInProcessAuth 标志。
idHttp.HTTPOptions := idHttp.HTTPOptions + [hoInProcessAuth];
【讨论】:
以上是关于在 Delphi XE 中使用 Indy IdHttp Digest 身份验证的 401的主要内容,如果未能解决你的问题,请参考以下文章
Delphi XE 的 SOAP 服务器和客户端应用程序 VCL+indy 演示?
delphi2009(10,xe)下indy10发送utf8字符串
Delphi XE10 IdTCPClient和IdTCPServer 通讯编码规则写法(Indy 10)(编码乱码)
Delphi XE10 IdTCPClient和IdTCPServer 通讯编码规则写法(Indy 10)(编码乱码)