Як здійснити виклик PHP SOAP за допомогою класу SoapClient


130

Я звик писати PHP-код, але не часто використовую об'єктно-орієнтоване кодування. Зараз мені потрібно взаємодіяти з SOAP (як клієнтом) і я не в змозі правильно визначити синтаксис. У мене є файл WSDL, який дозволяє мені правильно встановити нове з'єднання за допомогою класу SoapClient. Однак я не можу насправді здійснити правильний дзвінок та повернути дані. Мені потрібно надіслати такі (спрощені) дані:

  • Ідентифікатор контакту
  • Контактна Особа
  • Загальний опис
  • Сума

У документі WSDL є дві функції, але мені потрібна лише одна ("FirstFunction" нижче). Ось сценарій, який я запускаю, щоб отримати інформацію про доступні функції та типи:

$client = new SoapClient("http://example.com/webservices?wsdl");
var_dump($client->__getFunctions()); 
var_dump($client->__getTypes()); 

А ось результат, який він генерує:

array(
  [0] => "FirstFunction Function1(FirstFunction $parameters)",
  [1] => "SecondFunction Function2(SecondFunction $parameters)",
);

array(
  [0] => struct Contact {
    id id;
    name name;
  }
  [1] => string "string description"
  [2] => string "int amount"
}

Скажіть, я хочу зателефонувати на FirstFunction із даними:

  • Контактний номер: 100
  • Контактне ім'я: Джон
  • Загальний опис: Барель нафти
  • Сума: 500

Який би був правильний синтаксис? Я пробував всілякі варіанти, але, здається, структура мила досить гнучка, тому існує дуже багато способів зробити це. Не вдалося це зрозуміти і з посібника ...


ОНОВЛЕННЯ 1: випробуваний зразок з MMK:

$client = new SoapClient("http://example.com/webservices?wsdl");

$params = array(
  "id" => 100,
  "name" => "John",
  "description" => "Barrel of Oil",
  "amount" => 500,
);
$response = $client->__soapCall("Function1", array($params));

Але я отримую таку відповідь: Object has no 'Contact' property. Як ви бачите на виході getTypes(), є structдзвінок Contact, тож я думаю, мені якось потрібно уточнити, що мої параметри включають Контактні дані, але питання: як?

ОНОВЛЕННЯ 2: Я також спробував ці структури, така ж помилка.

$params = array(
  array(
    "id" => 100,
    "name" => "John",
  ),
  "Barrel of Oil",
  500,
);

Так само, як:

$params = array(
  "Contact" => array(
    "id" => 100,
    "name" => "John",
  ),
  "description" => "Barrel of Oil",
  "amount" => 500,
);

Помилка в обох випадках: у об’єкта немає властивості "Контакт"

php  soap 

Відповіді:


178

Це те, що вам потрібно зробити.

Я намагався відтворити ситуацію ...


  • Для цього прикладу я створив зразок .NET WebService (WS) з WebMethodвикликом, Function1очікуючи наступних параметрів:

Функція1 (Контакт контакту, опис рядка, сума int)

  • Де Contactце просто модель , яка має методи отримання і установки для idі nameяк у вашому випадку.

  • Ви можете завантажити .NET sample WS за адресою:

https://www.dropbox.com/s/6pz1w94a52o5xah/11593623.zip


Код.

Ось що вам потрібно зробити на стороні PHP:

(Випробувано та працює)

<?php
// Create Contact class
class Contact {
    public function __construct($id, $name) 
    {
        $this->id = $id;
        $this->name = $name;
    }
}

// Initialize WS with the WSDL
$client = new SoapClient("http://localhost:10139/Service1.asmx?wsdl");

// Create Contact obj
$contact = new Contact(100, "John");

// Set request params
$params = array(
  "Contact" => $contact,
  "description" => "Barrel of Oil",
  "amount" => 500,
);

// Invoke WS method (Function1) with the request params 
$response = $client->__soapCall("Function1", array($params));

// Print WS response
var_dump($response);

?>

Тестування всієї справи.

  • Якщо ви це зробите, print_r($params)ви побачите такий результат, як очікували б ваші WS:

Масив ([Контакт] => Контактний об'єкт ([id] => 100 [ім'я] => Джон) [опис] => Барель нафти [кількість] => 500)

  • Під час налагодження .NET зразка WS я отримав наступне:

введіть тут опис зображення

