如何使用SoapClient类进行PHP SOAP调用


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:

  • 联络人ID:100
  • 联系人姓名:约翰
  • 概述:油桶
  • 数量:500

正确的语法是什么?我一直在尝试各种选择,但似乎soap结构非常灵活,所以有很多方法可以做到这一点。也无法从手册中找出答案...


更新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,因此我想我需要以某种方式弄清楚我的参数包括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 

Answers:


177

这是您需要做的。

我试图重现这种情况...


  • 对于此示例,我创建了一个.NET示例WebService(WS),其WebMethod名称为,其Function1预期具有以下参数:

功能1(Contact Contact,字符串说明,整数值)

  • 其中Contact仅仅是对getter和setter的模型idname像你的情况。

  • 您可以从以下位置下载.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] =>每桶石油[amount] => 500)

  • 当我调试.NET示例WS时,得到以下信息:

在此处输入图片说明

(如您所见,Contact对象null既不是其他参数也不是其他参数。这意味着您的请求已从PHP端成功完成)

  • .NET示例WS的响应是预期的响应,这就是我在PHP方面获得的响应:

object(stdClass)[3] public'Function1Result'=>字符串'您的请求的详细信息!id:100,名称:John,描述:每桶石油,数量:500'(长度= 98)


编码愉快!


3
完善!我的举止似乎使我对SOAP服务的了解比我实际了解的多,这使我成为了我所需要的地方。
chapman84 2013年

1
我没有问这个问题,否则我会问。这个问题和这个答案虽然得到了我的反对。
chapman84

4
@user应该接受它:)顺便说一句,非常好的答案,完整而且非常清楚。+1
Yann39

谢谢你!努力学习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()函数充当如何进行呼叫的示例。
Salvador P.

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服务与您具有相同的结构,但是它不需要Class作为参数,只是数组。

例如:-我的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
您的示例更有帮助,因为它显示了结构
依赖性

3

读这个;-

http://php.net/manual/zh/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中生成的xml进行比较。

对我来说,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());

您可以查看Request 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中,数组是一种结构,非常灵活。通常,在进行肥皂调用时,我会使用XML包装器,因此不确定它是否会起作用。

编辑:

您可能想尝试的是创建json查询以发送或使用该查询来创建xml购买,其形式如下:http : //onwebdev.blogspot.com/2011/08/php-converting-rss- to-json.html


谢谢,但是也没有用。您如何准确地使用XML包装器,也许比这更容易使用...

首先,您必须确保WSDL可以处理XML包装器。但这是相似的,您使用XML构建请求,并且在大多数情况下使用curl。我已经将SOAP与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); 

为我工作。

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.