函数执行期间 SOAP 客户端超时
SOAP client timeouts during function execution
我需要使用 SOAP 从数据库中检索一些数据。我不是经验丰富的 PHP 程序员,这就是我需要一些帮助的原因。提供网络服务 (WSDL) 的公司给了我登录信息和 svc 和 wsdl 文件的链接。他们还在 C# 中给了我一个如何连接的例子:
var proxy = new ChannelFactory<ServiceReferenceWCF.IWebService2>("custom");
proxy.Credentials.UserName.UserName = login;
proxy.Credentials.UserName.Password = pass;
var result = proxy.CreateChannel();
var logged_in = result.loggedIn();
这是我的 PHP 代码:
$wsdl_proto = 'https';
$wsdl_host = 'their_wsdl_host';
$wsdl_host_path = 'their_wsdl_path';
$namespace_proto = 'https';
$namespace_host = 'their_namespace_host';
$namespace_path = 'their_namespace_path';
$location = $namespace_proto.'://'.$namespace_host.$namespace_path;
$wsdl_url = $wsdl_proto.'://'.$wsdl_host.$wsdl_host_path;
$connection = new SoapClient($wsdl_url, array('location' => $location, 'soap_version' => SOAP_1_1, 'connection_timeout'=> 600,
'proxy_login' => "my_login", 'proxy_password' => "my_password"));
$functions = $connection->__getFunctions();
var_dump($functions);
$logged_in = $connection->loggedIn();
它在 loggedIn()
函数调用期间挂起。此函数列在 $functions 变量中,因此它是有效的。我尝试了该服务提供的其他一些功能 - 结果始终相同:脚本只是冻结。我的意思是服务没有响应,PHP 等待 loggedIn()
函数完成。超过超时后,我得到一个错误:Error Fetching http headers in...
我究竟做错了什么?我该如何调试它?
更新:
我尝试了你们建议的每一件事。但我仍然没有设法解决问题。我不使用代理。您可以在下面找到结果:
1. 我安装了SoapUI。配置 some_method
函数的请求(创建带有凭据的基本 Auth)后,我收到了响应:An error occurred when verifying security for the message
。
勾选 Authenticate pre-emptively
选项没有帮助。我搜索了这个错误的解决方案,但我没有找到任何东西。
2. 我几乎尝试了所有可以想到的 SoapClient
class 选项组合。以下是其中一些:
$connection = new SoapClient($wsdl_url, array(
'login' => "login",
'password' => "pass",
'trace' => 1,
));
响应 headers 为空。请求 headers:
REQUEST HEADERS:
POST /file.svc HTTP/1.1
Host: host
Connection: Keep-Alive
User-Agent: PHP-SOAP/5.6.16
Content-Type: text/xml; charset=utf-8
SOAPAction: "http://tempuri.org/file/some_method"
Content-Length: 221
Authorization: Basic HASH
REQUEST:
<?xml version="1.0" encoding="UTF-8"?>
<SOAP-ENV:Envelope xmlns:SOAP-ENV="http://schemas.xmlsoap.org/soap/envelope/" xmlns:ns1="http://tempuri.org/"><SOAP-ENV:Body><ns1:some_method/></SOAP-ENV:Body></SOAP-ENV:Envelope>
下一个组合:
$connection = new SoapClient($wsdl,array(
'login' => "login",
'password' => "pass",
'trace' => 1,
'connection_timeout' => 500000,
'cache_wsdl' => WSDL_CACHE_BOTH,
'keep_alive' => false,
));
响应 headers 为空。请求headers和以前一样
3. 使用这个:
$connection->__setLocation('https://host.org/file.svc');
没有帮助。但是,当我将位置设置为 WSDL 文件而不是 SVC 文件时,我得到以下响应:
HTTP/1.1 405 Method Not Allowed
Connection: Keep-Alive
Content-Length: 1293
Date: Wed, 23 Dec 2015 14:28:53 GMT
Content-Type: text/html
Server: Microsoft-IIS/7.5
Allow: GET, HEAD, OPTIONS, TRACE
X-Powered-By: ASP.NET
我确定 WSDL 服务不够慢,无法超过超时(Ricardo Velhote 建议的)。
4. 我有一个 XML 配置文件随我之前提到的 C# 示例一起提供:
<client>
<endpoint address="https://host/file.svc" binding="customBinding" bindingConfiguration="custom" contract="ServiceReference1.file" name="custom" />
</client>
<bindings>
<customBinding>
<binding name="custom">
<security defaultAlgorithmSuite="Default" authenticationMode="UserNameOverTransport" requireDerivedKeys="true" includeTimestamp="true" messageSecurityVersion="WSSecurity11WSTrustFebruary2005WSSecureConversationFebruary2005WSSecurityPolicy11BasicSecurityProfile10">
<localClientSettings detectReplays="false" />
<localServiceSettings detectReplays="false" />
</security>
<textMessageEncoding messageVersion="Soap11WSAddressing10" />
<httpsTransport maxReceivedMessageSize="2147483647" maxBufferPoolSize="2147483647" />
</binding>
</customBinding>
</bindings>
我尝试按照建议 here 扩展 SoapClient class,但是您可以猜到它没有用 - 脚本的行为仍然相同。
您在选项中遗漏了 proxy_host
和 proxy_port
...如果您需要代理才能工作,请提供这些参数:
....
$connection = new SoapClient($wsdl_url, array(
'location' => $location,
'soap_version' => SOAP_1_1,
'connection_timeout'=> 600,
'proxy_host' => '....', // Your proxy host
'proxy_port' => 8080, // Your proxy port
'proxy_login' => "my_login",
'proxy_password' => "my_password"
));
根据您的更新和随示例提供的 XML 配置文件,此 Web 服务似乎正在使用 WS-Addressing
,因此常规 SOAPClient 将无法工作,需要对其进行扩展以支持WS-Addressing
.
这是赠送它的线
<textMessageEncoding messageVersion="Soap11WSAddressing10" />
我还没有使用 WS-Addressing
,但我最近正在分析一个 API 项目,我们将在未来开展需要它的项目。
请注意 this project or this other project,这可能会有用(搜索更多 PHP WS-Addressing
很容易)。
同样,我只做过研究,没有任何实践经验可以帮助您编写实际代码:)
[编辑:更新后的过时答案]
首先,您可能被示例代码中变量 proxy
的使用误导了。他们可能指的是 HTTP 基本身份验证而不是代理。
尝试用 login
和 password
替换 proxy_login
和 proxy_password
。
然而,话虽如此,如果您正在获取 WSDL,则意味着至少它正在连接并获取有关服务的信息(这很好)。
在正常情况下,您不需要在 SoapClient 中指定 location
,因为它应该由 WSDL 文件定义。通过设置 location
参数,您将覆盖 WSDL 文件中的设置,并且您可能会将其指向不存在的位置。
尝试从 SoapClient 构造函数中省略 location
和 soap_version
,让库自动处理这些参数:
$connection = new SoapClient($wsdl_url, array('connection_timeout'=> 600,
'proxy_login' => "my_login", 'proxy_password' => "my_password"));
另一方面,也许您正在处理极其缓慢的 Web 服务。 PHP 中有许多参数可能会影响超时所需的时间,而且它们很可能远低于您的 connection_timeout
参数:
使用您并不真正了解的语言进行编码时,您的第一个动作应该是使用手册和示例。
这是 SoapClient 的 link 到 PHP 手册 - http://php.net/manual/en/soapclient.soapclient.php
为了调试 SoapClient,您可以传递 "trace" 参数。引用手册:设置布尔跟踪选项可以使用方法 SoapClient->__getLastRequest、SoapClient->__getLastRequestHeaders、SoapClient->__getLastResponse 和 SoapClient-> __getLastResponseHeaders.
$client = new SoapClient("some.wsdl", array('trace' => true));
因此,对于您的情况,如果您想查看问题所在,请执行以下操作:
$client = SoapClient($wsdl_url, array('trace' => 1));
$result = $client->SomeFunction();
echo "REQUEST HEADERS:\n" . $client->__getLastRequestHeaders() . "\n";
echo "REQUEST:\n" . $client->__getLastRequest() . "\n";
echo "Response headers:\n" . $client->__getLastResponseHeaders() . "\n";
echo "Response:\n" . $client->__getLastResponse() . "\n";
此外,正如在另一个答案中指出的那样 - 您正在设置 proxy_login 和 proxy_password 参数请求,仅当您使用代理服务器连接到该 WSDL 服务时才应使用该请求。
我需要使用 SOAP 从数据库中检索一些数据。我不是经验丰富的 PHP 程序员,这就是我需要一些帮助的原因。提供网络服务 (WSDL) 的公司给了我登录信息和 svc 和 wsdl 文件的链接。他们还在 C# 中给了我一个如何连接的例子:
var proxy = new ChannelFactory<ServiceReferenceWCF.IWebService2>("custom");
proxy.Credentials.UserName.UserName = login;
proxy.Credentials.UserName.Password = pass;
var result = proxy.CreateChannel();
var logged_in = result.loggedIn();
这是我的 PHP 代码:
$wsdl_proto = 'https';
$wsdl_host = 'their_wsdl_host';
$wsdl_host_path = 'their_wsdl_path';
$namespace_proto = 'https';
$namespace_host = 'their_namespace_host';
$namespace_path = 'their_namespace_path';
$location = $namespace_proto.'://'.$namespace_host.$namespace_path;
$wsdl_url = $wsdl_proto.'://'.$wsdl_host.$wsdl_host_path;
$connection = new SoapClient($wsdl_url, array('location' => $location, 'soap_version' => SOAP_1_1, 'connection_timeout'=> 600,
'proxy_login' => "my_login", 'proxy_password' => "my_password"));
$functions = $connection->__getFunctions();
var_dump($functions);
$logged_in = $connection->loggedIn();
它在 loggedIn()
函数调用期间挂起。此函数列在 $functions 变量中,因此它是有效的。我尝试了该服务提供的其他一些功能 - 结果始终相同:脚本只是冻结。我的意思是服务没有响应,PHP 等待 loggedIn()
函数完成。超过超时后,我得到一个错误:Error Fetching http headers in...
我究竟做错了什么?我该如何调试它?
更新:
我尝试了你们建议的每一件事。但我仍然没有设法解决问题。我不使用代理。您可以在下面找到结果:
1. 我安装了SoapUI。配置 some_method
函数的请求(创建带有凭据的基本 Auth)后,我收到了响应:An error occurred when verifying security for the message
。
勾选 Authenticate pre-emptively
选项没有帮助。我搜索了这个错误的解决方案,但我没有找到任何东西。
2. 我几乎尝试了所有可以想到的 SoapClient
class 选项组合。以下是其中一些:
$connection = new SoapClient($wsdl_url, array(
'login' => "login",
'password' => "pass",
'trace' => 1,
));
响应 headers 为空。请求 headers:
REQUEST HEADERS:
POST /file.svc HTTP/1.1
Host: host
Connection: Keep-Alive
User-Agent: PHP-SOAP/5.6.16
Content-Type: text/xml; charset=utf-8
SOAPAction: "http://tempuri.org/file/some_method"
Content-Length: 221
Authorization: Basic HASH
REQUEST:
<?xml version="1.0" encoding="UTF-8"?>
<SOAP-ENV:Envelope xmlns:SOAP-ENV="http://schemas.xmlsoap.org/soap/envelope/" xmlns:ns1="http://tempuri.org/"><SOAP-ENV:Body><ns1:some_method/></SOAP-ENV:Body></SOAP-ENV:Envelope>
下一个组合:
$connection = new SoapClient($wsdl,array(
'login' => "login",
'password' => "pass",
'trace' => 1,
'connection_timeout' => 500000,
'cache_wsdl' => WSDL_CACHE_BOTH,
'keep_alive' => false,
));
响应 headers 为空。请求headers和以前一样
3. 使用这个:
$connection->__setLocation('https://host.org/file.svc');
没有帮助。但是,当我将位置设置为 WSDL 文件而不是 SVC 文件时,我得到以下响应:
HTTP/1.1 405 Method Not Allowed
Connection: Keep-Alive
Content-Length: 1293
Date: Wed, 23 Dec 2015 14:28:53 GMT
Content-Type: text/html
Server: Microsoft-IIS/7.5
Allow: GET, HEAD, OPTIONS, TRACE
X-Powered-By: ASP.NET
我确定 WSDL 服务不够慢,无法超过超时(Ricardo Velhote 建议的)。
4. 我有一个 XML 配置文件随我之前提到的 C# 示例一起提供:
<client>
<endpoint address="https://host/file.svc" binding="customBinding" bindingConfiguration="custom" contract="ServiceReference1.file" name="custom" />
</client>
<bindings>
<customBinding>
<binding name="custom">
<security defaultAlgorithmSuite="Default" authenticationMode="UserNameOverTransport" requireDerivedKeys="true" includeTimestamp="true" messageSecurityVersion="WSSecurity11WSTrustFebruary2005WSSecureConversationFebruary2005WSSecurityPolicy11BasicSecurityProfile10">
<localClientSettings detectReplays="false" />
<localServiceSettings detectReplays="false" />
</security>
<textMessageEncoding messageVersion="Soap11WSAddressing10" />
<httpsTransport maxReceivedMessageSize="2147483647" maxBufferPoolSize="2147483647" />
</binding>
</customBinding>
</bindings>
我尝试按照建议 here 扩展 SoapClient class,但是您可以猜到它没有用 - 脚本的行为仍然相同。
您在选项中遗漏了 proxy_host
和 proxy_port
...如果您需要代理才能工作,请提供这些参数:
....
$connection = new SoapClient($wsdl_url, array(
'location' => $location,
'soap_version' => SOAP_1_1,
'connection_timeout'=> 600,
'proxy_host' => '....', // Your proxy host
'proxy_port' => 8080, // Your proxy port
'proxy_login' => "my_login",
'proxy_password' => "my_password"
));
根据您的更新和随示例提供的 XML 配置文件,此 Web 服务似乎正在使用 WS-Addressing
,因此常规 SOAPClient 将无法工作,需要对其进行扩展以支持WS-Addressing
.
这是赠送它的线
<textMessageEncoding messageVersion="Soap11WSAddressing10" />
我还没有使用 WS-Addressing
,但我最近正在分析一个 API 项目,我们将在未来开展需要它的项目。
请注意 this project or this other project,这可能会有用(搜索更多 PHP WS-Addressing
很容易)。
同样,我只做过研究,没有任何实践经验可以帮助您编写实际代码:)
[编辑:更新后的过时答案]
首先,您可能被示例代码中变量 proxy
的使用误导了。他们可能指的是 HTTP 基本身份验证而不是代理。
尝试用 login
和 password
替换 proxy_login
和 proxy_password
。
然而,话虽如此,如果您正在获取 WSDL,则意味着至少它正在连接并获取有关服务的信息(这很好)。
在正常情况下,您不需要在 SoapClient 中指定 location
,因为它应该由 WSDL 文件定义。通过设置 location
参数,您将覆盖 WSDL 文件中的设置,并且您可能会将其指向不存在的位置。
尝试从 SoapClient 构造函数中省略 location
和 soap_version
,让库自动处理这些参数:
$connection = new SoapClient($wsdl_url, array('connection_timeout'=> 600,
'proxy_login' => "my_login", 'proxy_password' => "my_password"));
另一方面,也许您正在处理极其缓慢的 Web 服务。 PHP 中有许多参数可能会影响超时所需的时间,而且它们很可能远低于您的 connection_timeout
参数:
使用您并不真正了解的语言进行编码时,您的第一个动作应该是使用手册和示例。
这是 SoapClient 的 link 到 PHP 手册 - http://php.net/manual/en/soapclient.soapclient.php
为了调试 SoapClient,您可以传递 "trace" 参数。引用手册:设置布尔跟踪选项可以使用方法 SoapClient->__getLastRequest、SoapClient->__getLastRequestHeaders、SoapClient->__getLastResponse 和 SoapClient-> __getLastResponseHeaders.
$client = new SoapClient("some.wsdl", array('trace' => true));
因此,对于您的情况,如果您想查看问题所在,请执行以下操作:
$client = SoapClient($wsdl_url, array('trace' => 1));
$result = $client->SomeFunction();
echo "REQUEST HEADERS:\n" . $client->__getLastRequestHeaders() . "\n";
echo "REQUEST:\n" . $client->__getLastRequest() . "\n";
echo "Response headers:\n" . $client->__getLastResponseHeaders() . "\n";
echo "Response:\n" . $client->__getLastResponse() . "\n";
此外,正如在另一个答案中指出的那样 - 您正在设置 proxy_login 和 proxy_password 参数请求,仅当您使用代理服务器连接到该 WSDL 服务时才应使用该请求。