(Як ви бачите, Contactоб'єктом не є nullні інші параметри. Це означає, що ваш запит був успішно виконаний з боку PHP)

  • Відповідь від .NET зразка WS була очікуваною, і ось що я отримав на стороні PHP:

object (stdClass) [3] public 'Function1Result' => рядок 'Детальна інформація вашого запиту! id: 100, ім'я: Джон, опис: Барель нафти, кількість: 500 '(довжина = 98)


Щасливе кодування!


3
Ідеально! Я діяв так, ніби знав трохи більше про послуги SOAP, ніж насправді, і це дістало мене там, де я мав бути.
chapman84

1
Я не задавав питання, інакше мав би. Питання і ця відповідь, однак, отримали відгук у мене.
chapman84

4
@user повинен це сприймати :) До речі, дуже приємна відповідь, повна і дуже чітка. +1
Yann39

Дякую за це! Стимул Lil для розуміння структури SOAP.
EatCodePlaySleep

69

Ви також можете користуватися послугами SOAP так:

<?php 
//Create the client object
$soapclient = new SoapClient('http://www.webservicex.net/globalweather.asmx?WSDL');

//Use the functions of the client, the params of the function are in 
//the associative array
$params = array('CountryName' => 'Spain', 'CityName' => 'Alicante');
$response = $soapclient->getWeather($params);

var_dump($response);

// Get the Cities By Country
$param = array('CountryName' => 'Spain');
$response = $soapclient->getCitiesByCountry($param);

var_dump($response);

Це приклад з реальним сервісом, і він працює.

Сподіваюся, це допомагає.


Я отримую таку помилку: object (stdClass) # 70 (1) {["GetWeatherResult"] => рядок (14) "Дані не знайдено"} Будь-яка ідея?
Ilker Baltaci

Здається, вони змінили струни міст. Я щойно оновив приклад ще одним дзвінком до іншої послуги, яку вони надають, і вона працює. Я намагався використовувати рядки, які вони повертають як міста, але, здається, не працює нормально, інакше функція getCitiesByCountry () слугує прикладом того, як здійснити виклик.
Сальвадор П.

30

Спочатку ініціалізуйте веб-сервіси:

$client = new SoapClient("http://example.com/webservices?wsdl");

Потім встановіть і передайте параметри:

$params = array (
    "arg0" => $contactid,
    "arg1" => $desc,
    "arg2" => $contactname
);

$response = $client->__soapCall('methodname', array($params));

Зауважте, що ім'я методу доступне в WSDL як назва операції, наприклад:

<operation name="methodname">

Дякую! Я спробував це, але я отримую помилку "У об'єкта немає властивості" Контакт ". Оновлю моє запитання з усіма деталями. Будь-які ідеї?

@ user16441 Чи можете ви розмістити WSDL та схему служби? Зазвичай я починаю з того, щоб зрозуміти, що очікує XML, а потім за допомогою WireShark з'ясувати, що мій клієнт насправді надсилає.
davidfmatheson

21

Я не знаю, чому мій веб-сервіс має однакову структуру з вами, але йому не потрібен клас для параметра, просто це масив.

Наприклад: - Мій WSDL:

<soapenv:Envelope xmlns:soapenv="http://schemas.xmlsoap.org/soap/envelope/"
                  xmlns:ns="http://www.kiala.com/schemas/psws/1.0">
    <soapenv:Header/>
    <soapenv:Body>
        <ns:createOrder reference="260778">
            <identification>
                <sender>5390a7006cee11e0ae3e0800200c9a66</sender>
                <hash>831f8c1ad25e1dc89cf2d8f23d2af...fa85155f5c67627</hash>
                <originator>VITS-STAELENS</originator>
            </identification>
            <delivery>
                <from country="ES" node=””/>
                <to country="ES" node="0299"/>
            </delivery>
            <parcel>
                <description>Zoethout thee</description>
                <weight>0.100</weight>
                <orderNumber>10K24</orderNumber>
                <orderDate>2012-12-31</orderDate>
            </parcel>
            <receiver>
                <firstName>Gladys</firstName>
                <surname>Roldan de Moras</surname>
                <address>
                    <line1>Calle General Oraá 26</line1>
                    <line2>(4º izda)</line2>
                    <postalCode>28006</postalCode>
                    <city>Madrid</city>
                    <country>ES</country>
                </address>
                <email>gverbruggen@kiala.com</email>
                <language>es</language>
            </receiver>
        </ns:createOrder>
    </soapenv:Body>
