Подтвердить что ты не робот

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

Мне нужно использовать SOAP для извлечения некоторых данных из базы данных. Я не опытный программист PHP, поэтому мне нужна помощь. Компания, предоставляющая webservice (WSDL), предоставила мне информацию для входа и ссылки на файлы svc и wsdl. Они также дали мне пример в С# о том, как подключиться:

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, поэтому она действительна. Я пробовал другие функции, предоставляемые службой - результат всегда один и тот же: script просто зависает. И я имею в виду, что от службы нет ответа, и PHP ждет завершения функции loggedIn(). После превышения таймаута я получаю сообщение об ошибке: Error Fetching http headers in... Что я делаю не так? Как его отладить?

UPDATE:

Я пробовал каждую вещь, которую вы, ребята, предложили. Но мне все еще не удалось решить проблему. Я не использую прокси. Вы можете найти результаты ниже:

1. Я установил SoapUI. После настройки запроса для функции some_method (создания базового Auth с учетными данными) я получил ответ: 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 не достаточно медленная, чтобы превысить тайм-аут (предложил Рикардо Велоте).

4. У меня есть файл конфигурации 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>

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

4b9b3361

Ответ 1

Согласно вашему обновлению и файлу конфигурации XML, представленному в примере, этот Webservice использует 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, и вы можете указывать его на место, которое не существует.

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

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

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

Ответ 2

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

Ответ 3

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

Вот ссылка на PHP Manual на 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.