, Christian Binder * @version $Id$ */ use EGroupware\Api; /** * Ajax methods for notifications */ class notifications_ajax { /** * Appname */ const _appname = 'notifications'; /** * Mailappname */ const _mailappname = 'mail'; /** * Notification table in SQL database */ const _notification_table = 'egw_notificationpopup'; /** * Notification type */ const _type = 'base'; /** * Do NOT consider notifications older than this */ const CUT_OFF_DATE = '-30days'; /** * holds account object for user to notify * * @var object */ private $recipient; /** * holds config object (sitewide application config) * * @var object */ private $config; /** * holds preferences array of user to notify * * @var array */ private $preferences; /** * reference to global db object * * @var Api\Db */ private $db; /** * the xml response object * * @var Api\Json\Response */ private $response; /** * Do we have a real push server, or only a fallback * * @var bool */ private $isPushServer; /** * constructor * */ public function __construct() { $this->response = Api\Json\Response::get(); $this->recipient = (object)$GLOBALS['egw']->accounts->read($GLOBALS['egw_info']['user']['account_id']); $this->db = $GLOBALS['egw']->db; $this->isPushServer = Api\Cache::getInstance('notifications', 'isPushServer', function () { return !Api\Json\Push::onlyFallback(); }, [], 900); } /** * public AJAX trigger function to be called by the JavaScript client * * this function calls all other recurring AJAX notifications methods * to have ONE single recurring AJAX call per user * * @return xajax response */ public function get_notifications($browserNotify = false) { // close session now, to not block other user actions, as specially mail checks can be time consuming $GLOBALS['egw']->session->commit_session(); // call a hook for notifications on new mail //if ($GLOBALS['egw_info']['user']['apps']['mail']) $this->check_mailbox(); Api\Hooks::process('check_notify'); // update currentusers if ($GLOBALS['egw_info']['user']['apps']['admin'] && $GLOBALS['egw_info']['user']['preferences']['common']['show_currentusers']) { $this->response->jquery('#currentusers', 'text', array((string)$GLOBALS['egw']->session->session_count())); } if ($this->isPushServer) $this->response->data(['isPushServer' => true]); $this->get_egwpopup($browserNotify); } /** * Remove given notification id(s) from the table * * @param array $notifymessages one or multiple notify_id(s) */ public function delete_message(array $notifymessages) { $this->update($notifymessages, null); // null = delete } /** * Method to update message(s) status * * @param array $notifymessages one or more notify_id(s) * @param string $status = SEEN, status of message: * - SEEN: message has been seen * - UNSEEN: message has not been seen * - DISPLAYED: message has been shown but no further status applied yet * this status has been used more specifically for browser type * of notifications. */ public function update_status(array $notifymessages, $status = "SEEN") { $this->update($notifymessages, $status); } /** * Update or delete the given notification messages, incl. not explicitly mentioned ones with same app:id * * @param array $notifymessages * @param string|null $status use null to delete * @return array */ protected function update(array $notifymessages, $status='SEEN') { $notify_ids = $app_ids = []; foreach ($notifymessages as $data) { if (is_array($data) && !empty($data['id'])) { if (is_array($data['data'] ?? null) && !empty($data['data']['id'])) { $app_ids[$data['data']['app']][$data['data']['id']] = $data['data']['id']; } $notify_ids[] = $data['id']; } else { $notify_ids[] = $data; } } $cut_off = $this->db->quote(Api\DateTime::to(self::CUT_OFF_DATE, Api\DateTime::DATABASE)); try { // MariaDB code using JSON_EXTRACT() foreach($app_ids as $app => $ids) { $where = [ 'account_id' => $this->recipient->account_id, 'notify_type' => self::_type, "JSON_EXTRACT(notify_data, '$.appname') = ".$this->db->quote($app), "JSON_EXTRACT(notify_data, '$.data.id') IN (".implode(',', array_map([$this->db, 'quote'], array_unique($ids))).')', 'notify_created > '.$cut_off, ]; if (isset($status)) { $this->db->update(self::_notification_table, ['notify_status' => $status], $where, __LINE__, __FILE__, self::_appname); } else { $this->db->delete(self::_notification_table, $where, __LINE__, __FILE__, self::_appname); } } } // other DBs catch (Api\Db\Exception $e) { foreach($this->db->select(self::_notification_table, 'notify_id,notify_data', [ 'account_id' => $this->recipient->account_id, 'notify_type' => self::_type, 'notify_created > '.$cut_off, "notify_data <> '[]'", // does not return NULL or '[]' rows ]) as $row) { if (($data = json_decode($row['notify_data'], true)) && isset($data['data']['id']) && in_array($data['data']['id'], $app_ids[$data['appname']] ?? [])) { $notify_ids[] = $row['notify_id']; } } } $where = [ 'notify_id' => array_unique($notify_ids), 'account_id' => $this->recipient->account_id, 'notify_type' => self::_type ]; if (isset($status)) { $this->db->update(self::_notification_table, ['notify_status' => $status], $where, __LINE__, __FILE__, self::_appname); } else { $this->db->delete(self::_notification_table, $where, __LINE__, __FILE__, self::_appname); } // cleanup messages older than our cut-off-date $this->db->delete(self::_notification_table, [ 'notify_created <= '.$cut_off, 'notify_type' => self::_type ], __LINE__, __FILE__, self::_appname); } /** * gets all egwpopup notifications for calling user * * @return boolean true or false */ private function get_egwpopup($browserNotify = false) { $entries = notifications_popup::read($this->recipient->account_id); $this->response->apply('app.notifications.append', array($entries['rows']??[], $browserNotify, $entries['total']??0)); return true; } }