* * 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; /** * Page des discussions. */ class Discussion extends \descartes\Controller { private $internal_sended; private $internal_scheduled; private $internal_received; private $internal_contact; private $internal_phone; private $internal_media; /** * Cette fonction est appelée avant toute les autres : * Elle vérifie que l'utilisateur est bien connecté. * * @return void; */ public function __construct() { $bdd = \descartes\Model::_connect(DATABASE_HOST, DATABASE_NAME, DATABASE_USER, DATABASE_PASSWORD); $this->internal_sended = new \controllers\internals\Sended($bdd); $this->internal_scheduled = new \controllers\internals\Scheduled($bdd); $this->internal_received = new \controllers\internals\Received($bdd); $this->internal_contact = new \controllers\internals\Contact($bdd); $this->internal_phone = new \controllers\internals\Phone($bdd); $this->internal_media = new \controllers\internals\Media($bdd); \controllers\internals\Tool::verifyconnect(); } /** * Cette fonction retourne toutes les discussions, sous forme d'un tableau permettant l'administration de ces contacts. */ public function list() { $this->render('discussion/list'); } /** * Return discussions as json. */ public function list_json() { $entities = $this->internal_received->get_discussions_for_user($_SESSION['user']['id']); foreach ($entities as &$entity) { $entity['number_formatted'] = \controllers\internals\Tool::phone_link($entity['number']); $entity['link'] = \descartes\Router::url('Discussion', 'show', ['number' => $entity['number']]); } header('Content-Type: application/json'); echo json_encode(['data' => $entities]); } /** * Cette fonction permet d'afficher la discussion avec un numero. * * @param string $number : La numéro de téléphone avec lequel on discute */ public function show($number) { $contact = $this->internal_contact->get_by_number_and_user($_SESSION['user']['id'], $number); $last_sended = $this->internal_sended->get_last_for_destination_and_user($_SESSION['user']['id'], $number); $last_received = $this->internal_received->get_last_for_origin_and_user($_SESSION['user']['id'], $number); $response_phone_id = ($last_received['id_phone'] ?? $last_sended['id_phone'] ?? false); if ($response_phone_id) { $response_phone = $this->internal_phone->get_for_user($_SESSION['user']['id'], $response_phone_id); } $this->render('discussion/show', [ 'number' => $number, 'contact' => $contact, 'response_phone' => $response_phone ?? false, ]); } /** * Cette fonction récupère l'ensemble des messages pour un numéro, recçus, envoyés, en cours. * * @param string $number : Le numéro cible * @param string $transaction_id : Le numéro unique de la transaction ajax (sert à vérifier si la requete doit être prise en compte) * @param ?int $since : La date à partir de laquelle on veut les messages, sous forme de timestamp */ public function get_messages($number, $transaction_id, ?int $since = null) { $now = new \DateTime(); $now = $now->format('Y-m-d H:i:s'); $id_user = $_SESSION['user']['id']; if ($since && !($since = date('Y-m-d H:i:s', $since))) { echo json_encode(['transaction_id' => $transaction_id, 'messages' => [], 'error' => true, 'error_message' => 'Not a valid date.']); return false; } if ($since) { $sendeds = $this->internal_sended->gets_since_date_by_destination_and_user($id_user, $since, $number); $receiveds = $this->internal_received->gets_since_date_by_origin_and_user($id_user, $since, $number); $scheduleds = $this->internal_scheduled->gets_after_date_for_number_and_user($id_user, $since, $number); } else { $sendeds = $this->internal_sended->gets_by_destination_and_user($id_user, $number); $receiveds = $this->internal_received->gets_by_origin_and_user($id_user, $number); $scheduleds = $this->internal_scheduled->gets_before_date_for_number_and_user($id_user, $now, $number); } $messages = []; foreach ($sendeds as $sended) { $medias = []; if ($sended['mms']) { $medias = $this->internal_media->gets_for_sended($sended['id']); foreach ($medias as &$media) { $media = HTTP_PWD_DATA_PUBLIC . '/' . $media['path']; } } $message = [ 'uid' => 'sended-' . $sended['id'], 'date' => htmlspecialchars($sended['at']), 'text' => htmlspecialchars($sended['text']), 'type' => 'sended', 'medias' => $medias, 'status' => $sended['status'], ]; $messages[] = $message; } foreach ($receiveds as $received) { if ('read' !== $received['status']) { $this->internal_received->mark_as_read_for_user($id_user, $received['id']); } $medias = []; if ($received['mms']) { $medias = $this->internal_media->gets_for_received($received['id']); foreach ($medias as &$media) { $media = HTTP_PWD_DATA_PUBLIC . '/' . $media['path']; } } $messages[] = [ 'uid' => 'received-' . $received['id'], 'date' => htmlspecialchars($received['at']), 'text' => htmlspecialchars($received['text']), 'type' => 'received', 'medias' => $medias, ]; } foreach ($scheduleds as $scheduled) { $medias = []; if ($scheduled['mms']) { $medias = $this->internal_media->gets_for_scheduled($scheduled['id']); foreach ($medias as &$media) { $media = HTTP_PWD_DATA_PUBLIC . '/' . $media['path']; } } $messages[] = [ 'uid' => 'scheduled-' . $scheduled['id'], 'date' => htmlspecialchars($scheduled['at']), 'text' => htmlspecialchars($scheduled['text']), 'type' => 'inprogress', 'medias' => $medias, ]; } //On va trier le tableau des messages usort($messages, function ($a, $b) { return strtotime($a['date']) - strtotime($b['date']); }); //Sans limite de temps, on récupère uniquement les 25 derniers messages sur l'ensemble pour limiter la charge if (!$since) { $messages = \array_slice($messages, -25); } $response = [ 'transaction_id' => $transaction_id, 'messages' => $messages, ]; if ($messages) { $new_limit_date = (new \DateTime($messages[count($messages) - 1]['date']))->getTimestamp(); //Use latest message as the new limit date to search $response['new_limit_date'] = $new_limit_date; } echo json_encode($response); return true; } /** * Cette fonction permet d'envoyer facilement un sms à un numéro donné. * * @param string $csrf : Le jeton csrf * @param string $_POST['text'] : Le contenu du Sms * @param string $_POST['destination'] : Number to send sms to * @param string $_POST['id_phone'] : If of phone to send sms with * @param array $_FILES['medias'] : Medias to upload and link to sms * * @return string : json string Le statut de l'envoi */ public function send($csrf) { $return = ['success' => true, 'message' => '']; //On vérifie que le jeton csrf est bon if (!$this->verify_csrf($csrf)) { $return['success'] = false; $return['message'] = 'Jeton CSRF invalide'; echo json_encode($return); return false; } $now = new \DateTime(); $now = $now->format('Y-m-d H:i:s'); $id_user = $_SESSION['user']['id']; $at = $now; $text = $_POST['text'] ?? ''; $destination = $_POST['destination'] ?? false; $id_phone = $_POST['id_phone'] ?? false; $files = $_FILES['medias'] ?? false; //Iterate over files to re-create individual $_FILES array $files_arrays = []; if ($files && is_array($files['name'])) { foreach ($files as $property_name => $files_values) { foreach ($files_values as $file_key => $property_value) { if (!isset($files_arrays[$file_key])) { $files_arrays[$file_key] = []; } $files_arrays[$file_key][$property_name] = $property_value; } } } //Remove empty files input foreach ($files_arrays as $key => $file) { if ($file['error'] === UPLOAD_ERR_NO_FILE) { unset($files_arrays[$key]); } } if (!$text) { $return['success'] = false; $return['message'] = 'Vous devez renseigner le texte de votre sms.'; echo json_encode($return); return false; } if (!$destination) { $return['success'] = false; $return['message'] = 'Vous devez renseigner un numéro valide'; echo json_encode($return); return false; } if (!$id_phone) { $id_phone = null; } //If mms is enable and we have medias uploaded $media_ids = []; if ($_SESSION['user']['settings']['mms'] && $files_arrays) { foreach ($files_arrays as $file) { try { $new_media_id = $this->internal_media->create_from_uploaded_file_for_user($_SESSION['user']['id'], $file); } catch (\Exception $e) { $return['success'] = false; $return['message'] = $e->getMessage(); echo json_encode($return); return false; } $media_ids[] = $new_media_id; } } $mms = (bool) count($media_ids); //Destinations must be an array of number $destinations = [$destination]; if (!$this->internal_scheduled->create($id_user, $at, $text, $id_phone, false, $mms, $destinations, [], [], [], $media_ids)) { $return['success'] = false; $return['message'] = 'Impossible de créer le Sms'; echo json_encode($return); return false; } echo json_encode($return); return true; } /** * Cette fonction retourne les id des sms qui sont envoyés. * * @return string : json string Tableau des ids des sms qui sont envoyés */ public function checksendeds() { $_SESSION['discussion_wait_progress'] = isset($_SESSION['discussion_wait_progress']) ? $_SESSION['discussion_wait_progress'] : []; $scheduleds = $this->internal_scheduled->gets_in_for_user($_SESSION['user']['id'], $_SESSION['discussion_wait_progress']); //On va chercher à chaque fois si on a trouvé le sms. Si ce n'est pas le cas c'est qu'il a été envoyé $sendeds = []; foreach ($_SESSION['discussion_wait_progress'] as $key => $id_scheduled) { $found = false; foreach ($scheduleds as $scheduled) { if ($id_scheduled === $scheduled['id']) { $found = true; } } if (!$found) { unset($_SESSION['discussion_wait_progress'][$key]); $sendeds[] = $id_scheduled; } } echo json_encode($sendeds); return true; } /** * Cette fonction retourne les messages reçus pour un numéro après la date $_SESSION['discussion_last_checkreceiveds']. * * @param string $number : Le numéro de téléphone pour lequel on veux les messages * * @return string : json string Un tableau avec les messages */ public function checkreceiveds($number) { $now = new \DateTime(); $now = $now->format('Y-m-d H:i'); $_SESSION['discussion_last_checkreceiveds'] = isset($_SESSION['discussion_last_checkreceiveds']) ? $_SESSION['discussion_last_checkreceiveds'] : $now; $receiveds = $this->internal_received->get_since_for_number_by_date($_SESSION['discussion_last_checkreceiveds'], $number); //On va gérer le cas des messages en double en stockant ceux déjà reçus et en eliminant les autres $_SESSION['discussion_already_receiveds'] = isset($_SESSION['discussion_already_receiveds']) ? $_SESSION['discussion_already_receiveds'] : []; foreach ($receiveds as $key => $received) { //Sms jamais recu if (false === array_search($received['id'], $_SESSION['discussion_already_receiveds'], true)) { $_SESSION['discussion_already_receiveds'][] = $received['id']; continue; } //Sms déjà reçu => on le supprime des resultats unset($receiveds[$key]); } //On met à jour la date de dernière verif $_SESSION['discussion_last_checkreceiveds'] = $now; echo json_encode($receiveds); } }