2011-09-12 23:53:38 +00:00
|
|
|
<?php
|
|
|
|
|
|
|
|
/*
|
2011-09-22 23:16:46 +00:00
|
|
|
* PHP server library for EnvayaSMS
|
2011-09-12 23:53:38 +00:00
|
|
|
*
|
|
|
|
* For example usage see example/www/index.php
|
|
|
|
*/
|
|
|
|
|
2011-09-22 23:16:46 +00:00
|
|
|
class EnvayaSMS
|
2011-09-12 23:53:38 +00:00
|
|
|
{
|
|
|
|
const ACTION_INCOMING = 'incoming';
|
|
|
|
const ACTION_OUTGOING = 'outgoing';
|
|
|
|
const ACTION_SEND_STATUS = 'send_status';
|
2011-10-10 23:19:38 +00:00
|
|
|
const ACTION_DEVICE_STATUS = 'device_status';
|
2011-09-27 18:30:49 +00:00
|
|
|
const ACTION_TEST = 'test';
|
2011-09-12 23:53:38 +00:00
|
|
|
|
2011-09-13 00:37:37 +00:00
|
|
|
const STATUS_QUEUED = 'queued';
|
|
|
|
const STATUS_FAILED = 'failed';
|
|
|
|
const STATUS_SENT = 'sent';
|
2012-03-16 00:20:23 +00:00
|
|
|
const STATUS_CANCELLED = 'cancelled';
|
2011-09-12 23:53:38 +00:00
|
|
|
|
2011-10-10 23:19:38 +00:00
|
|
|
const DEVICE_STATUS_POWER_CONNECTED = "power_connected";
|
|
|
|
const DEVICE_STATUS_POWER_DISCONNECTED = "power_disconnected";
|
|
|
|
const DEVICE_STATUS_BATTERY_LOW = "battery_low";
|
|
|
|
const DEVICE_STATUS_BATTERY_OKAY = "battery_okay";
|
2012-03-16 00:20:23 +00:00
|
|
|
const DEVICE_STATUS_SEND_LIMIT_EXCEEDED = "send_limit_exceeded";
|
2011-10-10 23:19:38 +00:00
|
|
|
|
2011-09-20 01:51:45 +00:00
|
|
|
const MESSAGE_TYPE_SMS = 'sms';
|
2011-09-22 23:16:46 +00:00
|
|
|
const MESSAGE_TYPE_MMS = 'mms';
|
2012-03-16 00:20:23 +00:00
|
|
|
const MESSAGE_TYPE_CALL = 'call';
|
|
|
|
|
|
|
|
const NETWORK_MOBILE = "MOBILE";
|
|
|
|
const NETWORK_WIFI = "WIFI";
|
2011-09-12 23:53:38 +00:00
|
|
|
|
|
|
|
static function escape($val)
|
|
|
|
{
|
2011-10-09 23:31:06 +00:00
|
|
|
return htmlspecialchars($val, ENT_COMPAT, 'UTF-8');
|
2011-09-22 23:16:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private static $request;
|
2011-09-12 23:53:38 +00:00
|
|
|
|
2011-09-22 23:16:46 +00:00
|
|
|
static function get_request()
|
|
|
|
{
|
|
|
|
if (!isset(static::$request))
|
|
|
|
{
|
|
|
|
$version = @$_POST['version'];
|
|
|
|
|
|
|
|
// If API version changes, could return
|
|
|
|
// different EnvayaSMS_Request instance
|
|
|
|
// to support multiple phone versions
|
|
|
|
|
|
|
|
static::$request = new EnvayaSMS_Request();
|
|
|
|
}
|
|
|
|
return static::$request;
|
2012-03-16 00:20:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static function get_error_xml($message)
|
|
|
|
{
|
|
|
|
ob_start();
|
|
|
|
echo "<?xml version='1.0' encoding='UTF-8'?>\n";
|
|
|
|
echo "<response>";
|
|
|
|
echo "<error>";
|
|
|
|
echo EnvayaSMS::escape($message);
|
|
|
|
echo "</error>";
|
|
|
|
echo "</response>";
|
|
|
|
return ob_get_clean();
|
|
|
|
}
|
|
|
|
|
|
|
|
static function get_success_xml()
|
|
|
|
{
|
|
|
|
ob_start();
|
|
|
|
echo "<?xml version='1.0' encoding='UTF-8'?>\n";
|
|
|
|
echo "<response></response>";
|
|
|
|
return ob_get_clean();
|
|
|
|
}
|
2011-09-22 23:16:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
class EnvayaSMS_Request
|
|
|
|
{
|
2011-09-13 00:37:37 +00:00
|
|
|
private $request_action;
|
|
|
|
|
2011-09-27 18:30:49 +00:00
|
|
|
public $version;
|
|
|
|
public $phone_number;
|
2011-10-10 23:19:38 +00:00
|
|
|
public $log;
|
2012-03-16 00:20:23 +00:00
|
|
|
|
|
|
|
public $version_name;
|
|
|
|
public $sdk_int;
|
|
|
|
public $manufacturer;
|
|
|
|
public $model;
|
|
|
|
public $network;
|
2011-09-27 18:30:49 +00:00
|
|
|
|
|
|
|
function __construct()
|
|
|
|
{
|
|
|
|
$this->version = $_POST['version'];
|
|
|
|
$this->phone_number = $_POST['phone_number'];
|
2012-03-16 00:20:23 +00:00
|
|
|
$this->log = $_POST['log'];
|
|
|
|
$this->network = @$_POST['network'];
|
|
|
|
|
|
|
|
if (preg_match('#/(?P<version_name>[\w\.\-]+) \(Android; SDK (?P<sdk_int>\d+); (?P<manufacturer>[^;]*); (?P<model>[^\)]*)\)#',
|
|
|
|
@$_SERVER['HTTP_USER_AGENT'], $matches))
|
|
|
|
{
|
|
|
|
$this->version_name = $matches['version_name'];
|
|
|
|
$this->sdk_int = $matches['sdk_int'];
|
|
|
|
$this->manufacturer = $matches['manufacturer'];
|
|
|
|
$this->model = $matches['model'];
|
|
|
|
}
|
2011-09-27 18:30:49 +00:00
|
|
|
}
|
|
|
|
|
2011-09-22 23:16:46 +00:00
|
|
|
function get_action()
|
2011-09-13 00:37:37 +00:00
|
|
|
{
|
|
|
|
if (!$this->request_action)
|
|
|
|
{
|
2011-09-22 23:16:46 +00:00
|
|
|
$this->request_action = $this->_get_action();
|
2011-09-13 00:37:37 +00:00
|
|
|
}
|
|
|
|
return $this->request_action;
|
|
|
|
}
|
|
|
|
|
2011-09-22 23:16:46 +00:00
|
|
|
private function _get_action()
|
2011-09-12 23:53:38 +00:00
|
|
|
{
|
|
|
|
switch (@$_POST['action'])
|
|
|
|
{
|
2011-09-22 23:16:46 +00:00
|
|
|
case EnvayaSMS::ACTION_INCOMING:
|
|
|
|
return new EnvayaSMS_Action_Incoming($this);
|
|
|
|
case EnvayaSMS::ACTION_OUTGOING:
|
|
|
|
return new EnvayaSMS_Action_Outgoing($this);
|
|
|
|
case EnvayaSMS::ACTION_SEND_STATUS:
|
|
|
|
return new EnvayaSMS_Action_SendStatus($this);
|
2011-09-27 18:30:49 +00:00
|
|
|
case EnvayaSMS::ACTION_TEST:
|
|
|
|
return new EnvayaSMS_Action_Test($this);
|
2011-10-10 23:19:38 +00:00
|
|
|
case EnvayaSMS::ACTION_DEVICE_STATUS:
|
|
|
|
return new EnvayaSMS_Action_DeviceStatus($this);
|
2011-09-12 23:53:38 +00:00
|
|
|
default:
|
2011-09-22 23:16:46 +00:00
|
|
|
return new EnvayaSMS_Action($this);
|
2011-09-12 23:53:38 +00:00
|
|
|
}
|
2011-09-27 18:30:49 +00:00
|
|
|
}
|
2011-09-12 23:53:38 +00:00
|
|
|
|
2011-09-22 23:16:46 +00:00
|
|
|
function is_validated($correct_password)
|
2011-09-12 23:53:38 +00:00
|
|
|
{
|
2011-09-22 23:16:46 +00:00
|
|
|
$signature = @$_SERVER['HTTP_X_REQUEST_SIGNATURE'];
|
2011-09-12 23:53:38 +00:00
|
|
|
if (!$signature)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
$is_secure = (!empty($_SERVER['HTTPS']) AND filter_var($_SERVER['HTTPS'], FILTER_VALIDATE_BOOLEAN));
|
|
|
|
$protocol = $is_secure ? 'https' : 'http';
|
|
|
|
$full_url = $protocol . "://" . $_SERVER['HTTP_HOST'] . $_SERVER['REQUEST_URI'];
|
|
|
|
|
|
|
|
$correct_signature = $this->compute_signature($full_url, $_POST, $correct_password);
|
|
|
|
|
|
|
|
//error_log("Correct signature: '$correct_signature'");
|
|
|
|
|
|
|
|
return $signature === $correct_signature;
|
|
|
|
}
|
|
|
|
|
|
|
|
function compute_signature($url, $data, $password)
|
|
|
|
{
|
|
|
|
ksort($data);
|
|
|
|
|
|
|
|
$input = $url;
|
|
|
|
foreach($data as $key => $value)
|
|
|
|
$input .= ",$key=$value";
|
|
|
|
|
|
|
|
$input .= ",$password";
|
|
|
|
|
|
|
|
//error_log("Signed data: '$input'");
|
|
|
|
|
|
|
|
return base64_encode(sha1($input, true));
|
2012-03-16 00:20:23 +00:00
|
|
|
}
|
2011-09-27 18:30:49 +00:00
|
|
|
|
|
|
|
static function get_messages_xml($messages)
|
|
|
|
{
|
|
|
|
ob_start();
|
|
|
|
echo "<?xml version='1.0' encoding='UTF-8'?>\n";
|
2012-03-16 00:20:23 +00:00
|
|
|
echo "<response>";
|
2011-09-27 18:30:49 +00:00
|
|
|
echo "<messages>";
|
|
|
|
foreach ($messages as $message)
|
2012-03-16 00:20:23 +00:00
|
|
|
{
|
|
|
|
$type = isset($message->type) ? $message->type : EnvayaSMS::MESSAGE_TYPE_SMS;
|
2011-09-28 03:23:36 +00:00
|
|
|
$id = isset($message->id) ? " id=\"".EnvayaSMS::escape($message->id)."\"" : "";
|
|
|
|
$to = isset($message->to) ? " to=\"".EnvayaSMS::escape($message->to)."\"" : "";
|
2011-09-28 21:46:16 +00:00
|
|
|
$priority = isset($message->priority) ? " priority=\"".$message->priority."\"" : "";
|
2012-03-16 00:20:23 +00:00
|
|
|
echo "<$type$id$to$priority>".EnvayaSMS::escape($message->message)."</$type>";
|
2011-09-27 18:30:49 +00:00
|
|
|
}
|
|
|
|
echo "</messages>";
|
2012-03-16 00:20:23 +00:00
|
|
|
echo "</response>";
|
2011-09-27 18:30:49 +00:00
|
|
|
return ob_get_clean();
|
|
|
|
}
|
2011-09-12 23:53:38 +00:00
|
|
|
}
|
|
|
|
|
2011-09-22 23:16:46 +00:00
|
|
|
class EnvayaSMS_OutgoingMessage
|
2011-09-12 23:53:38 +00:00
|
|
|
{
|
2011-09-27 18:30:49 +00:00
|
|
|
public $id; // ID generated by server
|
2011-09-20 01:51:45 +00:00
|
|
|
public $to; // destination phone number
|
|
|
|
public $message; // content of SMS message
|
2011-09-28 21:46:16 +00:00
|
|
|
public $priority; // integer priority, higher numbers will be sent first
|
2012-03-16 00:20:23 +00:00
|
|
|
public $type; // EnvayaSMS::MESSAGE_TYPE_* value (default sms)
|
2011-09-12 23:53:38 +00:00
|
|
|
}
|
|
|
|
|
2011-09-22 23:16:46 +00:00
|
|
|
class EnvayaSMS_Action
|
2011-09-12 23:53:38 +00:00
|
|
|
{
|
|
|
|
public $type;
|
2011-09-22 23:16:46 +00:00
|
|
|
public $request;
|
2011-09-12 23:53:38 +00:00
|
|
|
|
2011-09-22 23:16:46 +00:00
|
|
|
function __construct($request)
|
2011-09-12 23:53:38 +00:00
|
|
|
{
|
2011-09-22 23:16:46 +00:00
|
|
|
$this->request = $request;
|
2011-09-12 23:53:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-22 23:16:46 +00:00
|
|
|
class EnvayaSMS_MMS_Part
|
2011-09-20 01:51:45 +00:00
|
|
|
{
|
|
|
|
public $form_name; // name of form field with MMS part content
|
|
|
|
public $cid; // MMS Content-ID
|
|
|
|
public $type; // Content type
|
|
|
|
public $filename; // Original filename of MMS part on sender phone
|
|
|
|
public $tmp_name; // Temporary file where MMS part content is stored
|
|
|
|
public $size; // Content length
|
|
|
|
public $error; // see http://www.php.net/manual/en/features.file-upload.errors.php
|
|
|
|
|
|
|
|
function __construct($args)
|
|
|
|
{
|
|
|
|
$this->form_name = $args['name'];
|
|
|
|
$this->cid = $args['cid'];
|
|
|
|
$this->type = $args['type'];
|
|
|
|
$this->filename = $args['filename'];
|
|
|
|
|
|
|
|
$file = $_FILES[$this->form_name];
|
|
|
|
|
|
|
|
$this->tmp_name = $file['tmp_name'];
|
|
|
|
$this->size = $file['size'];
|
|
|
|
$this->error = $file['error'];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-22 23:16:46 +00:00
|
|
|
class EnvayaSMS_Action_Incoming extends EnvayaSMS_Action
|
2011-09-12 23:53:38 +00:00
|
|
|
{
|
2011-09-20 01:51:45 +00:00
|
|
|
public $from; // Sender phone number
|
|
|
|
public $message; // The message body of the SMS, or the content of the text/plain part of the MMS.
|
2011-09-22 23:16:46 +00:00
|
|
|
public $message_type; // EnvayaSMS::MESSAGE_TYPE_MMS or EnvayaSMS::MESSAGE_TYPE_SMS
|
|
|
|
public $mms_parts; // array of EnvayaSMS_MMS_Part instances
|
2011-09-29 23:02:37 +00:00
|
|
|
public $timestamp; // timestamp of incoming message (added in version 12)
|
2012-03-16 00:20:23 +00:00
|
|
|
public $age; // delay in ms between time when message originally received and when forwarded to server (added in version 18)
|
2011-09-12 23:53:38 +00:00
|
|
|
|
2011-09-22 23:16:46 +00:00
|
|
|
function __construct($request)
|
2011-09-12 23:53:38 +00:00
|
|
|
{
|
2011-09-22 23:16:46 +00:00
|
|
|
parent::__construct($request);
|
|
|
|
$this->type = EnvayaSMS::ACTION_INCOMING;
|
2011-09-12 23:53:38 +00:00
|
|
|
$this->from = $_POST['from'];
|
2012-03-16 00:20:23 +00:00
|
|
|
$this->message = @$_POST['message'];
|
2011-09-20 01:51:45 +00:00
|
|
|
$this->message_type = $_POST['message_type'];
|
2011-09-29 23:02:37 +00:00
|
|
|
$this->timestamp = @$_POST['timestamp'];
|
2012-03-16 00:20:23 +00:00
|
|
|
$this->age = @$_POST['age'];
|
2011-09-20 01:51:45 +00:00
|
|
|
|
2011-09-22 23:16:46 +00:00
|
|
|
if ($this->message_type == EnvayaSMS::MESSAGE_TYPE_MMS)
|
2011-09-20 01:51:45 +00:00
|
|
|
{
|
|
|
|
$this->mms_parts = array();
|
|
|
|
foreach (json_decode($_POST['mms_parts'], true) as $mms_part)
|
|
|
|
{
|
2011-09-22 23:16:46 +00:00
|
|
|
$this->mms_parts[] = new EnvayaSMS_MMS_Part($mms_part);
|
2011-09-20 01:51:45 +00:00
|
|
|
}
|
|
|
|
}
|
2011-09-27 18:30:49 +00:00
|
|
|
}
|
2011-09-12 23:53:38 +00:00
|
|
|
|
|
|
|
function get_response_xml($messages)
|
|
|
|
{
|
2011-09-27 18:30:49 +00:00
|
|
|
return $this->request->get_messages_xml($messages);
|
|
|
|
}
|
2011-09-12 23:53:38 +00:00
|
|
|
}
|
|
|
|
|
2011-09-22 23:16:46 +00:00
|
|
|
class EnvayaSMS_Action_Outgoing extends EnvayaSMS_Action
|
2011-09-12 23:53:38 +00:00
|
|
|
{
|
2011-09-22 23:16:46 +00:00
|
|
|
function __construct($request)
|
2011-09-12 23:53:38 +00:00
|
|
|
{
|
2011-09-22 23:16:46 +00:00
|
|
|
parent::__construct($request);
|
|
|
|
$this->type = EnvayaSMS::ACTION_OUTGOING;
|
2011-09-27 18:30:49 +00:00
|
|
|
}
|
2011-09-12 23:53:38 +00:00
|
|
|
|
|
|
|
function get_response_xml($messages)
|
|
|
|
{
|
2011-09-27 18:30:49 +00:00
|
|
|
return $this->request->get_messages_xml($messages);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class EnvayaSMS_Action_Test extends EnvayaSMS_Action
|
|
|
|
{
|
|
|
|
function __construct($request)
|
|
|
|
{
|
|
|
|
parent::__construct($request);
|
|
|
|
$this->type = EnvayaSMS::ACTION_TEST;
|
2011-09-12 23:53:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-22 23:16:46 +00:00
|
|
|
class EnvayaSMS_Action_SendStatus extends EnvayaSMS_Action
|
2011-09-12 23:53:38 +00:00
|
|
|
{
|
2011-09-22 23:16:46 +00:00
|
|
|
public $status; // EnvayaSMS::STATUS_* values
|
|
|
|
public $id; // server ID previously used in EnvayaSMS_OutgoingMessage
|
2012-03-16 00:20:23 +00:00
|
|
|
public $error; // textual description of error (if applicable)
|
2011-09-20 01:51:45 +00:00
|
|
|
|
2011-09-22 23:16:46 +00:00
|
|
|
function __construct($request)
|
2011-09-12 23:53:38 +00:00
|
|
|
{
|
2011-09-22 23:16:46 +00:00
|
|
|
parent::__construct($request);
|
|
|
|
$this->type = EnvayaSMS::ACTION_SEND_STATUS;
|
2011-09-13 00:37:37 +00:00
|
|
|
$this->status = $_POST['status'];
|
2011-09-12 23:53:38 +00:00
|
|
|
$this->id = $_POST['id'];
|
2011-09-24 06:00:51 +00:00
|
|
|
$this->error = $_POST['error'];
|
2011-09-12 23:53:38 +00:00
|
|
|
}
|
2011-10-10 23:19:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
class EnvayaSMS_Action_DeviceStatus extends EnvayaSMS_Action
|
|
|
|
{
|
|
|
|
public $status; // EnvayaSMS::DEVICE_STATUS_* values
|
|
|
|
|
|
|
|
function __construct($request)
|
|
|
|
{
|
|
|
|
parent::__construct($request);
|
|
|
|
$this->type = EnvayaSMS::ACTION_DEVICE_STATUS;
|
|
|
|
$this->status = $_POST['status'];
|
|
|
|
}
|
|
|
|
}
|