* * This source file is subject to the GPL-3.0 license that is bundled * with this source code in the file LICENSE. */ namespace controllers\internals; use Monolog\Handler\StreamHandler; use Monolog\Logger; class Scheduled extends StandardController { protected $model; /** * Create a scheduled. * * @param int $id_user : User to insert scheduled for * @param $at : Scheduled date to send * @param string $text : Text of the message * @param ?int $id_phone : Id of the phone to send message with, null by default * @param bool $flash : Is the sms a flash sms, by default false * @param bool $mms : Is the sms a mms, by default false * @param array $numbers : Array of numbers to send message to, a number is an array ['number' => '+33XXX', 'data' => '{"key":"value", ...}'] * @param array $contacts_ids : Contact ids to send message to * @param array $groups_ids : Group ids to send message to * @param array $conditional_group_ids : Conditional Groups ids to send message to * @param array $media_ids : Ids of the medias to link to scheduled message * * @return bool : false on error, new id on success */ public function create(int $id_user, $at, string $text, ?int $id_phone = null, bool $flash = false, bool $mms = false, array $numbers = [], array $contacts_ids = [], array $groups_ids = [], array $conditional_group_ids = [], array $media_ids = []) { $scheduled = [ 'id_user' => $id_user, 'at' => $at, 'text' => $text, 'id_phone' => $id_phone, 'flash' => $flash, 'mms' => $mms, ]; if ('' === $text) { return false; } if (null !== $id_phone) { $internal_phone = new Phone($this->bdd); $find_phone = $internal_phone->get_for_user($id_user, $id_phone); if (!$find_phone) { return false; } } //Use transaction to garanty atomicity $this->bdd->beginTransaction(); $id_scheduled = $this->get_model()->insert($scheduled); if (!$id_scheduled) { $this->bdd->rollBack(); return false; } $internal_media = new Media($this->bdd); foreach ($media_ids as $media_id) { $id_media_scheduled = $internal_media->link_to($media_id, 'scheduled', $id_scheduled); if (!$id_media_scheduled) { $this->bdd->rollBack(); return false; } } foreach ($numbers as $number) { $this->get_model()->insert_scheduled_number($id_scheduled, $number['number'], $number['data']); } $internal_contact = new Contact($this->bdd); foreach ($contacts_ids as $contact_id) { $find_contact = $internal_contact->get_for_user($id_user, $contact_id); if (!$find_contact) { continue; } $this->get_model()->insert_scheduled_contact_relation($id_scheduled, $contact_id); } $internal_group = new Group($this->bdd); foreach ($groups_ids as $group_id) { $find_group = $internal_group->get_for_user($id_user, $group_id); if (!$find_group) { continue; } $this->get_model()->insert_scheduled_group_relation($id_scheduled, $group_id); } $internal_conditional_group = new ConditionalGroup($this->bdd); foreach ($conditional_group_ids as $conditional_group_id) { $find_group = $internal_conditional_group->get_for_user($id_user, $conditional_group_id); if (!$find_group) { continue; } $this->get_model()->insert_scheduled_conditional_group_relation($id_scheduled, $conditional_group_id); } $success = $this->bdd->commit(); if (!$success) { return false; } $date = date('Y-m-d H:i:s'); $internal_event = new Event($this->bdd); $internal_event->create($id_user, 'SCHEDULED_ADD', 'Ajout d\'un Sms pour le ' . $date . '.'); return $id_scheduled; } /** * Update a scheduled. * * @param int $id_user : User to insert scheduled for * @param int $id_scheduled : Scheduled id * @param $at : Scheduled date to send * @param string $text : Text of the message * @param ?int $id_phone : Id of the phone to send message with, null by default * @param bool $flash : Is the sms a flash sms, by default false * @param bool $mms : Is the sms a mms, by default false * @param array $numbers : Array of numbers to send message to, a number is an array ['number' => '+33XXX', 'data' => '{"key":"value", ...}'] * @param array $contacts_ids : Contact ids to send message to * @param array $groups_ids : Group ids to send message to * @param array $conditional_group_ids : Conditional Groups ids to send message to * @param array $media_ids : Ids of the medias to link to scheduled message * * @return bool : false on error, true on success */ public function update_for_user(int $id_user, int $id_scheduled, $at, string $text, ?string $id_phone = null, bool $flash = false, bool $mms = false, array $numbers = [], array $contacts_ids = [], array $groups_ids = [], array $conditional_group_ids = [], array $media_ids = []) { $scheduled = [ 'id_user' => $id_user, 'at' => $at, 'text' => $text, 'id_phone' => $id_phone, 'mms' => $mms, 'flash' => $flash, ]; if (null !== $id_phone) { $internal_phone = new Phone($this->bdd); $find_phone = $internal_phone->get_for_user($id_user, $id_phone); if (!$find_phone) { return false; } } //Ensure atomicity $this->bdd->beginTransaction(); $success = (bool) $this->get_model()->update_for_user($id_user, $id_scheduled, $scheduled); $this->get_model()->delete_scheduled_numbers($id_scheduled); $this->get_model()->delete_scheduled_contact_relations($id_scheduled); $this->get_model()->delete_scheduled_group_relations($id_scheduled); $this->get_model()->delete_scheduled_conditional_group_relations($id_scheduled); $internal_media = new Media($this->bdd); $internal_media->unlink_all_of('scheduled', $id_scheduled); foreach ($media_ids as $media_id) { $id_media_scheduled = $internal_media->link_to($media_id, 'scheduled', $id_scheduled); if (!$id_media_scheduled) { $this->bdd->rollBack(); return false; } } foreach ($numbers as $number) { $this->get_model()->insert_scheduled_number($id_scheduled, $number['number'], $number['data']); } $internal_contact = new Contact($this->bdd); foreach ($contacts_ids as $contact_id) { $find_contact = $internal_contact->get_for_user($id_user, $contact_id); if (!$find_contact) { continue; } $this->get_model()->insert_scheduled_contact_relation($id_scheduled, $contact_id); } $internal_group = new Group($this->bdd); foreach ($groups_ids as $group_id) { $find_group = $internal_group->get_for_user($id_user, $group_id); if (!$find_group) { continue; } $this->get_model()->insert_scheduled_group_relation($id_scheduled, $group_id); } $internal_conditional_group = new ConditionalGroup($this->bdd); foreach ($conditional_group_ids as $conditional_group_id) { $find_group = $internal_conditional_group->get_for_user($id_user, $conditional_group_id); if (!$find_group) { continue; } $this->get_model()->insert_scheduled_conditional_group_relation($id_scheduled, $conditional_group_id); } return $this->bdd->commit(); } /** * Get messages scheduled before a date for a number and a user. * * @param int $id_user : User id * @param $date : Date before which we want messages * @param string $number : Number for which we want messages * * @return array */ public function gets_before_date_for_number_and_user(int $id_user, $date, string $number) { return $this->get_model()->gets_before_date_for_number_and_user($id_user, $date, $number); } /** * Get messages scheduled after a date for a number and a user. * * @param int $id_user : User id * @param $date : Date after which we want messages * @param string $number : Number for which we want messages * * @return array */ public function gets_after_date_for_number_and_user(int $id_user, $date, string $number) { return $this->get_model()->gets_after_date_for_number_and_user($id_user, $date, $number); } /** * Parse a CSV file of numbers, potentially associated with datas. * * @param resource $file_handler : File handler pointing to the file * * @throws Exception : raise exception if file is not valid * * @return mixed : array of numbers ['number' => '+XXXX...', 'data' => ['key' => 'value', ...]] */ public function parse_csv_numbers_file($file_handler) { $numbers = []; $head = null; $line_nb = 0; while ($line = fgetcsv($file_handler)) { ++$line_nb; if (null === $head) { $head = $line; continue; } //Padding line with '' entries to make sure its same length as head //this allow to mix users with data with users without data $line = array_pad($line, \count($head), ''); $line = array_combine($head, $line); if (false === $line) { continue; } $phone_number = \controllers\internals\Tool::parse_phone($line[array_keys($line)[0]] ?? ''); if (!$phone_number) { throw new \Exception('Erreur à la ligne ' . $line_nb . ' colonne 1, numéro de téléphone invalide.'); } $data = []; $i = 0; foreach ($line as $key => $value) { ++$i; if ($i < 2) { // Ignore first column continue; } if ('' === $value) { continue; } $key = mb_ereg_replace('[\W]', '', $key); $data[$key] = $value; } $numbers[] = ['number' => $phone_number, 'data' => $data]; } return $numbers; } /** * Return numbers for a scheduled message. * * @param int $id_scheduled : Scheduled id * * @return array */ public function get_numbers(int $id_scheduled) { return $this->get_model()->get_numbers($id_scheduled); } /** * Return contacts for a scheduled message. * * @param int $id_scheduled : Scheduled id * * @return array */ public function get_contacts(int $id_scheduled) { return $this->get_model()->get_contacts($id_scheduled); } /** * Return groups for a scheduled message. * * @param int $id_scheduled : Scheduled id * * @return array */ public function get_groups(int $id_scheduled) { return $this->get_model()->get_groups($id_scheduled); } /** * Return conditional groups for a scheduled message. * * @param int $id_scheduled : Scheduled id * * @return array */ public function get_conditional_groups(int $id_scheduled) { return $this->get_model()->get_conditional_groups($id_scheduled); } /** * Get the model for the Controller. */ protected function get_model(): \models\Scheduled { $this->model = $this->model ?? new \models\Scheduled($this->bdd); return $this->model; } /** * Get all messages to send and the number to use to send theme. * * @return array : List of smss to send at this time per scheduled id ['1' => [['id_scheduled', 'text', 'id_phone', 'destination', 'flash', 'mms', 'medias'], ...], ...] */ public function get_smss_to_send() { $sms_per_scheduled = []; $internal_templating = new \controllers\internals\Templating(); $internal_setting = new \controllers\internals\Setting($this->bdd); $internal_group = new \controllers\internals\Group($this->bdd); $internal_conditional_group = new \controllers\internals\ConditionalGroup($this->bdd); $internal_phone = new \controllers\internals\Phone($this->bdd); $internal_smsstop = new \controllers\internals\SmsStop($this->bdd); $internal_sended = new \controllers\internals\Sended($this->bdd); $users_smsstops = []; $users_settings = []; $users_phones = []; $users_mms_phones = []; $now = new \DateTime(); $now = $now->format('Y-m-d H:i:s'); $scheduleds = $this->get_model()->gets_before_date($now); foreach ($scheduleds as $scheduled) { $id_scheduled = $scheduled['id']; $id_user = $scheduled['id_user']; $sms_per_scheduled[$id_scheduled] = []; // Forge cache of data about users, sms stops, phones, etc. if (!isset($users_settings[$id_user])) { $users_settings[$id_user] = []; $settings = $internal_setting->gets_for_user($id_user); foreach ($settings as $name => $value) { $users_settings[$id_user][$name] = $value; } } if (!isset($users_smsstops[$id_user]) && $users_settings[$id_user]['smsstop']) { $users_smsstops[$id_user] = []; $smsstops = $internal_smsstop->gets_for_user($id_user); foreach ($smsstops as $smsstop) { $users_smsstops[$id_user][] = $smsstop['number']; } } if (!isset($users_phones[$id_user])) { $users_phones[$id_user] = []; $users_mms_phones[$id_user] = []; $phones = $internal_phone->gets_for_user($id_user); foreach ($phones as &$phone) { $limits = $internal_phone->get_limits($phone['id']); $remaining_volume = PHP_INT_MAX; foreach ($limits as $limit) { $startpoint = new \DateTime($limit['startpoint']); $consumed = $internal_sended->count_since_for_phone_and_user($id_user, $phone['id'], $startpoint); $remaining_volume = min(($limit['volume'] - $consumed), $remaining_volume); } $phone['remaining_volume'] = $remaining_volume; $users_phones[$id_user][$phone['id']] = $phone; } $mms_phones = $internal_phone->gets_phone_supporting_mms_for_user($id_user, $internal_phone::MMS_SENDING); foreach ($mms_phones as &$mms_phone) { $limits = $internal_phone->get_limits($mms_phone['id']); $remaining_volume = PHP_INT_MAX; foreach ($limits as $limit) { $startpoint = new \DateTime($limit['startpoint']); $consumed = $internal_sended->count_since_for_phone_and_user($id_user, $mms_phone['id'], $startpoint); $remaining_volume = min(($limit['volume'] - $consumed), $remaining_volume); } $mms_phone['remaining_volume'] = $remaining_volume; $mms_phones[$id_user][$mms_phone['id']] = $mms_phone; } } //Add medias to mms $scheduled['medias'] = []; if ($scheduled['mms']) { $internal_media = new Media($this->bdd); $scheduled['medias'] = $internal_media->gets_for_scheduled($id_scheduled); } $phone_to_use = null; if ($scheduled['id_phone']) { $phone_to_use = $users_phones[$id_user][$scheduled['id_phone']] ?? null; } // We turn all contacts, groups and conditional groups into just contacts $contacts = $this->get_contacts($id_scheduled); $groups = $this->get_groups($id_scheduled); foreach ($groups as $group) { $contacts_to_add = $internal_group->get_contacts($group['id']); $contacts = array_merge($contacts, $contacts_to_add); } $conditional_groups = $this->get_conditional_groups($id_scheduled); foreach ($conditional_groups as $conditional_group) { $contacts_to_add = $internal_conditional_group->get_contacts_for_condition_and_user($id_user, $conditional_group['condition']); $contacts = array_merge($contacts, $contacts_to_add); } // We turn all numbers and contacts into simple targets with number, data and meta so we can forge all messages from onlye one data source $targets = []; $numbers = $this->get_numbers($id_scheduled); foreach ($numbers as $number) { $metas = ['number' => $number['number']]; $targets[] = [ 'number' => $number['number'], 'data' => $number['data'], 'metas' => $metas, ]; } foreach ($contacts as $contact) { $metas = $contact; unset($metas['data'], $metas['id_user']); $targets[] = [ 'number' => $contact['number'], 'data' => $contact['data'], 'metas' => $metas, ]; } // Pass on all targets to deduplicate destinations, remove number in sms stops, etc. $used_destinations = []; foreach ($targets as $key => $target) { if (in_array($target['number'], $used_destinations)) { unset($targets[$key]); continue; } //Remove messages to smsstops numbers if (($users_smsstops[$id_user] ?? false) && in_array($target['number'], $users_smsstops[$id_user])) { continue; } $used_destinations[] = $target['number']; } // Finally, we forge all messages and select phone to use foreach ($targets as $target) { // Forge message if templating enable $text = $scheduled['text']; if ((int) ($users_settings[$id_user]['templating'] ?? false)) // Cast to int because it is more reliable than bool on strings { $target['data'] = json_decode($target['data'], true); $data = ['contact' => $target['data'], 'contact_metas' => $target['metas']]; $render = $internal_templating->render($scheduled['text'], $data); if (!$render['success']) { continue; } $text = $render['result']; } // Ignore empty messages if ('' === trim($text) && !$scheduled['medias']) { continue; } // If we must force GSM 7 alphabet if ((int) ($users_settings[$id_user]['force_gsm_alphabet'] ?? false)) { $text = Tool::convert_to_gsm0338($text); } /* Choose phone if no phone defined for message Phones are choosen using type, priority and remaining volume : 1 - If sms is a mms, try to use mms phone if any available. If mms phone available use mms phone, else use default. 2 - In group of phones, keep only phones with remaining volume. If no phones with remaining volume, use all phones instead. 3 - Groupe keeped phones by priority get group with biggest priority. 4 - Get a random phone in this group. 5 - If their is no phone matching, keep phone at null so sender will directly mark it as failed */ $random_phone = null; if (null === $phone_to_use) { $phones_subset = $users_phones[$id_user]; if ($scheduled['mms']) { $phones_subset = $users_mms_phones[$id_user] ?: $phones_subset; } // Keep only phones with remaining volume and available status $remaining_volume_phones = array_filter($phones_subset, function ($phone) { return $phone['remaining_volume'] > 0 && $phone['status'] == \models\Phone::STATUS_AVAILABLE; }); $phones_subset = $remaining_volume_phones ?: $phones_subset; $max_priority_phones = []; $max_priority = PHP_INT_MIN; foreach ($phones_subset as $phone) { if ($phone['priority'] < $max_priority) { continue; } elseif ($phone['priority'] == $max_priority) { $max_priority_phones[] = $phone; } elseif ($phone['priority'] > $max_priority) { $max_priority_phones = [$phone]; $max_priority = $phone['priority']; } } $phones_subset = $max_priority_phones; if ($phones_subset) { $random_phone = $phones_subset[array_rand($phones_subset)]; } } // This should only happen if the user try to send a message without any phone in his account, then we simply ignore. if (!$random_phone && !$phone_to_use) { continue; } $id_phone = $phone_to_use['id'] ?? $random_phone['id']; $sms_per_scheduled[$id_scheduled][] = [ 'id_user' => $id_user, 'id_scheduled' => $id_scheduled, 'id_phone' => $id_phone, 'destination' => $target['number'], 'flash' => $scheduled['flash'], 'mms' => $scheduled['mms'], 'medias' => $scheduled['medias'], 'text' => $text, ]; // Consume one sms from remaining volume of phone, dont forget to do the same for the entry in mms phones $users_phones[$id_user][$id_phone]['remaining_volume'] --; if ($users_mms_phones[$id_user][$id_phone] ?? false) { $users_mms_phones[$id_user][$id_phone] --; } } } return $sms_per_scheduled; } }