* * This source file is subject to the GPL-3.0 license that is bundled * with this source code in the file LICENSE. */ namespace adapters; use Ovh\Api; /** * OVH SMS service with a shortcode allowing responses. */ class OvhSmsShortcodeAdapter implements AdapterInterface { /** * Data used to configure interaction with the implemented service. (e.g : Api credentials, ports numbers, etc.). */ private $data; /** * OVH Api instance. */ private $api; /** * Adapter constructor, called when instanciated by RaspiSMS. * * @param string $number : Phone number the adapter is used for * @param json string $data : JSON string of the data to configure interaction with the implemented service */ public function __construct(string $data) { $this->data = json_decode($data, true); $this->api = new Api( $this->data['app_key'], $this->data['app_secret'], 'ovh-eu', $this->data['consumer_key'] ); } /** * Classname of the adapter. */ public static function meta_classname(): string { return __CLASS__; } /** * Uniq name of the adapter * It should be the classname of the adapter un snakecase. */ public static function meta_uid(): string { return 'ovh_sms_shortcode_adapter'; } /** * Should this adapter be hidden in user interface for phone creation and * available to creation through API only. */ public static function meta_hidden(): bool { return false; } /** * Should this adapter data be hidden after creation * this help to prevent API credentials to other service leak if an attacker gain access to RaspiSMS through user credentials. */ public static function meta_hide_data(): bool { return false; } /** * Name of the adapter. * It should probably be the name of the service it adapt (e.g : Gammu SMSD, OVH SMS, SIM800L, etc.). */ public static function meta_name(): string { return 'OVH SMS Shortcode'; } /** * Description of the adapter. * A short description of the service the adapter implements. */ public static function meta_description(): string { $generate_credentials_url = 'https://eu.api.ovh.com/createToken/index.cgi?GET=/sms&GET=/sms/*&POST=/sms/*&PUT=/sms/*&DELETE=/sms/*&'; return ' Solution de SMS proposé par le groupe OVH. Pour générer les clefs API OVH, cliquez ici.
Pour plus d\'information sur l\'utilisation de ce type de téléphone, reportez-vous à la documentation sur les téléphones "OVH Shortcode". '; } /** * List of entries we want in data for the adapter. * * @return array : Every line is a field as an array with keys : name, title, description, required */ public static function meta_data_fields(): array { return [ [ 'name' => 'service_name', 'title' => 'Service Name', 'description' => 'Service Name de votre service SMS chez OVH. Il s\'agit du nom associé à votre service SMS dans la console OVH, probablement quelque chose comme "sms-xxxxx-1" ou "xxxx" est votre identifiant client OVH.', 'required' => true, ], [ 'name' => 'sender', 'title' => 'Nom de l\'expéditeur', 'description' => 'Nom de l\'expéditeur à afficher à la place du numéro (11 caractères max).
Laissez vide pour ne pas utiliser d\'expéditeur nommé.

Le nom doit avoir été validé au préallable. Si vous utilisez un expéditeur nommé, le destinataire ne pourra pas répondre.', 'required' => false, ], [ 'name' => 'app_key', 'title' => 'Application Key', 'description' => 'Paramètre "Application Key" obtenu lors de la génération de la clef API OVH.', 'required' => true, ], [ 'name' => 'app_secret', 'title' => 'Application Secret', 'description' => 'Paramètre "Application Secret" obtenu lors de la génération de la clef API OVH.', 'required' => true, ], [ 'name' => 'consumer_key', 'title' => 'Consumer Key', 'description' => 'Paramètre "Consumer Key" obtenu lors de la génération de la clef API OVH.', 'required' => true, ], [ 'name' => 'no_stop_clause', 'title' => 'Désactiver la clause "STOP SMS" automatique', 'description' => 'En cochant ce paramètre, la clause "STOP SMS" ne sera pas ajoutée automatiquement au SMS par OVH.', 'required' => false, 'type' => 'boolean' ], ]; } /** * Does the implemented service support reading smss. */ public static function meta_support_read(): bool { return true; } /** * Does the implemented service support updating phone status. */ public static function meta_support_phone_status(): bool { return false; } /** * Does the implemented service support flash smss. */ public static function meta_support_flash(): bool { return false; } /** * Does the implemented service support status change. */ public static function meta_support_status_change(): bool { return true; } /** * Does the implemented service support reception callback. */ public static function meta_support_reception(): bool { return false; } /** * Does the implemented service support mms reception. */ public static function meta_support_mms_reception(): bool { return false; } /** * Does the implemented service support mms sending. */ public static function meta_support_mms_sending(): bool { return false; } public static function meta_support_inbound_call_callback(): bool { return false; } public static function meta_support_end_call_callback(): bool { return false; } public function send(string $destination, string $text, bool $flash = false, bool $mms = false, array $medias = []): array { $response = [ 'error' => false, 'error_message' => null, 'uid' => null, ]; try { $success = true; $endpoint = '/sms/' . $this->data['service_name'] . '/jobs'; $params = [ 'message' => $text, 'receivers' => [$destination], 'senderForResponse' => true, 'noStopClause' => (bool) ($this->data['no_stop_clause'] ?? false), ]; if ($this->data['sender']) { $params['sender'] = $this->data['sender']; $params['senderForResponse'] = false; } $response = $this->api->post($endpoint, $params); $nb_invalid_receivers = \count(($response['invalidReceivers'] ?? [])); if ($nb_invalid_receivers > 0) { $response['error'] = true; $response['error_message'] = 'Invalid receiver'; return $response; } $uid = $response['ids'][0] ?? false; if (!$uid) { $response['error'] = true; $response['error_message'] = 'Cannot retrieve uid.'; return $response; } $response['uid'] = $uid; return $response; } catch (\Throwable $t) { $response['error'] = true; $response['error_message'] = $t->getMessage(); return $response; } } public function read(): array { $response = [ 'error' => false, 'error_message' => null, 'smss' => [], ]; try { //If we use a sender we cannot receive response, no need to make queries if ($this->data['sended']) { return $response; } $endpoint = '/sms/' . $this->data['service_name'] . '/incoming'; $uids = $this->api->get($endpoint); if (!\is_array($uids) || !$uids) { return $response; } foreach ($uids as $uid) { $endpoint = '/sms/' . $this->data['service_name'] . '/incoming/' . $uid; $sms_details = $this->api->get($endpoint); if (!isset($sms_details['creationDatetime'], $sms_details['message'], $sms_details['sender'])) { continue; } $response['smss'][] = [ 'at' => (new \DateTime($sms_details['creationDatetime']))->format('Y-m-d H:i:s'), 'text' => $sms_details['message'], 'origin' => $sms_details['sender'], ]; //Remove the sms to prevent double reading as ovh do not offer a filter for unread messages only $endpoint = '/sms/' . $this->data['service_name'] . '/incoming/' . $uid; $this->api->delete($endpoint); } return $response; } catch (\Throwable $t) { $response['error'] = true; $response['error_message'] = $t->getMessage(); return $response; } } /** * Method called to verify phone status * * @return string : Return one phone status among 'available', 'unavailable', 'no_credit' */ public function check_phone_status(): string { return \models\Phone::STATUS_AVAILABLE; } public function test(): bool { try { $success = true; if ($this->data['sender'] && (mb_strlen($this->data['sender']) < 3 || mb_strlen($this->data['sender']) > 11)) { return false; } //Check service name $endpoint = '/sms/' . $this->data['service_name']; $response = $this->api->get($endpoint); return $success && (bool) $response; } catch (\Throwable $t) { return false; } } public static function status_change_callback() { $uid = $_GET['id'] ?? false; $dlr = $_GET['dlr'] ?? false; if (false === $uid || false === $dlr) { return false; } switch ($dlr) { case 1: $status = \models\Sended::STATUS_DELIVERED; break; case 2: case 16: $status = \models\Sended::STATUS_FAILED; break; default: $status = \models\Sended::STATUS_UNKNOWN; break; } return ['uid' => $uid, 'status' => $status]; } public static function reception_callback(): array { return []; } public function inbound_call_callback(): array { return []; } public function end_call_callback(): array { return []; } }