</soapenv:Envelope>

I var_dump:

var_dump($client->getFunctions());
var_dump($client->getTypes());

Ось результат:

array
  0 => string 'OrderConfirmation createOrder(OrderRequest $createOrder)' (length=56)

array
  0 => string 'struct OrderRequest {
 Identification identification;
 Delivery delivery;
 Parcel parcel;
 Receiver receiver;
 string reference;
}' (length=130)
  1 => string 'struct Identification {
 string sender;
 string hash;
 string originator;
}' (length=75)
  2 => string 'struct Delivery {
 Node from;
 Node to;
}' (length=41)
  3 => string 'struct Node {
 string country;
 string node;
}' (length=46)
  4 => string 'struct Parcel {
 string description;
 decimal weight;
 string orderNumber;
 date orderDate;
}' (length=93)
  5 => string 'struct Receiver {
 string firstName;
 string surname;
 Address address;
 string email;
 string language;
}' (length=106)
  6 => string 'struct Address {
 string line1;
 string line2;
 string postalCode;
 string city;
 string country;
}' (length=99)
  7 => string 'struct OrderConfirmation {
 string trackingNumber;
 string reference;
}' (length=71)
  8 => string 'struct OrderServiceException {
 string code;
 OrderServiceException faultInfo;
 string message;
}' (length=97)

Отже, у моєму коді:

    $client  = new SoapClient('http://packandship-ws.kiala.com/psws/order?wsdl');

    $params = array(
        'reference' => $orderId,
        'identification' => array(
            'sender' => param('kiala', 'sender_id'),
            'hash' => hash('sha512', $orderId . param('kiala', 'sender_id') . param('kiala', 'password')),
            'originator' => null,
        ),
        'delivery' => array(
            'from' => array(
                'country' => 'es',
                'node' => '',
            ),
            'to' => array(
                'country' => 'es',
                'node' => '0299'
            ),
        ),
        'parcel' => array(
            'description' => 'Description',
            'weight' => 0.200,
            'orderNumber' => $orderId,
            'orderDate' => date('Y-m-d')
        ),
        'receiver' => array(
            'firstName' => 'Customer First Name',
            'surname' => 'Customer Sur Name',
            'address' => array(
                'line1' => 'Line 1 Adress',
                'line2' => 'Line 2 Adress',
                'postalCode' => 28006,
                'city' => 'Madrid',
                'country' => 'es',
                ),
            'email' => 'test.ceres@yahoo.com',
            'language' => 'es'
        )
    );
    $result = $client->createOrder($params);
    var_dump($result);

але це успішно!


1
Ваш приклад є більш корисним, тому що він показує структурні
залежності

3

Прочитай це;-

http://php.net/manual/en/soapclient.call.php

Або

Це хороший приклад для функції SOAP "__call". Однак це застаріло.

<?php
    $wsdl = "http://webservices.tekever.eu/ctt/?wsdl";
    $int_zona = 5;
    $int_peso = 1001;
    $cliente = new SoapClient($wsdl);
    print "<p>Envio Internacional: ";
    $vem = $cliente->__call('CustoEMSInternacional',array($int_zona, $int_peso));
    print $vem;
    print "</p>";
?>

3

По-перше, використовуйте SoapUI для створення свого проекту мила з wsdl. Спробуйте надіслати запит, щоб грати з операціями wsdl. Слідкуйте за тим, як xml запит складається з ваших полів даних.

І тоді, якщо у вас виникають проблеми з тим, щоб SoapClient діяв так, як вам хочеться, ось як я це налагоджую. Встановіть опцію трасування так, щоб функція __getLastRequest () доступний для використання.

$soapClient = new SoapClient('http://yourwdsdlurl.com?wsdl', ['trace' => true]);
$params = ['user' => 'Hey', 'account' => '12345'];
$response = $soapClient->__soapCall('<operation>', $params);
$xml = $soapClient->__getLastRequest();

Тоді змінна $ xml містить xml, який SoapClient складає для вашого запиту. Порівняйте цей xml з створеним у SoapUI.

Для мене SoapClient, здається, ігнорує ключі асоціативного масиву $ params та інтерпретує його як індексований масив, викликаючи неправильні дані параметрів у xml. Тобто, якщо я переупорядковую дані в $ params , $ відповідь буде зовсім іншим:

