【问题标题】:SOAP client timeouts during function execution函数执行期间的 SOAP 客户端超时
【发布时间】:2015-12-09 15:47:05
【问题描述】:

我需要使用 SOAP 从数据库中检索一些数据。我不是一个经验丰富的 PHP 程序员,所以我需要一些帮助。提供 Web 服务 (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.我有一个 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 类,但您可以猜到它不起作用 - 脚本的行为仍然相同。

【问题讨论】:

  • 只是一个建议,但下载一份 SOAP UI 并将其指向 Web 服务,然后创建一个示例请求。一旦您的请求在 SOAP UI 上运行,我将捕获 PHP 程序(使用 Wireshark)发出的 SOAP 请求,看看是否能发现差异。可能是请求格式不正确并导致问题。
  • 看看stackoverflow.com/questions/19010165/…可能对你有帮助
  • 是否必须使用原生 PHP 的 SoapClient?可以用 nu-soap 之类的第三方库吗?
  • @NaijaProgrammer 是的,我可以使用外部库。到目前为止,我无法测试所有的答案和建议(我正在度假)。但我发现提供proxy_hostproxy_port 属性,或将proxy_loginproxy_password 更改为loginpassword 并不能解决问题。同样删除 locationsoap_version 属性根本不起作用。当我从假期回来时,我将尝试使用 SOAP UI 并按照@Namphibian 的建议嗅探 SOAP 请求。
  • @user3125731 在最近的更新之后,我编辑了我的答案。根据您的更新和随示例提供的 XML 配置文件,此 Web 服务似乎正在使用 WS-Addressing,因此常规 SOAPClient 将不起作用。

标签: php soap wsdl soap-client


【解决方案1】:

根据您的更新和随示例提供的 XML 配置文件,此 Web 服务似乎正在使用 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 参数:

【讨论】:

  • 好的,谢谢您的回答。至少我知道使用 SOAPClient 是没有意义的。我会检查你链接的项目。
  • @user3125731 抱歉,我没有在回答中说清楚(将编辑)。我提供的项目是 SOAPClient 的扩展而不是替代品:)
【解决方案2】:

您在选项中错过了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"
));

【讨论】:

    【解决方案3】:

    使用您并不真正了解的语言进行编码时,您的第一个动作应该是使用手册和示例。

    这是 SoapClient 上 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 服务。

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2016-06-20
      • 1970-01-01
      • 2014-09-13
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多