2020-01-17 01:13:40 +01:00
< ? php
/*
* This file is part of RaspiSMS .
*
* ( c ) Pierre - Lin Bonnemaison < plebwebsas @ gmail . com >
*
* This source file is subject to the GPL - 3.0 license that is bundled
* with this source code in the file LICENSE .
*/
namespace controllers\publics ;
/**
2020-01-17 18:19:25 +01:00
* Api to interact with raspisms .
2020-01-17 01:13:40 +01:00
*/
class Api extends \descartes\ApiController
{
2020-01-17 18:19:25 +01:00
const DEFAULT_RETURN = [
2020-01-17 01:13:40 +01:00
'error' => 0 , //Error code
'message' => null , //Any message to describe a potential error
'response' => null , //The content of the response
'next' => null , //Link to the next results
'prev' => null , //Link to the previous results
];
2020-01-17 18:19:25 +01:00
const ERROR_CODES = [
2020-01-17 01:13:40 +01:00
'NONE' => 0 ,
'INVALID_CREDENTIALS' => 1 ,
'INVALID_PARAMETER' => 2 ,
2020-01-17 16:35:13 +01:00
'MISSING_PARAMETER' => 4 ,
'CANNOT_CREATE' => 8 ,
2020-03-30 01:52:53 +02:00
'SUSPENDED_USER' => 16 ,
2020-06-17 03:02:33 +02:00
'CANNOT_DELETE' => 32 ,
2020-01-17 01:13:40 +01:00
];
2020-01-17 18:19:25 +01:00
const ERROR_MESSAGES = [
2021-01-14 03:25:58 +01:00
'INVALID_CREDENTIALS' => 'Invalid API Key. Please provide a valid API key as GET or POST parameter "api_key" or a HTTP "X-Api-Key".' ,
2020-01-17 01:13:40 +01:00
'INVALID_PARAMETER' => 'You have specified an invalid parameter : ' ,
2020-01-17 16:35:13 +01:00
'MISSING_PARAMETER' => 'One require parameter is missing : ' ,
'CANNOT_CREATE' => 'Cannot create a new entry.' ,
2020-03-30 01:52:53 +02:00
'SUSPENDED_USER' => 'This user account is currently suspended.' ,
2020-06-17 03:02:33 +02:00
'CANNOT_DELETE' => 'Cannot delete this entry.' ,
2020-01-17 01:13:40 +01:00
];
private $internal_user ;
private $internal_phone ;
private $internal_received ;
private $internal_sended ;
2020-01-17 18:36:53 +01:00
private $internal_scheduled ;
2020-01-17 01:13:40 +01:00
private $internal_contact ;
private $internal_group ;
2020-01-17 18:36:53 +01:00
private $internal_conditional_group ;
2021-01-14 03:25:58 +01:00
private $internal_adapter ;
2020-01-17 01:13:40 +01:00
private $user ;
/**
2020-01-17 18:19:25 +01:00
* Construct the object and quit if failed authentication .
*
2020-01-17 01:13:40 +01:00
* @ return void ;
*/
public function __construct ()
{
parent :: __construct ();
2020-01-17 18:19:25 +01:00
2020-01-17 01:13:40 +01:00
$bdd = \descartes\Model :: _connect ( DATABASE_HOST , DATABASE_NAME , DATABASE_USER , DATABASE_PASSWORD );
$this -> internal_user = new \controllers\internals\User ( $bdd );
$this -> internal_phone = new \controllers\internals\Phone ( $bdd );
$this -> internal_received = new \controllers\internals\Received ( $bdd );
$this -> internal_sended = new \controllers\internals\Sended ( $bdd );
$this -> internal_scheduled = new \controllers\internals\Scheduled ( $bdd );
$this -> internal_contact = new \controllers\internals\Contact ( $bdd );
$this -> internal_group = new \controllers\internals\Group ( $bdd );
$this -> internal_conditional_group = new \controllers\internals\ConditionalGroup ( $bdd );
2021-01-14 03:25:58 +01:00
$this -> internal_adapter = new \controllers\internals\Adapter ();
2020-01-17 01:13:40 +01:00
//If no user, quit with error
$this -> user = false ;
2020-06-17 03:02:33 +02:00
$api_key = $_GET [ 'api_key' ] ? ? $_POST [ 'api_key' ] ? ? $_SERVER [ 'HTTP_X_API_KEY' ] ? ? false ;
2020-01-17 01:13:40 +01:00
if ( $api_key )
{
2020-01-17 18:19:25 +01:00
$this -> user = $this -> internal_user -> get_by_api_key ( $api_key );
2020-01-17 01:13:40 +01:00
}
if ( ! $this -> user )
{
$return = self :: DEFAULT_RETURN ;
$return [ 'error' ] = self :: ERROR_CODES [ 'INVALID_CREDENTIALS' ];
$return [ 'message' ] = self :: ERROR_MESSAGES [ 'INVALID_CREDENTIALS' ];
2021-01-16 23:49:40 +01:00
$this -> set_http_code ( 401 );
2020-01-17 01:13:40 +01:00
$this -> json ( $return );
exit ( self :: ERROR_CODES [ 'INVALID_CREDENTIALS' ]);
}
2020-06-23 21:06:13 +02:00
if ( \models\User :: STATUS_ACTIVE !== $this -> user [ 'status' ])
2020-03-30 01:52:53 +02:00
{
$return = self :: DEFAULT_RETURN ;
$return [ 'error' ] = self :: ERROR_CODES [ 'SUSPENDED_USER' ];
$return [ 'message' ] = self :: ERROR_MESSAGES [ 'SUSPENDED_USER' ];
2021-01-16 23:49:40 +01:00
$this -> set_http_code ( 403 );
2020-03-30 01:52:53 +02:00
$this -> json ( $return );
exit ( self :: ERROR_CODES [ 'SUSPENDED_USER' ]);
}
2020-01-17 01:13:40 +01:00
}
/**
* List all entries of a certain type for the current user , sorted by id .
2020-01-17 18:19:25 +01:00
*
2020-01-17 01:13:40 +01:00
* @ param string $entry_type : Type of entries we want to list [ 'sended' , 'received' , 'scheduled' , 'contact' , 'group' , 'conditional_group' , 'phone' ]
2020-01-17 18:19:25 +01:00
* @ param int $page : Pagination number , Default = 0. Group of 25 results .
*
2020-01-17 18:36:53 +01:00
* @ return : List of entries
2020-01-17 01:13:40 +01:00
*/
2020-01-17 18:19:25 +01:00
public function get_entries ( string $entry_type , int $page = 0 )
2020-01-17 01:13:40 +01:00
{
$entry_types = [ 'sended' , 'received' , 'scheduled' , 'contact' , 'group' , 'conditional_group' , 'phone' ];
2020-01-17 18:19:25 +01:00
if ( ! \in_array ( $entry_type , $entry_types , true ))
2020-01-17 01:13:40 +01:00
{
$return = self :: DEFAULT_RETURN ;
$return [ 'error' ] = self :: ERROR_CODES [ 'INVALID_PARAMETER' ];
2020-01-17 18:47:08 +01:00
$return [ 'message' ] = self :: ERROR_MESSAGES [ 'INVALID_PARAMETER' ] . 'entry_type must be one of : ' . implode ( ', ' , $entry_types ) . '.' ;
2020-01-17 01:13:40 +01:00
$this -> auto_http_code ( false );
2020-06-23 21:06:13 +02:00
2020-06-17 03:02:33 +02:00
return $this -> json ( $return );
2020-01-17 01:13:40 +01:00
}
2020-01-17 18:47:08 +01:00
$controller_str = 'internal_' . $entry_type ;
2020-01-17 18:19:25 +01:00
$controller = $this -> { $controller_str };
2020-01-17 01:13:40 +01:00
$page = ( int ) $page ;
$limit = 25 ;
$entries = $controller -> list_for_user ( $this -> user [ 'id' ], $limit , $page );
//Special case for scheduled, we must add numbers because its a join
2020-01-17 18:19:25 +01:00
if ( 'scheduled' === $entry_type )
2020-01-17 01:13:40 +01:00
{
foreach ( $entries as $key => $entry )
{
$entries [ $key ][ 'numbers' ] = $this -> internal_scheduled -> get_numbers ( $entry [ 'id' ]);
$entries [ $key ][ 'contacts' ] = $this -> internal_scheduled -> get_contacts ( $entry [ 'id' ]);
$entries [ $key ][ 'groups' ] = $this -> internal_scheduled -> get_groups ( $entry [ 'id' ]);
$entries [ $key ][ 'conditional_groups' ] = $this -> internal_scheduled -> get_conditional_groups ( $entry [ 'id' ]);
}
}
//Special case for group we must add contact because its a join
2020-01-17 18:19:25 +01:00
elseif ( 'group' === $entry_type )
2020-01-17 01:13:40 +01:00
{
foreach ( $entries as $key => $entry )
{
$entries [ $key ][ 'contacts' ] = $this -> internal_group -> get_contacts ( $entry [ 'id' ]);
}
}
$return = self :: DEFAULT_RETURN ;
$return [ 'response' ] = $entries ;
2020-01-17 18:19:25 +01:00
if ( \count ( $entries ) === $limit )
2020-01-17 01:13:40 +01:00
{
$return [ 'next' ] = \descartes\Router :: url ( 'Api' , __FUNCTION__ , [ 'entry_type' => $entry_type , 'page' => $page + 1 ], [ 'api_key' => $this -> user [ 'api_key' ]]);
}
if ( $page > 0 )
{
2020-06-16 17:04:48 +02:00
$return [ 'prev' ] = \descartes\Router :: url ( 'Api' , __FUNCTION__ , [ 'entry_type' => $entry_type , 'page' => $page - 1 ], [ 'api_key' => $this -> user [ 'api_key' ]]);
2020-01-17 01:13:40 +01:00
}
$this -> auto_http_code ( true );
2020-06-23 21:06:13 +02:00
2020-06-17 03:02:33 +02:00
return $this -> json ( $return );
2020-01-17 01:13:40 +01:00
}
2020-01-17 16:35:13 +01:00
/**
2020-01-17 18:19:25 +01:00
* Schedule a message to be send .
*
* @ param string $_POST [ 'at' ] : Date to send message at format Y - m - d H : i : s
* @ param string $_POST [ 'text' ] : Text of the message to send
2020-03-04 01:40:47 +01:00
* @ param string $_POST [ 'id_phone' ] : Default null . Id of phone to send the message from . If null use a random phone
2020-01-17 18:19:25 +01:00
* @ param string $_POST [ 'flash' ] : Default false . Is the sms a flash sms .
* @ param string $_POST [ 'numbers' ] : Array of numbers to send message to
* @ param string $_POST [ 'contacts' ] : Array of ids of contacts to send message to
* @ param string $_POST [ 'groups' ] : Array of ids of groups to send message to
2020-01-17 16:35:13 +01:00
* @ param string $_POST [ 'conditional_groups' ] : Array of ids of conditional groups to send message to
2020-01-17 18:19:25 +01:00
*
2020-01-17 18:36:53 +01:00
* @ return : Id of scheduled created
2020-01-17 16:35:13 +01:00
*/
2020-06-23 21:06:13 +02:00
public function post_scheduled ()
2020-01-17 16:35:13 +01:00
{
$at = $_POST [ 'at' ] ? ? false ;
$text = $_POST [ 'text' ] ? ? false ;
2020-03-04 01:40:47 +01:00
$id_phone = empty ( $_POST [ 'id_phone' ]) ? null : $_POST [ 'id_phone' ];
2020-01-17 16:35:13 +01:00
$flash = ( bool ) ( $_POST [ 'flash' ] ? ? false );
2020-01-17 18:19:25 +01:00
$numbers = $_POST [ 'numbers' ] ? ? [];
$contacts = $_POST [ 'contacts' ] ? ? [];
$groups = $_POST [ 'groups' ] ? ? [];
2020-01-17 16:35:13 +01:00
$conditional_groups = $_POST [ 'conditional_groups' ] ? ? [];
2020-06-23 21:06:13 +02:00
$numbers = \is_array ( $numbers ) ? $numbers : [ $numbers ];
$contacts = \is_array ( $contacts ) ? $contacts : [ $contacts ];
$groups = \is_array ( $groups ) ? $groups : [ $groups ];
$conditional_groups = \is_array ( $conditional_groups ) ? $conditional_groups : [ $conditional_groups ];
2020-06-04 13:00:10 +02:00
2020-04-19 18:09:59 +02:00
if ( ! $at )
{
$at = ( new \DateTime ()) -> format ( 'Y-m-d H:i:s' );
}
2020-01-17 16:35:13 +01:00
if ( ! $at || ! $text )
{
$return = self :: DEFAULT_RETURN ;
$return [ 'error' ] = self :: ERROR_CODES [ 'MISSING_PARAMETER' ];
2020-01-17 18:47:08 +01:00
$return [ 'message' ] = self :: ERROR_MESSAGES [ 'MISSING_PARAMETER' ] . ( $at ? '' : 'at ' ) . ( $text ? '' : 'text' );
2020-01-17 16:35:13 +01:00
$this -> auto_http_code ( false );
2020-06-23 21:06:13 +02:00
2020-06-17 03:02:33 +02:00
return $this -> json ( $return );
2020-01-17 16:35:13 +01:00
}
if ( ! \controllers\internals\Tool :: validate_date ( $at , 'Y-m-d H:i:s' ))
{
$return = self :: DEFAULT_RETURN ;
$return [ 'error' ] = self :: ERROR_CODES [ 'INVALID_PARAMETER' ];
2020-01-17 18:47:08 +01:00
$return [ 'message' ] = self :: ERROR_MESSAGES [ 'INVALID_PARAMETER' ] . 'at must be a date of format "Y-m-d H:i:s".' ;
2020-01-17 16:35:13 +01:00
$this -> auto_http_code ( false );
2020-06-23 21:06:13 +02:00
2020-06-17 03:02:33 +02:00
return $this -> json ( $return );
2020-01-17 16:35:13 +01:00
}
foreach ( $numbers as $key => $number )
{
$number = \controllers\internals\Tool :: parse_phone ( $number );
if ( ! $number )
{
unset ( $numbers [ $key ]);
2020-01-17 18:19:25 +01:00
2020-01-17 16:35:13 +01:00
continue ;
}
$numbers [ $key ] = $number ;
}
if ( ! $numbers && ! $contacts && ! $groups && ! $conditional_groups )
{
$return = self :: DEFAULT_RETURN ;
$return [ 'error' ] = self :: ERROR_CODES [ 'MISSING_PARAMETER' ];
2020-01-17 18:47:08 +01:00
$return [ 'message' ] = self :: ERROR_MESSAGES [ 'MISSING_PARAMETER' ] . 'You must specify at least one valid number, contact, group or conditional_group.' ;
2020-01-17 16:35:13 +01:00
$this -> auto_http_code ( false );
2020-06-23 21:06:13 +02:00
2020-06-17 03:02:33 +02:00
return $this -> json ( $return );
2020-01-17 16:35:13 +01:00
}
2020-03-04 01:40:47 +01:00
if ( $id_phone && ! $this -> internal_phone -> get_for_user ( $this -> user [ 'id' ], $id_phone ))
2020-01-17 16:35:13 +01:00
{
$return = self :: DEFAULT_RETURN ;
$return [ 'error' ] = self :: ERROR_CODES [ 'INVALID_PARAMETER' ];
2020-03-04 01:40:47 +01:00
$return [ 'message' ] = self :: ERROR_MESSAGES [ 'INVALID_PARAMETER' ] . 'id_phone : You must specify an id_phone number among thoses of user phones.' ;
2020-01-17 16:35:13 +01:00
$this -> auto_http_code ( false );
2020-06-23 21:06:13 +02:00
2020-06-17 03:02:33 +02:00
return $this -> json ( $return );
2020-01-17 16:35:13 +01:00
}
2020-03-04 01:40:47 +01:00
$scheduled_id = $this -> internal_scheduled -> create ( $this -> user [ 'id' ], $at , $text , $id_phone , $flash , $numbers , $contacts , $groups , $conditional_groups );
2020-01-17 16:35:13 +01:00
if ( ! $scheduled_id )
{
$return = self :: DEFAULT_RETURN ;
$return [ 'error' ] = self :: ERROR_CODES [ 'CANNOT_CREATE' ];
$return [ 'message' ] = self :: ERROR_MESSAGES [ 'CANNOT_CREATE' ];
$this -> auto_http_code ( false );
2020-06-23 21:06:13 +02:00
2020-06-17 03:02:33 +02:00
return $this -> json ( $return );
2020-01-17 16:35:13 +01:00
}
$return = self :: DEFAULT_RETURN ;
$return [ 'response' ] = $scheduled_id ;
$this -> auto_http_code ( true );
2020-06-23 21:06:13 +02:00
2020-06-17 03:02:33 +02:00
return $this -> json ( $return );
2020-01-17 16:35:13 +01:00
}
/**
2020-01-17 18:19:25 +01:00
* Delete a scheduled message .
*
2020-01-17 16:35:13 +01:00
* @ param int $id : Id of scheduled message to delete
2020-01-17 18:47:08 +01:00
*
2020-01-17 18:36:53 +01:00
* @ return bool : void
2020-01-17 16:35:13 +01:00
*/
2020-01-17 18:19:25 +01:00
public function delete_scheduled ( int $id )
2020-01-17 16:35:13 +01:00
{
2020-06-17 03:02:33 +02:00
$return = self :: DEFAULT_RETURN ;
2020-01-17 16:35:13 +01:00
$success = $this -> internal_scheduled -> delete_for_user ( $this -> user [ 'id' ], $id );
if ( ! $success )
{
2020-06-17 03:02:33 +02:00
$return [ 'error' ] = self :: ERROR_CODES [ 'CANNOT_DELETE' ];
$return [ 'message' ] = self :: ERROR_MESSAGES [ 'CANNOT_DELETE' ];
2020-01-17 16:35:13 +01:00
$this -> auto_http_code ( false );
2020-01-17 18:19:25 +01:00
2020-06-17 03:02:33 +02:00
return $this -> json ( $return );
2020-01-17 16:35:13 +01:00
}
2020-06-17 03:02:33 +02:00
$return [ 'response' ] = true ;
2020-01-17 16:35:13 +01:00
$this -> auto_http_code ( true );
2020-06-23 21:06:13 +02:00
2020-06-17 03:02:33 +02:00
return $this -> json ( $return );
2020-01-17 16:35:13 +01:00
}
2021-01-14 03:25:58 +01:00
/**
2021-01-14 03:32:17 +01:00
* Create a new phone .
*
* @ param string $_POST [ 'name' ] : Phone name
* @ param string $_POST [ 'adapter' ] : Phone adapter
* @ param array $_POST [ 'adapter_datas' ] : Phone adapter datas
2021-01-14 03:25:58 +01:00
*
* @ return int : id phone the new phone on success
*/
public function post_phone ()
{
$return = self :: DEFAULT_RETURN ;
2021-01-14 03:32:17 +01:00
2021-01-14 03:25:58 +01:00
$name = $_POST [ 'name' ] ? ? false ;
$adapter = $_POST [ 'adapter' ] ? ? false ;
$adapter_datas = ! empty ( $_POST [ 'adapter_datas' ]) ? $_POST [ 'adapter_datas' ] : [];
if ( ! $name )
{
$return [ 'error' ] = self :: ERROR_CODES [ 'MISSING_PARAMETER' ];
$return [ 'message' ] = self :: ERROR_MESSAGES [ 'MISSING_PARAMETER' ] . ' You must specify phone name.' ;
$this -> auto_http_code ( false );
return $this -> json ( $return );
}
2021-01-14 03:32:17 +01:00
2021-01-14 03:25:58 +01:00
if ( ! $adapter )
{
$return [ 'error' ] = self :: ERROR_CODES [ 'MISSING_PARAMETER' ];
$return [ 'message' ] = self :: ERROR_MESSAGES [ 'MISSING_PARAMETER' ] . ' You must specify adapter name.' ;
$this -> auto_http_code ( false );
return $this -> json ( $return );
}
$name_exist = $this -> internal_phone -> get_by_name ( $name );
if ( $name_exist )
{
$return [ 'error' ] = self :: ERROR_CODES [ 'INVALID_PARAMETER' ];
$return [ 'message' ] = self :: ERROR_MESSAGES [ 'INVALID_PARAMETER' ] . ' This name is already used for another phone.' ;
$this -> auto_http_code ( false );
return $this -> json ( $return );
}
$adapters = $this -> internal_adapter -> list_adapters ();
$find_adapter = false ;
foreach ( $adapters as $metas )
{
if ( $metas [ 'meta_classname' ] === $adapter )
{
$find_adapter = $metas ;
break ;
}
}
if ( ! $find_adapter )
{
$return [ 'error' ] = self :: ERROR_CODES [ 'INVALID_PARAMETER' ];
$return [ 'message' ] = self :: ERROR_MESSAGES [ 'INVALID_PARAMETER' ] . ' adapter. Adapter "' . $adapter . '" does not exists.' ;
$this -> auto_http_code ( false );
return $this -> json ( $return );
}
//If missing required data fields, error
foreach ( $find_adapter [ 'meta_datas_fields' ] as $field )
{
if ( false === $field [ 'required' ])
{
continue ;
}
if ( ! empty ( $adapter_datas [ $field [ 'name' ]]))
{
continue ;
}
$return [ 'error' ] = self :: ERROR_CODES [ 'MISSING_PARAMETER' ];
$return [ 'message' ] = self :: ERROR_MESSAGES [ 'MISSING_PARAMETER' ] . ' You must speicify param ' . $field [ 'name' ] . ' (' . $field [ 'description' ] . ') for this phone.' ;
$this -> auto_http_code ( false );
return $this -> json ( $return );
}
//If field phone number is invalid
foreach ( $find_adapter [ 'meta_datas_fields' ] as $field )
{
if ( false === ( $field [ 'number' ] ? ? false ))
{
continue ;
}
if ( ! empty ( $adapter_datas [ $field [ 'name' ]]))
{
$adapter_datas [ $field [ 'name' ]] = \controllers\internals\Tool :: parse_phone ( $adapter_datas [ $field [ 'name' ]]);
if ( $adapter_datas [ $field [ 'name' ]])
{
continue ;
}
}
2021-01-14 03:32:17 +01:00
2021-01-14 03:25:58 +01:00
$return [ 'error' ] = self :: ERROR_CODES [ 'INVALID_PARAMETER' ];
$return [ 'message' ] = self :: ERROR_MESSAGES [ 'INVALID_PARAMETER' ] . ' field ' . $field [ 'name' ] . ' is not a valid phone number.' ;
$this -> auto_http_code ( false );
return $this -> json ( $return );
}
$adapter_datas = json_encode ( $adapter_datas );
//Check adapter is working correctly with thoses names and datas
$adapter_classname = $find_adapter [ 'meta_classname' ];
$adapter_instance = new $adapter_classname ( $adapter_datas );
$adapter_working = $adapter_instance -> test ();
if ( ! $adapter_working )
{
$return [ 'error' ] = self :: ERROR_CODES [ 'CANNOT_CREATE' ];
$return [ 'message' ] = self :: ERROR_MESSAGES [ 'CANNOT_CREATE' ] . ' : Impossible to validate this phone, verify adapters parameters.' ;
$this -> auto_http_code ( false );
return $this -> json ( $return );
}
$phone_id = $this -> internal_phone -> create ( $this -> user [ 'id' ], $name , $adapter , $adapter_datas );
2021-01-14 03:32:17 +01:00
if ( false === $phone_id )
2021-01-14 03:25:58 +01:00
{
$return [ 'error' ] = self :: ERROR_CODES [ 'CANNOT_CREATE' ];
$return [ 'message' ] = self :: ERROR_MESSAGES [ 'CANNOT_CREATE' ];
$this -> auto_http_code ( false );
return $this -> json ( $return );
}
$return [ 'response' ] = $phone_id ;
$this -> auto_http_code ( true );
2021-01-14 03:32:17 +01:00
2021-01-14 03:25:58 +01:00
return $this -> json ( $return );
}
2021-01-14 03:32:17 +01:00
2021-01-14 03:25:58 +01:00
/**
* Delete a phone .
*
* @ param int $id : Id of phond to delete
*
* @ return bool : void
*/
public function delete_phone ( int $id )
{
$return = self :: DEFAULT_RETURN ;
$success = $this -> internal_phone -> delete_for_user ( $this -> user [ 'id' ], $id );
if ( ! $success )
{
$return [ 'error' ] = self :: ERROR_CODES [ 'CANNOT_DELETE' ];
$return [ 'message' ] = self :: ERROR_MESSAGES [ 'CANNOT_DELETE' ];
$this -> auto_http_code ( false );
return $this -> json ( $return );
}
$return [ 'response' ] = true ;
$this -> auto_http_code ( true );
return $this -> json ( $return );
}
2020-01-17 01:13:40 +01:00
}