$params = ['account' => '12345', 'user' => 'Hey'];
$response = $soapClient->__soapCall('<operation>', $params);

3

Якщо ви створили об’єкт SoapParam, це вирішить вашу проблему. Створіть клас та відображіть його з типом об'єкта, заданим WebService, ініціалізуйте значення та надішліть запит. Дивіться зразок нижче.

struct Contact {

    function Contact ($pid, $pname)
    {
      id = $pid;
      name = $pname;
  }
}

$struct = new Contact(100,"John");

$soapstruct = new SoapVar($struct, SOAP_ENC_OBJECT, "Contact","http://soapinterop.org/xsd");

$ContactParam = new SoapParam($soapstruct, "Contact")

$response = $client->Function1($ContactParam);

1

У мене було те саме питання, але я просто обгорнув такі аргументи, і він працює зараз.

    $args = array();
    $args['Header'] = array(
        'CustomerCode' => 'dsadsad',
        'Language' => 'fdsfasdf'
    );
    $args['RequestObject'] = $whatever;

    // this was the catch, double array with "Request"
    $response = $this->client->__soapCall($name, array(array( 'Request' => $args )));

Використовуючи цю функцію:

 print_r($this->client->__getLastRequest());

Ви можете побачити Запит XML, змінюється він чи ні, залежно від ваших аргументів.

Використовуйте [слід = 1, винятки = 0] у параметрах SoapClient.


0

Вам потрібно оголосити клас Контракт

class Contract {
  public $id;
  public $name;
}

$contract = new Contract();
$contract->id = 100;
$contract->name = "John";

$params = array(
  "Contact" => $contract,
  "description" => "Barrel of Oil",
  "amount" => 500,
);

або

$params = array(
  $contract,
  "description" => "Barrel of Oil",
  "amount" => 500,
);

Тоді

$response = $client->__soapCall("Function1", array("FirstFunction" => $params));

або

$response = $client->__soapCall("Function1", $params);

0

Вам потрібен багатовимірний масив, ви можете спробувати наступне:

$params = array(
   array(
      "id" => 100,
      "name" => "John",
   ),
   "Barrel of Oil",
   500
);

у PHP масив - це структура і дуже гнучка. Зазвичай з мильними дзвінками я використовую обгортку XML настільки впевненою, чи спрацює вона.

Редагувати:

Ви можете спробувати створити запит на json, щоб надіслати або використовувати його для створення xml-купівлі такого типу, як описано на цій сторінці: http://onwebdev.blogspot.com/2011/08/php-converting-rss- до-json.html


дякую, але і це не вийшло. Як ви точно використовуєте обгортки XML, можливо, це простіше у використанні, ніж це ...

Спочатку ви повинні переконатися, що ваш WSDL може обробляти XML обгортки. Але це схоже, ви будуєте запит у XML і в більшості випадків використовуєте curl. Я використовую SOAP з XML для обробки транзакцій через банки. Ви можете перевірити їх як початковий пункт. forums.digitalpoint.com/showthread.php?t=424619#post4004636 w3schools.com/soap/soap_intro.asp
Джеймс Вільямс

0

Існує можливість генерувати об’єкти php5 з класом WsdlInterpreter. Детальніше дивіться тут: https://github.com/gkwelding/WSDLInterpreter

наприклад:

require_once 'WSDLInterpreter-v1.0.0/WSDLInterpreter.php';
$wsdlLocation = '<your wsdl url>?wsdl';
$wsdlInterpreter = new WSDLInterpreter($wsdlLocation);
$wsdlInterpreter->savePHP('.');

0

getLastRequest ():

Цей метод працює лише в тому випадку, якщо об’єкт SoapClient був створений з параметром сліду, встановленим на TRUE.

ІСТИНА в цьому випадку представлена ​​1

$wsdl = storage_path('app/mywsdl.wsdl');
try{

  $options = array(
               // 'soap_version'=>SOAP_1_1,
               'trace'=>1,
               'exceptions'=>1,

                'cache_wsdl'=>WSDL_CACHE_NONE,
             //   'stream_context' => stream_context_create($arrContextOptions)
        );
           // $client = new \SoapClient($wsdl, array('cache_wsdl' => WSDL_CACHE_NONE) );
        $client = new \SoapClient($wsdl, array('cache_wsdl' => WSDL_CACHE_NONE));
        $client     = new \SoapClient($wsdl,$options); 

працював на мене.

Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.