SoapClientクラスを使用してPHP SOAP呼び出しを行う方法


130

私はPHPコードの作成に慣れていますが、オブジェクト指向コーディングを使用することはあまりありません。今度は(クライアントとして)SOAPと対話する必要があり、構文を正しく取得できません。SoapClientクラスを使用して新しい接続を適切に設定できるようにするWSDLファイルがあります。しかし、実際に正しい呼び出しを行ってデータを返すことができません。次の(簡略化した)データを送信する必要があります。

  • 連絡先ID
  • 連絡先
  • 概要

WSDLドキュメントには2つの関数が定義されていますが、必要なのは1つだけです(以下の「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を呼び出したいとしましょう:

  • 連絡先ID: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,
);

どちらの場合もエラー:オブジェクトに 'Contact'プロパティがありません `

php  soap 

回答:


178

これはあなたがする必要があることです。

状況を再現してみました...


  • この例では、次のパラメーターを期待してWebMethod呼び出される.NETサンプルWebService(WS)を作成しましたFunction1

Function1(Contact Contact、文字列の説明、int量)

  • あなたのケースでContactゲッターとセッターがidあり、name好きなモデルがあるのはどこですか?

  • .NETサンプル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が期待するよう、次のような出力が表示されます。

配列([Contact] => Contact Object([id] => 100 [name] => John)[description] => Barrel of Oil [amount] => 500)

  • .NETサンプルWSをデバッグすると、次の結果が得られました。

ここに画像の説明を入力してください

(ご覧のとおり、Contactオブジェクトはnull他のパラメータでもありません。つまり、リクエストがPHP側から正常に実行されたことを意味します)

  • .NETサンプルWSからの応答は予想されたものであり、これは私がPHP側で得たものです。

object(stdClass)[3] public 'Function1Result' => string 'リクエストの詳細情報!id:100、name:John、description:Barrel of Oil、amount:500 '(length = 98)


ハッピーコーディング!


3
パーフェクト!私は、SOAPサービスについて実際よりも少し知っているように振る舞いました。
chapman84 2013年

1
私は質問しませんでした。質問とこの答えは私から賛成票を得ました。
chapman84 2013

4
@ユーザーはそれを受け入れる必要があります:) BTW、非常に良い答え、完全で非常に明確。+1
Yann39 2013

これをありがとう!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"] => string(14) "Data Not Found"}何かアイデアはありますか?
Ilker Baltaci 2017年

彼らは都市のひもを変えたようです。彼らが提供する別のサービスへの別の呼び出しで例を更新しましたが、それは機能しています。都市として返される文字列を使用しようとしましたが、うまく機能していないようです。とにかくgetCitiesByCountry()関数は、呼び出し方法の例として機能します。
サルバドールP.17年

29

最初にWebサービスを初期化します。

$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">

ありがとう!私はこれを試しましたが、「オブジェクトに 'Contact'プロパティがありません」というエラーが表示されます。完全な詳細で私の質問を更新します。何か案は?

@ user16441サービスのWSDLとスキーマを投稿できますか?私は通常、サービスが期待するXMLを理解することから始め、次に、WireSharkを使用してクライアントが実際に送信しているものを理解します。
davidfmatheson 2012

21

私のWebサービスがあなたと同じ構造を持っている理由はわかりませんが、パラメーターにクラスは必要ありません。配列だけです。

例:-私の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>

私は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
あなたの例はより有用です、それは構造依存性を示しているためです
vladkras

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からsoapプロジェクトを作成します。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変数には、リクエストに対してSoapClientが作成するxmlが含まれます。このxmlをSoapUIで生成されたものと比較します。

私にとって、SoapClientは連想配列$ paramsのキーを無視し、それをインデックス付き配列として解釈するため、xmlで誤ったパラメーターデータが発生します。つまり、$ paramsのデータを並べ替えると、$ responseは完全に異なります。

$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を確認できます。

SoapClientオプションで[trace = 1、exceptions = 0]を使用します。


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では、配列は構造体であり、非常に柔軟です。通常、soap呼び出しではXMLラッパーを使用するため、機能するかどうかは不明です。

編集:

あなたが試したいのは、送信するjsonクエリを作成するか、それを使用してxmlを作成することです。このページにある次のようなものを購入します。 to-json.html


ありがとう、でもうまくいきませんでした。どのように正確にXMLのラッパーを使用しない、おそらくこれよりも使いやすくなっている...

まず、WSDLがXMLラッパーを処理できることを確認する必要があります。しかし、それは似ています。XMLでリクエストを作成し、ほとんどの場合curlを使用します。SOAP with XMLを使用して、銀行を介したトランザクションを処理しています。これらを出発点として確認できます。forums.digitalpoint.com/showthread.php?t=424619#post4004636 w3schools.com/soap/soap_intro.asp
James Williams


0

getLastRequest():

このメソッドは、トレースオプションをTRUEに設定して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); 

私のために働いた。

弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.