函数执行期间的SOAP客户端超时

时间:2015-12-09 15:47:06

标签: php soap wsdl soap-client

我需要使用SOAP从数据库中检索一些数据。我不是一个经验丰富的PHP程序员,这就是我需要一些帮助的原因。提供webservice(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功能请求(使用凭据创建基本身份验证)后,我收到了响应:An error occurred when verifying security for the message。 选择Authenticate pre-emptively选项并没有帮助。我搜索了这个错误的解决方案,但我找不到任何东西。

2。我为SoapClient课程尝试了几乎所有可以想象的选项组合。以下是其中一些:

$connection = new SoapClient($wsdl_url, array(
    'login' => "login",
    'password' => "pass",
    'trace' => 1,
));

响应标头为空。请求标头:

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,
));

响应标头为空。请求标头与以前一样。

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. 我已经提供了我之前提到的C#示例提供的XML配置文件:

<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类,但是你可以猜测它没有用 - 脚本的行为仍然是相同的。

3 个答案:

答案 0 :(得分:1)

根据您的更新和示例提供的XML配置文件,此Webservice似乎正在使用WS-Addressing,因此常规SOAPClient将无法工作,并且需要对其进行扩展才能支持WS-Addressing

这是放弃它的线

<textMessageEncoding messageVersion="Soap11WSAddressing10" />

我还没有使用WS-Addressing,但我最近正在分析我们将来需要处理的项目的API。

请注意可能有用的this projectthis other project(可以轻松搜索更多PHP WS-Addressing)。

同样,我只做过研究,没有任何动手经验来帮助您使用实际代码:)

[编辑:更新后的过时答案]

首先,您可能会因示例代码中使用变量proxy而误导您。它们可能是指HTTP基本身份验证而不是代理。

尝试使用proxy_loginproxy_password替换loginpassword

然而,话虽如此,如果您获得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参数:

答案 1 :(得分:0)

您在选项中错过了proxy_hostproxy_port ...如果您需要代理工作,请提供theese参数:

....
$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"
));

答案 2 :(得分:0)

使用您不熟悉的语言进行编码时的第一个操作应该是使用手册和示例。

以下是SoapClient上的PHP手册的链接 - http://php.net/manual/en/soapclient.soapclient.php

为了调试SoapClient,您可以传递&#34; trace&#34;论点。从手册中引用:设置布尔跟踪选项可以使用方法SoapClient-&gt; __ getLastRequest,SoapClient-&gt; __ getLastRequestHeaders,SoapClient-&gt; __ getLastResponse和SoapClient-&gt; __ 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服务。