* @license GNU General Public License version 2 or later; see LICENSE.txt */ namespace Joomla\Component\Postinstall\Administrator\Model; use Joomla\CMS\Cache\CacheControllerFactoryInterface; use Joomla\CMS\Cache\Controller\CallbackController; use Joomla\CMS\Extension\ExtensionHelper; use Joomla\CMS\Factory; use Joomla\CMS\Filesystem\File; use Joomla\CMS\HTML\HTMLHelper; use Joomla\CMS\Language\Text; use Joomla\CMS\MVC\Model\BaseDatabaseModel; use Joomla\Component\Postinstall\Administrator\Helper\PostinstallHelper; use Joomla\Database\ParameterType; // phpcs:disable PSR1.Files.SideEffects \defined('_JEXEC') or die; // phpcs:enable PSR1.Files.SideEffects /** * Model class to manage postinstall messages * * @since 3.2 */ class MessagesModel extends BaseDatabaseModel { /** * Method to auto-populate the state. * * This method should only be called once per instantiation and is designed * to be called on the first call to the getState() method unless the * configuration flag to ignore the request is set. * * @return void * * @note Calling getState in this method will result in recursion. * @since 4.0.0 */ protected function populateState() { parent::populateState(); $eid = (int) Factory::getApplication()->input->getInt('eid'); if ($eid) { $this->setState('eid', $eid); } } /** * Gets an item with the given id from the database * * @param integer $id The item id * * @return Object * * @since 3.2 */ public function getItem($id) { $db = $this->getDatabase(); $id = (int) $id; $query = $db->getQuery(true); $query->select( [ $db->quoteName('postinstall_message_id'), $db->quoteName('extension_id'), $db->quoteName('title_key'), $db->quoteName('description_key'), $db->quoteName('action_key'), $db->quoteName('language_extension'), $db->quoteName('language_client_id'), $db->quoteName('type'), $db->quoteName('action_file'), $db->quoteName('action'), $db->quoteName('condition_file'), $db->quoteName('condition_method'), $db->quoteName('version_introduced'), $db->quoteName('enabled'), ] ) ->from($db->quoteName('#__postinstall_messages')) ->where($db->quoteName('postinstall_message_id') . ' = :id') ->bind(':id', $id, ParameterType::INTEGER); $db->setQuery($query); $result = $db->loadObject(); return $result; } /** * Unpublishes specified post-install message * * @param integer $id The message id * * @return void */ public function unpublishMessage($id) { $db = $this->getDatabase(); $id = (int) $id; $query = $db->getQuery(true); $query ->update($db->quoteName('#__postinstall_messages')) ->set($db->quoteName('enabled') . ' = 0') ->where($db->quoteName('postinstall_message_id') . ' = :id') ->bind(':id', $id, ParameterType::INTEGER); $db->setQuery($query); $db->execute(); Factory::getCache()->clean('com_postinstall'); } /** * Archives specified post-install message * * @param integer $id The message id * * @return void * * @since 4.2.0 */ public function archiveMessage($id) { $db = $this->getDatabase(); $id = (int) $id; $query = $db->getQuery(true); $query ->update($db->quoteName('#__postinstall_messages')) ->set($db->quoteName('enabled') . ' = 2') ->where($db->quoteName('postinstall_message_id') . ' = :id') ->bind(':id', $id, ParameterType::INTEGER); $db->setQuery($query); $db->execute(); Factory::getCache()->clean('com_postinstall'); } /** * Republishes specified post-install message * * @param integer $id The message id * * @return void * * @since 4.2.0 */ public function republishMessage($id) { $db = $this->getDatabase(); $id = (int) $id; $query = $db->getQuery(true); $query ->update($db->quoteName('#__postinstall_messages')) ->set($db->quoteName('enabled') . ' = 1') ->where($db->quoteName('postinstall_message_id') . ' = :id') ->bind(':id', $id, ParameterType::INTEGER); $db->setQuery($query); $db->execute(); Factory::getCache()->clean('com_postinstall'); } /** * Returns a list of messages from the #__postinstall_messages table * * @return array * * @since 3.2 */ public function getItems() { // Add a forced extension filtering to the list $eid = (int) $this->getState('eid', $this->getJoomlaFilesExtensionId()); // Build a cache ID for the resulting data object $cacheId = 'postinstall_messages.' . $eid; $db = $this->getDatabase(); $query = $db->getQuery(true); $query->select( [ $db->quoteName('postinstall_message_id'), $db->quoteName('extension_id'), $db->quoteName('title_key'), $db->quoteName('description_key'), $db->quoteName('action_key'), $db->quoteName('language_extension'), $db->quoteName('language_client_id'), $db->quoteName('type'), $db->quoteName('action_file'), $db->quoteName('action'), $db->quoteName('condition_file'), $db->quoteName('condition_method'), $db->quoteName('version_introduced'), $db->quoteName('enabled'), ] ) ->from($db->quoteName('#__postinstall_messages')); $query->where($db->quoteName('extension_id') . ' = :eid') ->bind(':eid', $eid, ParameterType::INTEGER); // Force filter only enabled messages $query->whereIn($db->quoteName('enabled'), [1, 2]); $db->setQuery($query); try { /** @var CallbackController $cache */ $cache = $this->getCacheControllerFactory()->createCacheController('callback', ['defaultgroup' => 'com_postinstall']); $result = $cache->get(array($db, 'loadObjectList'), array(), md5($cacheId), false); } catch (\RuntimeException $e) { $app = Factory::getApplication(); $app->getLogger()->warning( Text::sprintf('JLIB_APPLICATION_ERROR_MODULE_LOAD', $e->getMessage()), array('category' => 'jerror') ); return array(); } $this->onProcessList($result); return $result; } /** * Returns a count of all enabled messages from the #__postinstall_messages table * * @return integer * * @since 4.0.0 */ public function getItemsCount() { $db = $this->getDatabase(); $query = $db->getQuery(true); $query->select( [ $db->quoteName('language_extension'), $db->quoteName('language_client_id'), $db->quoteName('condition_file'), $db->quoteName('condition_method'), ] ) ->from($db->quoteName('#__postinstall_messages')); // Force filter only enabled messages $query->where($db->quoteName('enabled') . ' = 1'); $db->setQuery($query); try { /** @var CallbackController $cache */ $cache = Factory::getContainer()->get(CacheControllerFactoryInterface::class) ->createCacheController('callback', ['defaultgroup' => 'com_postinstall']); // Get the resulting data object for cache ID 'all.1' from com_postinstall group. $result = $cache->get(array($db, 'loadObjectList'), array(), md5('all.1'), false); } catch (\RuntimeException $e) { $app = Factory::getApplication(); $app->getLogger()->warning( Text::sprintf('JLIB_APPLICATION_ERROR_MODULE_LOAD', $e->getMessage()), array('category' => 'jerror') ); return 0; } $this->onProcessList($result); return \count($result); } /** * Returns the name of an extension, as registered in the #__extensions table * * @param integer $eid The extension ID * * @return string The extension name * * @since 3.2 */ public function getExtensionName($eid) { // Load the extension's information from the database $db = $this->getDatabase(); $eid = (int) $eid; $query = $db->getQuery(true) ->select( [ $db->quoteName('name'), $db->quoteName('element'), $db->quoteName('client_id'), ] ) ->from($db->quoteName('#__extensions')) ->where($db->quoteName('extension_id') . ' = :eid') ->bind(':eid', $eid, ParameterType::INTEGER) ->setLimit(1); $db->setQuery($query); $extension = $db->loadObject(); if (!is_object($extension)) { return ''; } // Load language files $basePath = JPATH_ADMINISTRATOR; if ($extension->client_id == 0) { $basePath = JPATH_SITE; } $lang = Factory::getApplication()->getLanguage(); $lang->load($extension->element, $basePath); // Return the localised name return Text::_(strtoupper($extension->name)); } /** * Resets all messages for an extension * * @param integer $eid The extension ID whose messages we'll reset * * @return mixed False if we fail, a db cursor otherwise * * @since 3.2 */ public function resetMessages($eid) { $db = $this->getDatabase(); $eid = (int) $eid; $query = $db->getQuery(true) ->update($db->quoteName('#__postinstall_messages')) ->set($db->quoteName('enabled') . ' = 1') ->where($db->quoteName('extension_id') . ' = :eid') ->bind(':eid', $eid, ParameterType::INTEGER); $db->setQuery($query); $result = $db->execute(); Factory::getCache()->clean('com_postinstall'); return $result; } /** * Hides all messages for an extension * * @param integer $eid The extension ID whose messages we'll hide * * @return mixed False if we fail, a db cursor otherwise * * @since 3.8.7 */ public function hideMessages($eid) { $db = $this->getDatabase(); $eid = (int) $eid; $query = $db->getQuery(true) ->update($db->quoteName('#__postinstall_messages')) ->set($db->quoteName('enabled') . ' = 0') ->where($db->quoteName('extension_id') . ' = :eid') ->bind(':eid', $eid, ParameterType::INTEGER); $db->setQuery($query); $result = $db->execute(); Factory::getCache()->clean('com_postinstall'); return $result; } /** * List post-processing. This is used to run the programmatic display * conditions against each list item and decide if we have to show it or * not. * * Do note that this a core method of the RAD Layer which operates directly * on the list it's being fed. A little touch of modern magic. * * @param array &$resultArray A list of items to process * * @return void * * @since 3.2 */ protected function onProcessList(&$resultArray) { $unset_keys = array(); $language_extensions = array(); // Order the results DESC so the newest is on the top. $resultArray = array_reverse($resultArray); foreach ($resultArray as $key => $item) { // Filter out messages based on dynamically loaded programmatic conditions. if (!empty($item->condition_file) && !empty($item->condition_method)) { $helper = new PostinstallHelper(); $file = $helper->parsePath($item->condition_file); if (File::exists($file)) { require_once $file; $result = call_user_func($item->condition_method); if ($result === false) { $unset_keys[] = $key; } } } // Load the necessary language files. if (!empty($item->language_extension)) { $hash = $item->language_client_id . '-' . $item->language_extension; if (!in_array($hash, $language_extensions)) { $language_extensions[] = $hash; Factory::getApplication()->getLanguage()->load($item->language_extension, $item->language_client_id == 0 ? JPATH_SITE : JPATH_ADMINISTRATOR); } } } if (!empty($unset_keys)) { foreach ($unset_keys as $key) { unset($resultArray[$key]); } } } /** * Get the dropdown options for the list of component with post-installation messages * * @since 3.4 * * @return array Compatible with JHtmlSelect::genericList */ public function getComponentOptions() { $db = $this->getDatabase(); $query = $db->getQuery(true) ->select($db->quoteName('extension_id')) ->from($db->quoteName('#__postinstall_messages')) ->group($db->quoteName('extension_id')); $db->setQuery($query); $extension_ids = $db->loadColumn(); $options = array(); Factory::getApplication()->getLanguage()->load('files_joomla.sys', JPATH_SITE, null, false, false); foreach ($extension_ids as $eid) { $options[] = HTMLHelper::_('select.option', $eid, $this->getExtensionName($eid)); } return $options; } /** * Adds or updates a post-installation message (PIM) definition. You can use this in your post-installation script using this code: * * require_once JPATH_LIBRARIES . '/fof/include.php'; * FOFModel::getTmpInstance('Messages', 'PostinstallModel')->addPostInstallationMessage($options); * * The $options array contains the following mandatory keys: * * extension_id The numeric ID of the extension this message is for (see the #__extensions table) * * type One of message, link or action. Their meaning is: * message Informative message. The user can dismiss it. * link The action button links to a URL. The URL is defined in the action parameter. * action A PHP action takes place when the action button is clicked. You need to specify the action_file * (RAD path to the PHP file) and action (PHP function name) keys. See below for more information. * * title_key The Text language key for the title of this PIM. * Example: COM_FOOBAR_POSTINSTALL_MESSAGEONE_TITLE * * description_key The Text language key for the main body (description) of this PIM * Example: COM_FOOBAR_POSTINSTALL_MESSAGEONE_DESCRIPTION * * action_key The Text language key for the action button. Ignored and not required when type=message * Example: COM_FOOBAR_POSTINSTALL_MESSAGEONE_ACTION * * language_extension The extension name which holds the language keys used above. * For example, com_foobar, mod_something, plg_system_whatever, tpl_mytemplate * * language_client_id Should we load the frontend (0) or backend (1) language keys? * * version_introduced Which was the version of your extension where this message appeared for the first time? * Example: 3.2.1 * * enabled Must be 1 for this message to be enabled. If you omit it, it defaults to 1. * * condition_file The RAD path to a PHP file containing a PHP function which determines whether this message should be shown to * the user. @see FOFTemplateUtils::parsePath() for RAD path format. Joomla! will include this file before calling * the condition_method. * Example: admin://components/com_foobar/helpers/postinstall.php * * condition_method The name of a PHP function which will be used to determine whether to show this message to the user. This must be * a simple PHP user function (not a class method, static method etc) which returns true to show the message and false * to hide it. This function is defined in the condition_file. * Example: com_foobar_postinstall_messageone_condition * * When type=message no additional keys are required. * * When type=link the following additional keys are required: * * action The URL which will open when the user clicks on the PIM's action button * Example: index.php?option=com_foobar&view=tools&task=installSampleData * * When type=action the following additional keys are required: * * action_file The RAD path to a PHP file containing a PHP function which performs the action of this PIM. @see FOFTemplateUtils::parsePath() * for RAD path format. Joomla! will include this file before calling the function defined in the action key below. * Example: admin://components/com_foobar/helpers/postinstall.php * * action The name of a PHP function which will be used to run the action of this PIM. This must be a simple PHP user function * (not a class method, static method etc) which returns no result. * Example: com_foobar_postinstall_messageone_action * * @param array $options See description * * @return $this * * @throws \Exception */ public function addPostInstallationMessage(array $options) { // Make sure there are options set if (!is_array($options)) { throw new \Exception('Post-installation message definitions must be of type array', 500); } // Initialise array keys $defaultOptions = array( 'extension_id' => '', 'type' => '', 'title_key' => '', 'description_key' => '', 'action_key' => '', 'language_extension' => '', 'language_client_id' => '', 'action_file' => '', 'action' => '', 'condition_file' => '', 'condition_method' => '', 'version_introduced' => '', 'enabled' => '1', ); $options = array_merge($defaultOptions, $options); // Array normalisation. Removes array keys not belonging to a definition. $defaultKeys = array_keys($defaultOptions); $allKeys = array_keys($options); $extraKeys = array_diff($allKeys, $defaultKeys); if (!empty($extraKeys)) { foreach ($extraKeys as $key) { unset($options[$key]); } } // Normalisation of integer values $options['extension_id'] = (int) $options['extension_id']; $options['language_client_id'] = (int) $options['language_client_id']; $options['enabled'] = (int) $options['enabled']; // Normalisation of 0/1 values foreach (array('language_client_id', 'enabled') as $key) { $options[$key] = $options[$key] ? 1 : 0; } // Make sure there's an extension_id if (!(int) $options['extension_id']) { throw new \Exception('Post-installation message definitions need an extension_id', 500); } // Make sure there's a valid type if (!in_array($options['type'], array('message', 'link', 'action'))) { throw new \Exception('Post-installation message definitions need to declare a type of message, link or action', 500); } // Make sure there's a title key if (empty($options['title_key'])) { throw new \Exception('Post-installation message definitions need a title key', 500); } // Make sure there's a description key if (empty($options['description_key'])) { throw new \Exception('Post-installation message definitions need a description key', 500); } // If the type is anything other than message you need an action key if (($options['type'] != 'message') && empty($options['action_key'])) { throw new \Exception('Post-installation message definitions need an action key when they are of type "' . $options['type'] . '"', 500); } // You must specify the language extension if (empty($options['language_extension'])) { throw new \Exception('Post-installation message definitions need to specify which extension contains their language keys', 500); } // The action file and method are only required for the "action" type if ($options['type'] == 'action') { if (empty($options['action_file'])) { throw new \Exception('Post-installation message definitions need an action file when they are of type "action"', 500); } $helper = new PostinstallHelper(); $file_path = $helper->parsePath($options['action_file']); if (!@is_file($file_path)) { throw new \Exception('The action file ' . $options['action_file'] . ' of your post-installation message definition does not exist', 500); } if (empty($options['action'])) { throw new \Exception('Post-installation message definitions need an action (function name) when they are of type "action"', 500); } } if ($options['type'] == 'link') { if (empty($options['link'])) { throw new \Exception('Post-installation message definitions need an action (URL) when they are of type "link"', 500); } } // The condition file and method are only required when the type is not "message" if ($options['type'] != 'message') { if (empty($options['condition_file'])) { throw new \Exception('Post-installation message definitions need a condition file when they are of type "' . $options['type'] . '"', 500); } $helper = new PostinstallHelper(); $file_path = $helper->parsePath($options['condition_file']); if (!@is_file($file_path)) { throw new \Exception('The condition file ' . $options['condition_file'] . ' of your post-installation message definition does not exist', 500); } if (empty($options['condition_method'])) { throw new \Exception( 'Post-installation message definitions need a condition method (function name) when they are of type "' . $options['type'] . '"', 500 ); } } // Check if the definition exists $table = $this->getTable(); $tableName = $table->getTableName(); $extensionId = (int) $options['extension_id']; $db = $this->getDatabase(); $query = $db->getQuery(true) ->select('*') ->from($db->quoteName($tableName)) ->where( [ $db->quoteName('extension_id') . ' = :extensionId', $db->quoteName('type') . ' = :type', $db->quoteName('title_key') . ' = :titleKey', ] ) ->bind(':extensionId', $extensionId, ParameterType::INTEGER) ->bind(':type', $options['type']) ->bind(':titleKey', $options['title_key']); $existingRow = $db->setQuery($query)->loadAssoc(); // Is the existing definition the same as the one we're trying to save? if (!empty($existingRow)) { $same = true; foreach ($options as $k => $v) { if ($existingRow[$k] != $v) { $same = false; break; } } // Trying to add the same row as the existing one; quit if ($same) { return $this; } // Otherwise it's not the same row. Remove the old row before insert a new one. $query = $db->getQuery(true) ->delete($db->quoteName($tableName)) ->where( [ $db->quoteName('extension_id') . ' = :extensionId', $db->quoteName('type') . ' = :type', $db->quoteName('title_key') . ' = :titleKey', ] ) ->bind(':extensionId', $extensionId, ParameterType::INTEGER) ->bind(':type', $options['type']) ->bind(':titleKey', $options['title_key']); $db->setQuery($query)->execute(); } // Insert the new row $options = (object) $options; $db->insertObject($tableName, $options); Factory::getCache()->clean('com_postinstall'); return $this; } /** * Returns the library extension ID. * * @return integer * * @since 4.0.0 */ public function getJoomlaFilesExtensionId() { return ExtensionHelper::getExtensionRecord('joomla', 'file')->extension_id; } }