Howto:SOAP API PHP5 Sample Code: Difference between revisions
|  fix link to SOAPClient PHP Lib | |||
| (35 intermediate revisions by 6 users not shown) | |||
| Line 2: | Line 2: | ||
| This information applies to | This information applies to | ||
| *  | * V9 and later innovaphone PBX platform | ||
| * web server running PHP 5 | * web server running PHP 5 | ||
| Line 13: | Line 13: | ||
| You will need php with the SoapClient class available. | You will need php with the SoapClient class available. | ||
| The innovaphone PBX wsdl file is required.  See [[ | The innovaphone PBX wsdl file is required.  See [[Reference8:SOAP API]] for information how to obtain this file.   | ||
| To access older WSDL versions, you will need to change the WSDL URL setting (in <code>const ___wsdl</code>) and adapt to the required number of version booleans in the call to <code>Initialize()</code> in the pbx wrapper class. | |||
| If you have access to the internet, you can try out the sample code given below right away.  Of course you may also adapt the sample script configuration and try with your own innovaphone PBX. | If you have access to the internet, you can try out the sample code given below right away.  Of course you may also adapt the sample script configuration and try with your own innovaphone PBX. | ||
| ===Running Sample Code=== | ===Running the Sample Code=== | ||
| To demonstrate SOAP usage with PHP, a simple script is presented that will allow you to initiate a call using a web page. | To demonstrate SOAP usage with PHP, a simple script is presented that will allow you to initiate a call using a web page. | ||
| The following class '''innoPBX''' is used to wrap the (somewhat clumsy) PHP SOAP access mechanism: | The following class '''innoPBX''' is used to wrap the (somewhat clumsy) PHP SOAP access mechanism: | ||
| < | <code php> | ||
| <?php | <?php | ||
| Line 39: | Line 40: | ||
|      ); |      ); | ||
|      const ___wsdl = 'http://www.innovaphone.com/wsdl/ |      const ___wsdl = 'http://www.innovaphone.com/wsdl/pbx900.wsdl'; | ||
|      // class constructor |      // class constructor | ||
| Line 46: | Line 47: | ||
| 	$httpu,		// the HTTP user id (e.g. "admin") | 	$httpu,		// the HTTP user id (e.g. "admin") | ||
| 	$httpp,		// the HTTP password (e.g. "ip800") | 	$httpp,		// the HTTP password (e.g. "ip800") | ||
| 	$user,		// the PBX user CN to work with   | 	$user = null,		// the PBX user CN to work with   | ||
| 	$options = null, | 	$options = null, | ||
| 			// extra or overriding options for SOAPClient::__construct | 			// extra or overriding options for SOAPClient::__construct | ||
| Line 64: | Line 65: | ||
| 	parent::__construct($wsdl, $usedoptions); | 	parent::__construct($wsdl, $usedoptions); | ||
| 	// get the connection | 	// get the connection (using and activating v9 wsdl) | ||
| 	$init = $this->Initialize($user, "PHP SOAP Wrapper"); | 	$init = $this->Initialize($user, "PHP SOAP Wrapper", true, true, true, true, true); | ||
| 	$this->___key = $init['key']; | 	$this->___key = $init['key']; | ||
| 	$this->___session = $init['return']; | 	$this->___session = $init['return']; | ||
| Line 76: | Line 77: | ||
| ?> | ?> | ||
| </ | </code> | ||
| Save this code into a file named '''innopbx.class.php''' on your web server.  This class is of course not required to use PHP SOAP with the innovaphone PBX API.  However, it comes handy and should be generic enough to be useful in most cases. | Save this code into a file named '''innopbx.class.php''' on your web server.  This class is of course not required to use PHP SOAP with the innovaphone PBX API.  However, it comes handy and should be generic enough to be useful in most cases. | ||
| The SOAP client uses the following code: | The SOAP client uses the following code: | ||
| < | <code php> | ||
| <?php | <?php | ||
| Line 99: | Line 100: | ||
| // config, adapt to your need | // config, adapt to your need | ||
| // this data below will work with innovaphones demo PBX.  Just register 2 phones as user-3 and user-4 and hve fun! | // this data below will work with innovaphones demo PBX.  Just register 2 phones as user-3 and user-4 and hve fun! | ||
| $server = " | $server = "demo.innovaphone.com"; | ||
| $user = "SOAP"; | $user = "SOAP"; | ||
| $httpu = "demo"; | $httpu = "demo"; | ||
| Line 131: | Line 132: | ||
| function showInfos(&$poll, $head, $cn = "", $user = "", $call = "") { | function showInfos(&$poll, $head, $cn = "", $user = "", $call = "") { | ||
|      print $head . "\n"; |      print $head . "\n"; | ||
|      if (cn !== null) { |      if ($cn !== null) { | ||
| 	print count($poll->user) . " UserInfos\n"; | 	print count($poll->user) . " UserInfos\n"; | ||
| 	foreach($poll->user as $ui) { | 	foreach($poll->user as $ui) { | ||
| Line 144: | Line 145: | ||
| 	    if ((($user === "") || ($user == $ci->user)) && | 	    if ((($user === "") || ($user == $ci->user)) && | ||
| 		(($call === "") || ($call == $ci->call))) { | 		(($call === "") || ($call == $ci->call))) { | ||
| 		    print "    {$ci->user}/{$ci->call} (remote {$ci->h323} #{$ci->e164}) msg {$ci->msg}\n"; | 		    print "    {$ci->user}/{$ci->call} {$ci->No[1]->h323} #{$ci->No[1]->e164} (remote {$ci->No[0]->h323} #{$ci->No[0]->e164}) msg {$ci->msg}\n"; | ||
| 		} | 		} | ||
| 	} | 	} | ||
| Line 153: | Line 154: | ||
| // You cannot assume that you will receive this list within a certain number of Poll results, so please iterate | // You cannot assume that you will receive this list within a certain number of Poll results, so please iterate | ||
| print "Retrieving User list for "; foreach ($v as $name => $value) print "\n  $name=$value "; print "...\n\n"; | print "Retrieving User list for "; foreach ($v as $name => $value) print "\n  $name=$value "; print "...\n\n"; | ||
| $seen =  | $seen = false; | ||
| $i = 1; | $i = 1; | ||
| while (!$seen) { | while (!$seen) { | ||
| Line 188: | Line 189: | ||
| $done = false; | $done = false; | ||
| while (!$done) { | while (!$done) { | ||
|      ob_flush();   // this is  |      ob_flush();   // this is just to see the progress on the call states | ||
|      $p = $inno->Poll($inno->session()); |      $p = $inno->Poll($inno->session()); | ||
|      showInfos($p, "Call States #$i for #{$call}", $caller, $uhandle, $call); $i++; |      showInfos($p, "Call States #$i for #{$call}", $caller, $uhandle, $call); $i++; | ||
| Line 207: | Line 208: | ||
| ?> | ?> | ||
| </ | </code> | ||
| Save this into a file named '''innopbx.sample.php''' in the same directory of your web server. | Save this into a file named '''innopbx.sample.php''' in the same directory of your web server. | ||
| To run the code, you need to examine and possibly adapt the configuration parameters in '''innopbx.sample.php''': | To run the code, you need to examine and possibly adapt the configuration parameters in '''innopbx.sample.php''': | ||
| < | <code php> | ||
| // config, adapt to your need | // config, adapt to your need | ||
| // this data below will work with innovaphones demo PBX.  Just register 2 phones as user-3 and user-4 and hve fun! | // this data below will work with innovaphones demo PBX.  Just register 2 phones as user-3 and user-4 and hve fun! | ||
| $server = " | $server = "demo.innovaphone.com"; | ||
| $user = "SOAP"; | $user = "SOAP"; | ||
| $httpu = "demo"; | $httpu = "demo"; | ||
| Line 219: | Line 220: | ||
| $caller = "PBX User Four"; | $caller = "PBX User Four"; | ||
| $destination = "130"; | $destination = "130"; | ||
| </ | </code> | ||
| This configuration will work right away with our publicly available [http:// | This configuration will work right away with our publicly available [http://demo.innovaphone.com demo PBX].  You simply need to register 2 phones with the '''user-3''' and '''user-4''' users. | ||
| Of course you can also use your own PBX.  Set the configuration parameters as follow: | Of course you can also use your own PBX.  Set the configuration parameters as follow: | ||
| Line 249: | Line 250: | ||
| The constructor takes the following arguments: | The constructor takes the following arguments: | ||
| < | <code php> | ||
| public function __construct($server, $httpu, $httpp, $user, $options = null, $wsdl = null); | public function __construct($server, $httpu, $httpp, $user, $options = null, $wsdl = null); | ||
| </ | </code> | ||
| {| | {| | ||
| Line 257: | Line 258: | ||
| |- | |- | ||
| | $options   | | $options   | ||
| | an optional array of further options to SOAPClient::__construct.  Normally not required.  See [ | | an optional array of further options to SOAPClient::__construct.  Normally not required.  See [https://www.php.net/manual/en/class.soapclient.php the PHP SOAPClient class documentation] for details. | ||
| |- | |- | ||
| | wsdl | | $wsdl | ||
| | the optional URI of the pbx wsdl file.  Please note that you should use ''' | | the optional URI of the pbx wsdl file.  Please note that you should use '''pbx10_00.wsdl''' instead of '''pbx.wsdl''', as this is the more recent version of the wsdl.  By default, the wsdl is received from the innovaphone web and (subject to PHP configuration options) cached in PHP's wsdl cache.  If you intend to supply it locally (which is recommended, as you otherwise depend on www.innovaphone.com being available each time your program runs), use a '''file://''' URI. | ||
| |} | |} | ||
| Calls to PBX API functions defined in the wsdl can be done directly through an instance of '''class innoPBX'''.  These member functions - which are neither defined in '''class innoPBX''' nor in '''class SOAPClient''' - are created automagically by reading the wsdl: | Calls to PBX API functions defined in the wsdl can be done directly through an instance of '''class innoPBX'''.  These member functions - which are neither defined in '''class innoPBX''' nor in '''class SOAPClient''' - are created automagically by reading the wsdl: | ||
| < | <code php> | ||
| $v = $inno->Version(); | $v = $inno->Version(); | ||
| </ | </code> | ||
| To see the functions available and their PHP signatures, you may call | To see the functions available and their PHP signatures, you may call | ||
| < | <code php> | ||
| $inno = new innoPBX(...); | $inno = new innoPBX(...); | ||
| print_r($inno->$inno->__getFunctions()); | print_r($inno->$inno->__getFunctions()); | ||
| </ | </code> | ||
| Many PBX SOAP API functions required either the ''session-key'' or ''session-handle''.   These are available from the class instance for convenience: | Many PBX SOAP API functions required either the ''session-key'' or ''session-handle''.   These are available from the class instance for convenience: | ||
| < | <code php> | ||
| $p = $inno->Poll($inno->session()); | $p = $inno->Poll($inno->session()); | ||
| </ | </code> | ||
| ==== How to use the innovaphone PBX API wrapper class with SSL ==== | |||
| {{3rd Party Input}} | |||
| In order to make a secure connection to the PBX, you have to change the '''inno.class.php''' as follows: | |||
| <code php>'location' => "https://$server/PBX0/user.soap"</code> | |||
| ''Note'': in '''PHP 5.6''' all encrypted client streams now '''enable peer verification by default'''. By default, this will use OpenSSL's default CA bundle to verify the peer certificate. | |||
| See: [http://php.net/manual/en/soapclient.getfunctions.php OpenSSL changes in PHP 5.6.x] | |||
| To disable peer verification for the soap connection, you have to pass this as an option to the constructor: | |||
| <code php> | |||
| $context = stream_context_create(array( | |||
|  'ssl' => array( | |||
|   'verify_peer' => false, | |||
|   'verify_peer_name' => false, | |||
|   'allow_self_signed' => false | |||
|  ) | |||
| )); | |||
| $options = array('stream_context' => $context);  | |||
| $inno = new innoPBX($server, $httpu, $httpp, $user, $options, $wsdl);  | |||
| </code> | |||
| More information on stream_context_create and ssl options: | |||
| [http://php.net/stream_context_create stream_context_create] and | |||
| [http://php.net/manual/en/context.ssl.php SSL context options] | |||
| ====How PHP maps SOAP parameters to method parameters and results==== | ====How PHP maps SOAP parameters to method parameters and results==== | ||
| Line 283: | Line 313: | ||
| If a SOAP method returns a single item, then it is returned by the function directly with appropriate type: | If a SOAP method returns a single item, then it is returned by the function directly with appropriate type: | ||
| < | <code php> | ||
| int UserInitialize(int $session, string $user); | int UserInitialize(int $session, string $user); | ||
| </ | </code> | ||
| The single return value from the '''UserInitialize''' SOAP method is returned as '''int''' by the '''UserInitialize''' '''innoPBX'''-method. | The single return value from the '''UserInitialize''' SOAP method is returned as '''int''' by the '''UserInitialize''' '''innoPBX'''-method. | ||
| If the SOAP methods returns multiple items, then the corresponding class method will return an array: | If the SOAP methods returns multiple items, then the corresponding class method will return an array: | ||
| < | <code php> | ||
| array Initialize(string $user, string $appl); | array Initialize(string $user, string $appl); | ||
| </ | </code> | ||
| will return an array such as | will return an array such as | ||
| < | <code php> | ||
| array( | array( | ||
|      ['return'] => value |      ['return'] => value | ||
|      ['key'] => value |      ['key'] => value | ||
| ) | ) | ||
| </ | </code> | ||
| In the method signature printed out by '''__getFunctions()''' this looks like | In the method signature printed out by '''__getFunctions()''' this looks like | ||
| < | <code php> | ||
| list(int $return, int $key) Initialize(string $user, string $appl) | list(int $return, int $key) Initialize(string $user, string $appl) | ||
| </ | </code> | ||
| Line 310: | Line 340: | ||
| A new '''innoPBX''' class instance object '''$inno''' is created using the values from the configuration variables discussed above. | A new '''innoPBX''' class instance object '''$inno''' is created using the values from the configuration variables discussed above. | ||
| < | <code php> | ||
| $inno = new innoPBX($server, $httpu, $httpp, $user, | $inno = new innoPBX($server, $httpu, $httpp, $user, | ||
| 	    array('classmap' => array("UserInfo" => "innoUserInfo",   | 	    array('classmap' => array("UserInfo" => "innoUserInfo",   | ||
| Line 319: | Line 349: | ||
| 				"Info" => "innoInfo", | 				"Info" => "innoInfo", | ||
| 			    ))); | 			    ))); | ||
| </ | </code> | ||
| The option fifth argument will add a '''classmap''' entry to the options passed to '''SOAPClient::__construct'''.  If this is done, the SOAP class will use your defined classes when returning values to the caller.  Otherwise, structured return values will be indexed arrays (using the element names as array indexes).    This is neat, but not required for the class to function. | The option fifth argument will add a '''classmap''' entry to the options passed to '''SOAPClient::__construct'''.  If this is done, the SOAP class will use your defined classes when returning values to the caller.  Otherwise, structured return values will be indexed arrays (using the element names as array indexes).    This is neat, but not required for the class to function. | ||
| Line 326: | Line 356: | ||
| The   | The   | ||
| < | <code php> | ||
| while (!$seen) { | while (!$seen) { | ||
|      $p = $inno->Poll($inno->session()); |      $p = $inno->Poll($inno->session()); | ||
|      ... |      ... | ||
| </ | </code> | ||
| loop will retrieve and show all these entries.    | loop will retrieve and show all these entries.    | ||
| To be able to perform calls, a user session must be opened by | To be able to perform calls, a user session must be opened by | ||
| < | <code php> | ||
| $uhandle = $inno->UserInitialize($inno->session(), $caller); | $uhandle = $inno->UserInitialize($inno->session(), $caller); | ||
| </ | </code> | ||
| The '''$uhandle''' retrieved will identify the user to be worked with on subsequent calls. | The '''$uhandle''' retrieved will identify the user to be worked with on subsequent calls. | ||
| A call on behalf of '''$caller''' is then initiated: | A call on behalf of '''$caller''' is then initiated: | ||
| < | <code php> | ||
| $call = $inno->UserCall($uhandle, null, $destination, null, 0, array()); | $call = $inno->UserCall($uhandle, null, $destination, null, 0, array()); | ||
| </ | </code> | ||
| When '''UserCall''' returns a non-zero call handle, the call is initiated and can be monitored by examining '''CallInfo''' events returned by subsequent '''Poll()''' calls: | When '''UserCall''' returns a non-zero call handle, the call is initiated and can be monitored by examining '''CallInfo''' events returned by subsequent '''Poll()''' calls: | ||
| < | <code php> | ||
| while (!$done) { | while (!$done) { | ||
|      $p = $inno->Poll($inno->session()); |      $p = $inno->Poll($inno->session()); | ||
| </ | </code> | ||
| this will retrieve events from the PBX   | this will retrieve events from the PBX   | ||
| < | <code php> | ||
| 	if (($ci->user == $uhandle) &&   | 	if (($ci->user == $uhandle) &&   | ||
| 	    ($ci->call == $call) && | 	    ($ci->call == $call) && | ||
| Line 361: | Line 391: | ||
| 		print "\nCall terminated!\n"; | 		print "\nCall terminated!\n"; | ||
| 		$done = true; | 		$done = true; | ||
| </ | </code> | ||
| until a '''del''' event is seen for the call previously created by UserCall.  Please note that any number of call events may occur, e.g. because someone else called $caller.  You cannot assume that all events retrieved related to the call you created.  Also, any number of '''UserInfo''' events may be retrieved. | until a '''del''' event is seen for the call previously created by UserCall.  Please note that any number of call events may occur, e.g. because someone else called $caller.  You cannot assume that all events retrieved are related to the call you created.  Also, any number of '''UserInfo''' events may be retrieved. | ||
| Please note that there is no need to consume all of the events related to the call for it to work.  However, you must not terminate the session before the call is actually proceeding.  If you do so, strange things will happen.  To be sure the call proceeds, it is ok to wait for ''just any'' '''CallInfo''' event related to the call. You can then safely terminate your session and exit the web page. | Please note that there is no need to consume all of the events related to the call for it to work.  However, you must not terminate the session before the call is actually proceeding.  If you do so, strange things will happen.  To be sure the call proceeds, it is ok to wait for ''just any'' '''CallInfo''' event related to the call. You can then safely terminate your session and exit the web page. | ||
| ===Related Articles | ==Working with Call States from CallInfo== | ||
| [[Reference:SOAP API]] | The ''state'' field of the [[Reference10:Concept_SOAP_API#CallInfo|''CallInfo'']] contains a bit field encoded information about direction of the call, hold information and an actual call state. | ||
| Here is an example how to get relevant states: | |||
| <code php> | |||
| <?php | |||
| // decimal state value from CallInfo | |||
| // $callstate = 2; | |||
| // call state? | |||
| switch ($callstate & 0xF) { | |||
|     case 1: | |||
|         echo ("setup\r\n"); | |||
|         break; | |||
|     case 2: | |||
|         echo ("setup-ack\r\n"); | |||
|         break; | |||
|     case 3: | |||
|         echo ("call-proc\r\n"); | |||
|         break; | |||
|     case 4: | |||
|         echo ("Alert\r\n"); | |||
|         break; | |||
|     case 5: | |||
|         echo ("Connect\r\n"); | |||
|         break; | |||
|     case 6: | |||
|         echo ("disconnect sent\r\n"); | |||
|         break; | |||
|     case 7: | |||
|         echo ("disconnect received\r\n"); | |||
|         break; | |||
|     case 8: | |||
|         echo ("parked\r\n"); | |||
|         break; | |||
|     default: | |||
|         echo ("UNKNOWN STATE: " . ($callstate & 0xF) . "\r\n"); | |||
|         break; | |||
| } | |||
| // call direction? | |||
| switch ($callstate & 0x80) { | |||
|     case 0: | |||
|         echo ("inbound call\r\n"); | |||
|         break; | |||
|     case 128: | |||
|         echo ("outbound call\r\n"); | |||
|         break; | |||
|     default: | |||
|         echo ("UNKNOWN STATE: " . ($callstate & 0x80) . "\r\n"); | |||
|         break; | |||
| } | |||
| // is call on hold? | |||
| switch ($callstate & 0x100) { | |||
|     case 0: | |||
|         echo ("active call\r\n"); | |||
|         break; | |||
|     case 256: | |||
|         echo ("call on hold\r\n"); | |||
|         break; | |||
|     default: | |||
|         echo ("UNKNOWN STATE: " . ($callstate & 0x100) . "\r\n"); | |||
|         break; | |||
| } | |||
| // is call being held by peer? | |||
| switch ($callstate & 0x200) { | |||
|     case 0: | |||
|         echo ("active call\r\n"); | |||
|         break; | |||
|     case 512: | |||
|         echo ("active call put on hold by peer\r\n"); | |||
|         break; | |||
|     default: | |||
|         echo ("UNKNOWN STATE: " . ($callstate & 0x200) . "\r\n"); | |||
|         break; | |||
| } | |||
| ?> | |||
| </code> | |||
| ==More Sample Code== | |||
| Much more elaborate SOAP sample code is available for download from [http://wiki.innovaphone.com/index.php?title=Howto:Wiki_Sources#soapphp5 the samples section] on http://download.innovaphone.com. | |||
| Note that if you want to use [[Howto:Use_the_innovaphone_Demo_PBX | innovaphone's demo PBX ]] to test the code, you need to change the PBX IP address from 145.253.157.200 to <code>demo.innovaphone.com</code> in the downloaded code.  | |||
| This code shows how to deal with call control on master/slave scenarios as well as how to perform PBX configuration updates or PBX object creation using the [[Howto:Using the SOAP Admin Function|SOAP Admin function]].  Currently, there are the following sample functions available: | |||
| * create a SOAP session to the master PBX | |||
| * create a user session on a device | |||
| * initiate a call via SOAP and terminate it once connected | |||
| * List all available SOAP functions | |||
| * Show an object's configuration using the Admin() function | |||
| * clone of adminshow which forces config=1 | |||
| * Modify a user object using the Admin() function | |||
| * Clone a PBX object using the Admin() function | |||
| * Add call forwarding using Admin() function | |||
| * Delete call forwarding using Admin() function | |||
| * Get the PBX key | |||
| * Generate an encrypted PBX user password | |||
| * Set a new password for user | |||
| * Get the password of a PBX object in clear | |||
| * Decrypt a password from VARs (only works for boxes with default admin password!) | |||
| * Encrypt a password for use in VARs (only works for boxes with default admin password!) | |||
| * Find matching users | |||
| Although this code is written in PHP5, it is also good reading if you do not intend to use PHP as it exemplifies the principles of using SOAP and the SOAP Admin() function. | |||
| ==Download == | |||
| *[http://wiki.innovaphone.com/index.php?title=Howto:Wiki_Sources#soapphp5 Download] the complete file package of scripts and files described in this article. | |||
| ==Related Articles== | |||
| * [[Reference:SOAP API]] | |||
| * [[Reference7:SOAP API]] | |||
| * [[Reference8:SOAP API]] | |||
| * [[Reference9:Concept SOAP API]] | |||
| * [[Reference10:Concept SOAP API]] | |||
| * [[Howto:Use the innovaphone Demo PBX]] | |||
| * [[Howto:Using the SOAP Admin Function]] | |||
| [ | [http://wiki.innovaphone.com/index.php?search=soap+sample&go=Go Search all SOAP sample code in the wiki] | ||
| <!--*[[Main_Page|wiki-innovaphone]]--> | <!--*[[Main_Page|wiki-innovaphone]]--> | ||
| [[Category:Sample|{{PAGENAME}}]] | [[Category:Sample|{{PAGENAME}}]] | ||
Latest revision as of 14:42, 23 August 2023
Applies To
This information applies to
- V9 and later innovaphone PBX platform
- web server running PHP 5
innovaphone PBX web services can be used from PHP easily. Here is how
More Information
Configuration
You will need php with the SoapClient class available.
The innovaphone PBX wsdl file is required.  See Reference8:SOAP API for information how to obtain this file.  
To access older WSDL versions, you will need to change the WSDL URL setting (in const ___wsdl) and adapt to the required number of version booleans in the call to Initialize() in the pbx wrapper class.
If you have access to the internet, you can try out the sample code given below right away. Of course you may also adapt the sample script configuration and try with your own innovaphone PBX.
Running the Sample Code
To demonstrate SOAP usage with PHP, a simple script is presented that will allow you to initiate a call using a web page.
The following class innoPBX is used to wrap the (somewhat clumsy) PHP SOAP access mechanism:
<?php
// innovaphone PBX SOAP API PHP wrapper class
//
class innoPBX extends SOAPClient {
    protected $___key; 		// the session key
    protected $___session; 	// the session id
    protected $___options = array(
				// default SOAPClient::__construct options used by the class
	"connection_timeout" => 10,
	"exceptions" => true,
    );
    const ___wsdl = 'http://www.innovaphone.com/wsdl/pbx900.wsdl';
    // class constructor
    public function __construct(
	$server, 	// the PBX IP
	$httpu,		// the HTTP user id (e.g. "admin")
	$httpp,		// the HTTP password (e.g. "ip800")
	$user = null,		// the PBX user CN to work with 
	$options = null,
			// extra or overriding options for SOAPClient::__construct
	$wsdl = null    // the wsdl file location
    ) {	
	$wsdl = ($wsdl === null) ? self::___wsdl : $wsdl;
	$usedoptions = array(			// forced options
	    'login' => $httpu,
	    'password' => $httpp,
	    'location' => "http://$server/PBX0/user.soap",
			);
	if (is_array($options)) $usedoptions += $options;	
						// merge in user options
	$usedoptions += $this->___options;	// merged in class global options
	// construct parent class
	parent::__construct($wsdl, $usedoptions);
	// get the connection (using and activating v9 wsdl)
	$init = $this->Initialize($user, "PHP SOAP Wrapper", true, true, true, true, true);
	$this->___key = $init['key'];
	$this->___session = $init['return'];
    }
    public function key() { return $this->___key; }
    public function session() { return $this->___session; }
}
?>
Save this code into a file named innopbx.class.php on your web server. This class is of course not required to use PHP SOAP with the innovaphone PBX API. However, it comes handy and should be generic enough to be useful in most cases.
The SOAP client uses the following code:
<?php
// get the innoPBX wrapper class
require_once('innopbx.class.php');
// dummy classes to map SOAP results to (really would love to use namespaces here...)
// you can add methods and variables to these classes as needed
class innoUserInfo { };
class innoCallInfo { };
class innoAnyInfo { };
class innoGroup { };
class innoNo { };
class innoInfo { };
// config, adapt to your need
// this data below will work with innovaphones demo PBX.  Just register 2 phones as user-3 and user-4 and hve fun!
$server = "demo.innovaphone.com";
$user = "SOAP";
$httpu = "demo";
$httpp = "demo";
$caller = "PBX User Four";
$destination = "130";
print "<pre>";
// create connector to PBX
// class mapping is optional
$inno = new innoPBX($server, $httpu, $httpp, $user,
	    array('classmap' => array("UserInfo" => "innoUserInfo", 
				"CallInfo" => "innoCallInfo",
				"AnyInfo" => "innoAnyInfo",
				"Group" => "innoGroup",
				"No" => "innoNo",
				"Info" => "innoInfo",
			    )));
if ($inno->key() == 0) die("failed to login to PBX");
// show all methods supported by this object
print "All the functions provided by class innoPBX:\n";
print_r($inno->__getFunctions());
print "\n\n";
// get version info
$v = $inno->Version();
// show both types of infos list
function showInfos(&$poll, $head, $cn = "", $user = "", $call = "") {
    print $head . "\n";
    if ($cn !== null) {
	print count($poll->user) . " UserInfos\n";
	foreach($poll->user as $ui) {
	    if (($cn === "") || ($cn == $ui->cn)) {
		print "     {$ui->cn} ({$ui->h323} #{$ui->e164}) state {$ui->state}\n";
	    }
	}
    }
    if ($call !== null) {
	print count($poll->call) . " CallInfos\n";
	foreach($poll->call as $ci) {
	    if ((($user === "") || ($user == $ci->user)) &&
		(($call === "") || ($call == $ci->call))) {
		    print "    {$ci->user}/{$ci->call} {$ci->No[1]->h323} #{$ci->No[1]->e164} (remote {$ci->No[0]->h323} #{$ci->No[0]->e164}) msg {$ci->msg}\n";
		}
	}
    }
}
// retrieve the full user list.  Foreach object in the PBX, one userinfo is posted, terminated by an empty one
// You cannot assume that you will receive this list within a certain number of Poll results, so please iterate
print "Retrieving User list for "; foreach ($v as $name => $value) print "\n  $name=$value "; print "...\n\n";
$seen = false;
$i = 1;
while (!$seen) {
    $p = $inno->Poll($inno->session());
    showInfos($p, "Poll() result #$i", "", null, null); $i++;
    if ($p->user[count($p->user)-1]->cn == "") {
	// we have seen all entries
	print " --- END OF LIST ---\n\n";
	$seen = true;
	break;
    }
}
// create a call on behalf of some user
// first we need a user handle
print "\nObtaining a user handle for $caller...";
$uhandle = $inno->UserInitialize($inno->session(), $caller);
print " $uhandle\n";
if ($uhandle == 0) {
    die("cant get user handle for $caller");
}
// then we create a call
print "Creating call from $caller to $destination...";
$call = $inno->UserCall($uhandle, null, $destination, null, 0, array());
print " $call\n";
if ($call == 0) {
    die("cant call on behalf of user $caller ($uhandle)");
}
// get call state(s) until call disappears
// note: you must at least poll for one call info related to this call, if you end
// the session right after UserCall, the call will not succeed!
$done = false;
while (!$done) {
    ob_flush();   // this is just to see the progress on the call states
    $p = $inno->Poll($inno->session());
    showInfos($p, "Call States #$i for #{$call}", $caller, $uhandle, $call); $i++;
    // see if there was a del event (there is ALWAYS a del event)
    // we could as well look for an event with active = false
    foreach ($p->call as $ci) {
	if (($ci->user == $uhandle) && 
	    ($ci->call == $call) &&
	    ($ci->msg == "del")) {
		print "\nCall terminated!\n";
		$done = true;
	    }
    }
}
// terminate the session
$e = $inno->End($inno->session());
?>
Save this into a file named innopbx.sample.php in the same directory of your web server. To run the code, you need to examine and possibly adapt the configuration parameters in innopbx.sample.php:
// config, adapt to your need
// this data below will work with innovaphones demo PBX.  Just register 2 phones as user-3 and user-4 and hve fun!
$server = "demo.innovaphone.com";
$user = "SOAP";
$httpu = "demo";
$httpp = "demo";
$caller = "PBX User Four";
$destination = "130";
This configuration will work right away with our publicly available demo PBX. You simply need to register 2 phones with the user-3 and user-4 users.
Of course you can also use your own PBX. Set the configuration parameters as follow:
| $server | IP address of your innovaphone PBX | 
| $user | CN (aka Long name) of a PBX user that is active group member of a group where all others are (active or passive) members of | 
| $httpu | the admin account of the box running the innovaphone PBX (e.g.admin) | 
| $httpp | the admin password of this box (e.g. ipxxx) | 
| $caller | CN (aka Long name) of a PBX user you will have the script make a call for | 
| $destination | the number to be called | 
When the configuration parameters are set, you can point your browser to http://your-web-server-ip/path-to-folder/innopbx.sample.php which should create a call on $callers phone.
Please note that you need internet access to retrieve the wsdl from innovaphone's web site.
How the code works
The innovaphone PBX API wrapper class
This class is derived from (extends in PHP speak) the standard SOAPClient class. It basically simplifies the use of the standard class and performs the required call to Initialize within its constructor.
The constructor takes the following arguments:
public function __construct($server, $httpu, $httpp, $user, $options = null, $wsdl = null);
| $server, $httpu, $httpp, $user | see table above | 
| $options | an optional array of further options to SOAPClient::__construct. Normally not required. See the PHP SOAPClient class documentation for details. | 
| $wsdl | the optional URI of the pbx wsdl file. Please note that you should use pbx10_00.wsdl instead of pbx.wsdl, as this is the more recent version of the wsdl. By default, the wsdl is received from the innovaphone web and (subject to PHP configuration options) cached in PHP's wsdl cache. If you intend to supply it locally (which is recommended, as you otherwise depend on www.innovaphone.com being available each time your program runs), use a file:// URI. | 
Calls to PBX API functions defined in the wsdl can be done directly through an instance of class innoPBX. These member functions - which are neither defined in class innoPBX nor in class SOAPClient - are created automagically by reading the wsdl:
$v = $inno->Version();
To see the functions available and their PHP signatures, you may call
$inno = new innoPBX(...);
print_r($inno->$inno->__getFunctions());
Many PBX SOAP API functions required either the session-key or session-handle. These are available from the class instance for convenience:
$p = $inno->Poll($inno->session());
How to use the innovaphone PBX API wrapper class with SSL
In order to make a secure connection to the PBX, you have to change the inno.class.php as follows:
'location' => "https://$server/PBX0/user.soap"
Note: in PHP 5.6 all encrypted client streams now enable peer verification by default. By default, this will use OpenSSL's default CA bundle to verify the peer certificate. See: OpenSSL changes in PHP 5.6.x
To disable peer verification for the soap connection, you have to pass this as an option to the constructor:
$context = stream_context_create(array(
 'ssl' => array(
  'verify_peer' => false,
  'verify_peer_name' => false,
  'allow_self_signed' => false
 )
));
$options = array('stream_context' => $context); 
$inno = new innoPBX($server, $httpu, $httpp, $user, $options, $wsdl);
More information on stream_context_create and ssl options: stream_context_create and SSL context options
How PHP maps SOAP parameters to method parameters and results
The mapping of parameters is pretty straight forward. Arguments passed in to a SOAP method are passed as a value parameters to the class methods. The thing to understand is that return values are always passed as return values, never as by-reference method arguments.
If a SOAP method returns a single item, then it is returned by the function directly with appropriate type:
int UserInitialize(int $session, string $user);
The single return value from the UserInitialize SOAP method is returned as int by the UserInitialize innoPBX-method.
If the SOAP methods returns multiple items, then the corresponding class method will return an array:
array Initialize(string $user, string $appl);
will return an array such as
array(
    ['return'] => value
    ['key'] => value
)
In the method signature printed out by __getFunctions() this looks like
list(int $return, int $key) Initialize(string $user, string $appl)
The sample class user
This section will discuss the sample code. It is not a decent description of the PBX SOAP API. For this, see the related articles.
A new innoPBX class instance object $inno is created using the values from the configuration variables discussed above.
$inno = new innoPBX($server, $httpu, $httpp, $user,
	    array('classmap' => array("UserInfo" => "innoUserInfo", 
				"CallInfo" => "innoCallInfo",
				"AnyInfo" => "innoAnyInfo",
				"Group" => "innoGroup",
				"No" => "innoNo",
				"Info" => "innoInfo",
			    )));
The option fifth argument will add a classmap entry to the options passed to SOAPClient::__construct. If this is done, the SOAP class will use your defined classes when returning values to the caller. Otherwise, structured return values will be indexed arrays (using the element names as array indexes). This is neat, but not required for the class to function.
When Initialize (which is done implicitly in the innoPBX constructor) was successfull, one UserInfo class object per PBX object is posted to the caller. This list is terminated by an empty entry (more precisely, an entry with an empty cn member).
The
while (!$seen) {
    $p = $inno->Poll($inno->session());
    ...
loop will retrieve and show all these entries.
To be able to perform calls, a user session must be opened by
$uhandle = $inno->UserInitialize($inno->session(), $caller);
The $uhandle retrieved will identify the user to be worked with on subsequent calls.
A call on behalf of $caller is then initiated:
$call = $inno->UserCall($uhandle, null, $destination, null, 0, array());
When UserCall returns a non-zero call handle, the call is initiated and can be monitored by examining CallInfo events returned by subsequent Poll() calls:
while (!$done) {
    $p = $inno->Poll($inno->session());
this will retrieve events from the PBX
	if (($ci->user == $uhandle) && 
	    ($ci->call == $call) &&
	    ($ci->msg == "del")) {
		print "\nCall terminated!\n";
		$done = true;
until a del event is seen for the call previously created by UserCall. Please note that any number of call events may occur, e.g. because someone else called $caller. You cannot assume that all events retrieved are related to the call you created. Also, any number of UserInfo events may be retrieved.
Please note that there is no need to consume all of the events related to the call for it to work. However, you must not terminate the session before the call is actually proceeding. If you do so, strange things will happen. To be sure the call proceeds, it is ok to wait for just any CallInfo event related to the call. You can then safely terminate your session and exit the web page.
Working with Call States from CallInfo
The state field of the CallInfo contains a bit field encoded information about direction of the call, hold information and an actual call state.
Here is an example how to get relevant states:
<?php
// decimal state value from CallInfo
// $callstate = 2;
// call state?
switch ($callstate & 0xF) {
    case 1:
        echo ("setup\r\n");
        break;
    case 2:
        echo ("setup-ack\r\n");
        break;
    case 3:
        echo ("call-proc\r\n");
        break;
    case 4:
        echo ("Alert\r\n");
        break;
    case 5:
        echo ("Connect\r\n");
        break;
    case 6:
        echo ("disconnect sent\r\n");
        break;
    case 7:
        echo ("disconnect received\r\n");
        break;
    case 8:
        echo ("parked\r\n");
        break;
    default:
        echo ("UNKNOWN STATE: " . ($callstate & 0xF) . "\r\n");
        break;
}
// call direction?
switch ($callstate & 0x80) {
    case 0:
        echo ("inbound call\r\n");
        break;
    case 128:
        echo ("outbound call\r\n");
        break;
    default:
        echo ("UNKNOWN STATE: " . ($callstate & 0x80) . "\r\n");
        break;
}
// is call on hold?
switch ($callstate & 0x100) {
    case 0:
        echo ("active call\r\n");
        break;
    case 256:
        echo ("call on hold\r\n");
        break;
    default:
        echo ("UNKNOWN STATE: " . ($callstate & 0x100) . "\r\n");
        break;
}
// is call being held by peer?
switch ($callstate & 0x200) {
    case 0:
        echo ("active call\r\n");
        break;
    case 512:
        echo ("active call put on hold by peer\r\n");
        break;
    default:
        echo ("UNKNOWN STATE: " . ($callstate & 0x200) . "\r\n");
        break;
}
?>
More Sample Code
Much more elaborate SOAP sample code is available for download from the samples section on http://download.innovaphone.com.
Note that if you want to use  innovaphone's demo PBX  to test the code, you need to change the PBX IP address from 145.253.157.200 to demo.innovaphone.com in the downloaded code. 
This code shows how to deal with call control on master/slave scenarios as well as how to perform PBX configuration updates or PBX object creation using the SOAP Admin function. Currently, there are the following sample functions available:
- create a SOAP session to the master PBX
- create a user session on a device
- initiate a call via SOAP and terminate it once connected
- List all available SOAP functions
- Show an object's configuration using the Admin() function
- clone of adminshow which forces config=1
- Modify a user object using the Admin() function
- Clone a PBX object using the Admin() function
- Add call forwarding using Admin() function
- Delete call forwarding using Admin() function
- Get the PBX key
- Generate an encrypted PBX user password
- Set a new password for user
- Get the password of a PBX object in clear
- Decrypt a password from VARs (only works for boxes with default admin password!)
- Encrypt a password for use in VARs (only works for boxes with default admin password!)
- Find matching users
Although this code is written in PHP5, it is also good reading if you do not intend to use PHP as it exemplifies the principles of using SOAP and the SOAP Admin() function.
Download
- Download the complete file package of scripts and files described in this article.
