函数执行期间 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_hostproxy_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 基本身份验证而不是代理。

尝试用 loginpassword 替换 proxy_loginproxy_password

然而,话虽如此,如果您正在获取 WSDL,则意味着至少它正在连接并获取有关服务的信息(这很好)。

在正常情况下,您不需要在 SoapClient 中指定 location,因为它应该由 WSDL 文件定义。通过设置 location 参数,您将覆盖 WSDL 文件中的设置,并且您可能会将其指向不存在的位置。

尝试从 SoapClient 构造函数中省略 locationsoap_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_loginproxy_password 参数请求,仅当您使用代理服务器连接到该 WSDL 服务时才应使用该请求。