Current File : /home/pacjaorg/www/nsa/administrator/components/com_postinstall/src/Model/MessagesModel.php
<?php
/**
 * @package     Joomla.Administrator
 * @subpackage  com_postinstall
 *
 * @copyright   (C) 2013 Open Source Matters, Inc. <https://www.joomla.org>
 * @license     GNU General Public License version 2 or later; see LICENSE.txt
 */

namespace Joomla\Component\Postinstall\Administrator\Model;

\defined('_JEXEC') or die;

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;

/**
 * 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->getDbo();
		$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->getDbo();
		$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');
	}

	/**
	 * 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());
		$published = (int) $this->getState('published', 1);

		// Build a cache ID for the resulting data object
		$cacheId = $eid . '.' . $published;

		$db = $this->getDbo();
		$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->where($db->quoteName('enabled') . ' = :published')
			->bind(':published', $published, ParameterType::INTEGER);
		$db->setQuery($query);

		try
		{
			/** @var CallbackController $cache */
			$cache = Factory::getContainer()->get(CacheControllerFactoryInterface::class)
				->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->getDbo();
		$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->getDbo();
		$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->getDbo();
		$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->getDbo();
		$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->getDbo();

		$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->getDbo();
		$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;
	}
}
Site is undergoing maintenance

PACJA Events

Maintenance mode is on

Site will be available soon. Thank you for your patience!