Таймауты клиента SOAP во время выполнения функции

мне нужно использовать 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 ждет до конца. После того, как он превышает тайм-аут, я получаю сообщение об ошибке: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 недостаточно медленная, чтобы превысить тайм-аут (Рикардо 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>

я попытался расширить класс SoapClient, как было предложено здесь, но, как вы можете догадаться, это не сработало-поведение сценария по-прежнему то же самое.

3 ответов


в соответствии с вашим обновлением и файлом конфигурации XML, приведенным в Примере, этот веб-сервис, похоже, использует WS-Addressing таким образом, обычный SOAPClient не будет работать и требует его расширения для поддержки WS-Addressing.

это строка, которая выдает его

<textMessageEncoding messageVersion="Soap11WSAddressing10" />

я еще не использовал WS-Addressing но недавно я анализировал API для проекта, над которым мы будем работать в будущем, который требует этого.

Пожалуйста, примите к сведению этот проект или другой проект что может быть полезно (легко искать больше PHP WS-Addressing).

опять же, я только провел исследование и не имею практического опыта, чтобы помочь вам с фактическим кодом:)


[EDIT: устаревший ответ после обновления]

прежде всего, вас может ввести в заблуждение использование переменной proxy в примере кода. Они, вероятно, ссылаясь на HTTP Обычная проверка подлинности, а не прокси.

попробуйте заменить proxy_login и proxy_password С login и password.

однако, сказав это, если вы получаете WSDL, это означает, что, по крайней мере, он подключается и получает информацию об услуге (что хорошо).

в нормальных ситуациях вам не нужно указывать location в SoapClient, как это должно быть определено файлом WSDL. Установив location параметр вы переопределяете то, что установлено в файле WSDL, и вы можете указать его на место, которое не существует.

попробуйте ommiting location и soap_version из конструктора SoapClient и пусть библиотека обрабатывает эти параметры автоматически:

$connection = new SoapClient($wsdl_url, array('connection_timeout'=> 600,
    'proxy_login' => "my_login", 'proxy_password' => "my_password"));

С другой стороны, возможно, вы имеете дело с крайне медленным веб-службы. В PHP есть много параметров, которые могут влиять на время ожидания, и, скорее всего, они намного ниже вашего connection_timeout параметр:


вы пропустили 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"
));

ваше первое действие при кодировании на языке, который вы действительно не знаете, должно использовать руководства и примеры.

вот ссылка на руководство PHP по SoapClient -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.