2021-02-02 22:14:16 +00:00
|
|
|
<?php
|
|
|
|
/**
|
|
|
|
* @package Joomla.JEDChecker
|
|
|
|
*
|
2021-03-11 13:00:58 +00:00
|
|
|
* @copyright Copyright (C) 2021 Open Source Matters, Inc. All rights reserved.
|
2021-02-02 22:14:16 +00:00
|
|
|
*
|
|
|
|
* @license GNU General Public License version 2 or later; see LICENSE.txt
|
|
|
|
*/
|
|
|
|
|
|
|
|
defined('_JEXEC') or die('Restricted access');
|
|
|
|
|
2023-08-13 09:22:27 +00:00
|
|
|
use Joomla\CMS\Filesystem\Folder;
|
|
|
|
use Joomla\CMS\Language\Text;
|
2021-02-02 22:14:16 +00:00
|
|
|
|
|
|
|
// Include the rule base class
|
|
|
|
require_once JPATH_COMPONENT_ADMINISTRATOR . '/models/rule.php';
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* class JedcheckerRulesLanguage
|
|
|
|
*
|
|
|
|
* This class validates language ini file
|
|
|
|
*
|
2021-02-24 11:52:01 +00:00
|
|
|
* @since 3.0
|
2021-02-02 22:14:16 +00:00
|
|
|
*/
|
|
|
|
class JedcheckerRulesLanguage extends JEDcheckerRule
|
|
|
|
{
|
2021-02-22 21:00:06 +00:00
|
|
|
/**
|
|
|
|
* The formal ID of this rule. For example: SE1.
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
protected $id = 'LANG';
|
2021-02-02 22:14:16 +00:00
|
|
|
|
2021-02-22 21:00:06 +00:00
|
|
|
/**
|
|
|
|
* The title or caption of this rule.
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
protected $title = 'COM_JEDCHECKER_LANG';
|
2021-02-02 22:14:16 +00:00
|
|
|
|
2021-02-22 21:00:06 +00:00
|
|
|
/**
|
|
|
|
* The description of this rule.
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
protected $description = 'COM_JEDCHECKER_LANG_DESC';
|
2021-02-02 22:14:16 +00:00
|
|
|
|
2021-05-17 20:25:11 +00:00
|
|
|
/**
|
|
|
|
* The ordering value to sort rules in the menu.
|
|
|
|
*
|
|
|
|
* @var integer
|
|
|
|
*/
|
|
|
|
public static $ordering = 1100;
|
|
|
|
|
2021-05-11 20:52:40 +00:00
|
|
|
/**
|
|
|
|
* Key-value map for language translations
|
|
|
|
*
|
|
|
|
* @var array
|
|
|
|
*/
|
2021-05-09 15:39:07 +00:00
|
|
|
protected $langKeys = array();
|
|
|
|
|
2021-02-22 21:00:06 +00:00
|
|
|
/**
|
|
|
|
* Initiates the search and check
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function check()
|
|
|
|
{
|
2021-03-09 20:42:18 +00:00
|
|
|
// Find all INI files of the extension
|
2023-08-13 09:22:27 +00:00
|
|
|
$files = Folder::files($this->basedir, '\.ini$', true, true);
|
2021-02-02 22:14:16 +00:00
|
|
|
|
2021-02-22 21:00:06 +00:00
|
|
|
// Iterate through all the ini files
|
|
|
|
foreach ($files as $file)
|
|
|
|
{
|
2021-03-09 20:42:18 +00:00
|
|
|
/* Language file format is either tag.extension.ini or tag.extension.sys.ini
|
|
|
|
(where "tag" is a language code, e.g. en-GB, and "extension" is the extension element name, e.g. com_content)
|
|
|
|
Joomla!4 allows to skip tag prefix inside of the tag directory
|
|
|
|
(i.e. to name files as extension.ini and extension.sys.ini) */
|
2021-05-09 15:39:07 +00:00
|
|
|
if (preg_match('#(?:^|/)([a-z]{2,3}-[A-Z]{2})(?:[./]\w+)?(?:\.sys)?\.ini$#', $file, $match))
|
2021-03-09 20:42:18 +00:00
|
|
|
{
|
2021-05-09 15:39:07 +00:00
|
|
|
$tag = $match[1];
|
|
|
|
|
2021-03-09 20:42:18 +00:00
|
|
|
// Try to validate the file
|
2021-05-09 15:39:07 +00:00
|
|
|
$this->find($file, $tag);
|
|
|
|
|
|
|
|
if ($tag === 'en-GB')
|
|
|
|
{
|
|
|
|
$this->populateLangKeys($file);
|
|
|
|
}
|
2021-03-09 20:42:18 +00:00
|
|
|
}
|
2021-02-22 21:00:06 +00:00
|
|
|
}
|
2021-05-09 15:39:07 +00:00
|
|
|
|
|
|
|
// Load default Joomla's translations
|
|
|
|
$files = version_compare(JVERSION, '4.0', '>=') ? array('joomla.ini', 'lib_joomla.ini') : array('en-GB.ini', 'en-GB.lib_joomla.ini');
|
|
|
|
|
|
|
|
foreach ($files as $file)
|
|
|
|
{
|
|
|
|
$this->populateLangKeys(JPATH_ROOT . '/language/en-GB/' . $file);
|
|
|
|
$this->populateLangKeys(JPATH_ADMINISTRATOR . '/language/en-GB/' . $file);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check JText usage
|
2023-08-13 09:22:27 +00:00
|
|
|
$files = Folder::files($this->basedir, '\.php$', true, true);
|
2021-05-09 15:39:07 +00:00
|
|
|
|
|
|
|
foreach ($files as $file)
|
|
|
|
{
|
|
|
|
$this->findJText($file);
|
|
|
|
}
|
2021-02-22 21:00:06 +00:00
|
|
|
}
|
2021-02-02 22:14:16 +00:00
|
|
|
|
2021-02-22 21:00:06 +00:00
|
|
|
/**
|
|
|
|
* Reads and validates an ini file
|
|
|
|
*
|
|
|
|
* @param string $file - The path to the file
|
2021-03-09 20:42:18 +00:00
|
|
|
* @param string $tag - Language tag code
|
2021-02-22 21:00:06 +00:00
|
|
|
*
|
2021-04-04 10:40:43 +00:00
|
|
|
* @return boolean True on success, otherwise False.
|
2021-02-22 21:00:06 +00:00
|
|
|
*/
|
2021-03-09 20:42:18 +00:00
|
|
|
protected function find($file, $tag)
|
2021-02-22 21:00:06 +00:00
|
|
|
{
|
2021-03-09 20:43:21 +00:00
|
|
|
$content = file_get_contents($file);
|
|
|
|
|
|
|
|
if ($content === false)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check EOL format is \n (not \r or \n\r)
|
|
|
|
if (strpos($content, "\r") !== false)
|
|
|
|
{
|
2023-08-13 09:22:27 +00:00
|
|
|
$this->report->addNotice($file, Text::_('COM_JEDCHECKER_LANG_INCORRECT_EOL', false, false));
|
2021-03-09 20:43:21 +00:00
|
|
|
}
|
|
|
|
|
2021-02-22 21:00:06 +00:00
|
|
|
$lines = file($file);
|
2021-03-09 20:50:26 +00:00
|
|
|
|
|
|
|
if ($lines === false)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-02-23 20:41:30 +00:00
|
|
|
$nLines = count($lines);
|
2021-02-23 20:44:53 +00:00
|
|
|
$keys = array();
|
2021-02-23 20:41:30 +00:00
|
|
|
|
2021-03-09 20:45:24 +00:00
|
|
|
// Use mb_check_encoding (if exists) to validate UTF-8
|
|
|
|
$mbExists = function_exists('mb_check_encoding');
|
|
|
|
|
2021-02-23 20:41:30 +00:00
|
|
|
for ($lineno = 0; $lineno < $nLines; $lineno++)
|
2021-02-22 21:00:06 +00:00
|
|
|
{
|
2021-02-23 20:41:30 +00:00
|
|
|
$startLineno = $lineno + 1;
|
|
|
|
$line = trim($lines[$lineno]);
|
|
|
|
|
2021-03-10 22:35:15 +00:00
|
|
|
// Check for BOM sequence
|
2021-02-23 19:26:09 +00:00
|
|
|
if ($lineno === 0 && strncmp($line, "\xEF\xBB\xBF", 3) === 0)
|
|
|
|
{
|
2023-08-13 09:22:27 +00:00
|
|
|
$this->report->addWarning($file, Text::_('COM_JEDCHECKER_LANG_BOM_FOUND'), $startLineno);
|
2021-02-24 11:52:01 +00:00
|
|
|
|
2021-05-11 20:53:52 +00:00
|
|
|
// Remove BOM for further checks
|
2021-02-23 19:26:09 +00:00
|
|
|
$line = substr($line, 3);
|
|
|
|
}
|
2021-03-10 22:35:15 +00:00
|
|
|
|
|
|
|
// Skip empty lines, comments, and section names
|
2021-02-22 21:00:06 +00:00
|
|
|
if ($line === '' || $line[0] === ';' || $line[0] === '[')
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
2021-03-10 22:35:15 +00:00
|
|
|
|
|
|
|
// Report incorrect comment character
|
2021-02-22 21:00:06 +00:00
|
|
|
if ($line[0] === '#')
|
|
|
|
{
|
2023-08-13 09:22:27 +00:00
|
|
|
$this->report->addError($file, Text::_('COM_JEDCHECKER_LANG_INCORRECT_COMMENT'), $startLineno, $line);
|
2021-02-22 21:00:06 +00:00
|
|
|
continue;
|
|
|
|
}
|
2021-03-10 22:35:15 +00:00
|
|
|
|
|
|
|
// Check for "=" character in the line
|
2021-02-22 21:00:06 +00:00
|
|
|
if (strpos($line, '=') === false)
|
|
|
|
{
|
2023-08-13 09:22:27 +00:00
|
|
|
$this->report->addError($file, Text::_('COM_JEDCHECKER_LANG_WRONG_LINE'), $startLineno, $line);
|
2021-02-22 21:00:06 +00:00
|
|
|
continue;
|
|
|
|
}
|
2021-03-10 22:35:15 +00:00
|
|
|
|
|
|
|
// Extract key and value
|
2021-02-22 21:00:06 +00:00
|
|
|
list ($key, $value) = explode('=', $line, 2);
|
2021-02-02 22:14:16 +00:00
|
|
|
|
2021-02-23 20:41:30 +00:00
|
|
|
// Validate key
|
2021-02-22 21:00:06 +00:00
|
|
|
$key = rtrim($key);
|
2021-03-10 22:35:15 +00:00
|
|
|
|
|
|
|
// Check for empty key
|
2021-02-22 21:00:06 +00:00
|
|
|
if ($key === '')
|
|
|
|
{
|
2023-08-13 09:22:27 +00:00
|
|
|
$this->report->addError($file, Text::_('COM_JEDCHECKER_LANG_KEY_EMPTY'), $startLineno, $line);
|
2021-02-22 21:00:06 +00:00
|
|
|
continue;
|
|
|
|
}
|
2021-03-10 22:35:15 +00:00
|
|
|
|
|
|
|
// Check for spaces in the key name
|
2021-03-09 20:50:59 +00:00
|
|
|
if (preg_match('/\s/', $key))
|
2021-02-22 21:00:06 +00:00
|
|
|
{
|
2023-08-13 09:22:27 +00:00
|
|
|
$this->report->addError($file, Text::_('COM_JEDCHECKER_LANG_KEY_WHITESPACE'), $startLineno, $line);
|
2021-02-22 21:00:06 +00:00
|
|
|
continue;
|
|
|
|
}
|
2021-03-10 22:35:15 +00:00
|
|
|
|
|
|
|
// Check for invalid characters (see https://www.php.net/manual/en/function.parse-ini-file.php)
|
2021-02-22 21:00:06 +00:00
|
|
|
if (strpbrk($key, '{}|&~![()^"') !== false)
|
|
|
|
{
|
2023-08-13 09:22:27 +00:00
|
|
|
$this->report->addError($file, Text::_('COM_JEDCHECKER_LANG_KEY_INVALID_CHARACTER'), $startLineno, $line);
|
2021-02-22 21:00:06 +00:00
|
|
|
continue;
|
|
|
|
}
|
2021-03-10 22:35:15 +00:00
|
|
|
|
|
|
|
// Check for invalid key names (see https://www.php.net/manual/en/function.parse-ini-file.php)
|
2021-02-22 21:00:06 +00:00
|
|
|
if (in_array($key, array('null', 'yes', 'no', 'true', 'false', 'on', 'off', 'none'), true))
|
|
|
|
{
|
2023-08-13 09:22:27 +00:00
|
|
|
$this->report->addError($file, Text::_('COM_JEDCHECKER_LANG_KEY_RESERVED'), $startLineno, $line);
|
2021-02-22 21:00:06 +00:00
|
|
|
continue;
|
|
|
|
}
|
2021-03-10 22:35:15 +00:00
|
|
|
|
2021-04-04 11:54:06 +00:00
|
|
|
// Check key contains ASCII characters only
|
2021-02-23 20:49:12 +00:00
|
|
|
if (preg_match('/[\x00-\x1F\x80-\xFF]/', $key))
|
|
|
|
{
|
2023-08-13 09:22:27 +00:00
|
|
|
$this->report->addWarning($file, Text::_('COM_JEDCHECKER_LANG_KEY_NOT_ASCII'), $startLineno, $line);
|
2021-02-23 20:49:12 +00:00
|
|
|
}
|
|
|
|
|
2021-04-04 11:54:06 +00:00
|
|
|
// Check key is uppercase
|
2021-02-23 20:49:12 +00:00
|
|
|
if ($key !== strtoupper($key))
|
|
|
|
{
|
2023-08-13 09:22:27 +00:00
|
|
|
$this->report->addWarning($file, Text::_('COM_JEDCHECKER_LANG_KEY_NOT_UPPERCASE'), $startLineno, $line);
|
2021-02-23 20:49:12 +00:00
|
|
|
}
|
|
|
|
|
2021-04-04 11:54:06 +00:00
|
|
|
// Check for duplicated keys
|
2021-02-23 20:44:53 +00:00
|
|
|
if (isset($keys[$key]))
|
|
|
|
{
|
2023-08-13 09:22:27 +00:00
|
|
|
$this->report->addWarning($file, Text::sprintf('COM_JEDCHECKER_LANG_KEY_DUPLICATED', $keys[$key]), $startLineno, $line);
|
2021-02-23 20:44:53 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$keys[$key] = $startLineno;
|
|
|
|
}
|
|
|
|
|
2021-02-23 20:41:30 +00:00
|
|
|
// Validate value
|
2021-02-22 21:00:06 +00:00
|
|
|
$value = ltrim($value);
|
2021-03-10 22:35:15 +00:00
|
|
|
|
2024-01-08 08:56:04 +00:00
|
|
|
if (!preg_match('/^"((?>[^"\\\\]+|\\\\.)*)"\s*(;[^"]*)?$/', $value, $matches))
|
2021-02-22 21:00:06 +00:00
|
|
|
{
|
2024-01-08 08:56:04 +00:00
|
|
|
// The value doesn't match INI format
|
2024-01-08 09:54:26 +00:00
|
|
|
$this->report->addError($file, Text::_('COM_JEDCHECKER_LANG_TRANSLATION_ERROR'), $startLineno, $line);
|
2021-02-22 21:00:06 +00:00
|
|
|
continue;
|
|
|
|
}
|
2021-03-10 22:35:15 +00:00
|
|
|
|
2021-04-04 11:54:06 +00:00
|
|
|
// Get value w/o comment
|
2024-01-08 08:56:04 +00:00
|
|
|
$value = $matches[1];
|
2021-02-23 20:41:30 +00:00
|
|
|
|
2021-03-10 22:35:15 +00:00
|
|
|
// Check for empty value
|
2024-01-08 08:56:04 +00:00
|
|
|
if ($value === '')
|
2021-02-22 21:00:06 +00:00
|
|
|
{
|
2023-08-13 09:22:27 +00:00
|
|
|
$this->report->addNotice($file, Text::_('COM_JEDCHECKER_LANG_TRANSLATION_EMPTY'), $startLineno, $line);
|
2021-02-22 21:00:06 +00:00
|
|
|
continue;
|
|
|
|
}
|
2021-02-02 22:14:16 +00:00
|
|
|
|
2021-03-09 20:45:24 +00:00
|
|
|
// Check it's a valid UTF-8 string
|
|
|
|
$validUTF8 = $mbExists ? mb_check_encoding($value, 'UTF-8') : preg_match('//u', $value);
|
|
|
|
|
|
|
|
if (!$validUTF8)
|
|
|
|
{
|
2023-08-13 09:22:27 +00:00
|
|
|
$this->report->addWarning($file, Text::_('COM_JEDCHECKER_LANG_INVALID_UTF8'), $startLineno, $line);
|
2021-03-09 20:45:24 +00:00
|
|
|
}
|
|
|
|
|
2024-01-08 08:56:04 +00:00
|
|
|
// Process backwards compatibility break introduced in Joomla 5.0.1
|
|
|
|
if (preg_match('/\\\\[\\\\\\$]/', $value))
|
2021-02-23 20:53:52 +00:00
|
|
|
{
|
2024-01-08 08:56:04 +00:00
|
|
|
$this->report->addWarning($file, Text::_('COM_JEDCHECKER_LANG_JOOMLA501_BC'), $startLineno, $line);
|
2021-02-23 20:53:52 +00:00
|
|
|
}
|
|
|
|
|
2021-03-01 22:22:39 +00:00
|
|
|
// The code below detects incorrect format of numbered placeholders (e.g. "%1s" instead of "%1$s")
|
2021-02-23 20:41:30 +00:00
|
|
|
|
2021-03-01 22:22:39 +00:00
|
|
|
// Count numbered placeholders in the string (e.g. "%1s")
|
2021-07-11 14:30:49 +00:00
|
|
|
$count = preg_match_all('/(?<=^|[^%])%(\d+)\w/', $value, $matches, PREG_SET_ORDER);
|
2021-02-23 20:41:30 +00:00
|
|
|
|
2021-03-01 22:22:39 +00:00
|
|
|
if ($count)
|
2021-02-24 11:52:01 +00:00
|
|
|
{
|
2021-03-01 22:22:39 +00:00
|
|
|
// To avoid false-positives (e.g. %10s for a ten-characters-wide output string in a CLI),
|
|
|
|
// we check that placeholder numbers form a sequence from 1 to N.
|
|
|
|
|
|
|
|
$maxNumber = 0;
|
|
|
|
|
|
|
|
foreach ($matches as $match)
|
|
|
|
{
|
|
|
|
$maxNumber = max($maxNumber, (int) $match[1]);
|
|
|
|
}
|
|
|
|
|
|
|
|
// If placeholder numbers form a sequence, the maximal value is equal to the number of elements
|
|
|
|
if ($maxNumber === $count)
|
|
|
|
{
|
2023-08-13 09:22:27 +00:00
|
|
|
$this->report->addWarning($file, Text::_('COM_JEDCHECKER_LANG_INCORRECT_ARGNUM'), $startLineno, $line);
|
2021-03-01 22:22:39 +00:00
|
|
|
}
|
2021-02-23 20:41:30 +00:00
|
|
|
}
|
2021-03-09 20:48:46 +00:00
|
|
|
|
|
|
|
// Some extra checks for en-GB only (to don't duplicate false-positives)
|
|
|
|
if ($tag === 'en-GB')
|
|
|
|
{
|
|
|
|
// Check spaces around (but allow trailing space after colon)
|
|
|
|
if (preg_match('/^\s|[^:]\s+$/', $value))
|
|
|
|
{
|
2023-08-13 09:22:27 +00:00
|
|
|
$this->report->addNotice($file, Text::_('COM_JEDCHECKER_LANG_SPACES_AROUND'), $startLineno, $line);
|
2021-03-09 20:48:46 +00:00
|
|
|
}
|
|
|
|
}
|
2021-02-22 21:00:06 +00:00
|
|
|
}
|
2021-02-02 22:14:16 +00:00
|
|
|
|
2021-02-22 21:00:06 +00:00
|
|
|
// All checks passed. Return true
|
|
|
|
return true;
|
|
|
|
}
|
2021-05-09 15:39:07 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Appends keys from INI file to the list
|
|
|
|
*
|
2021-05-11 20:52:40 +00:00
|
|
|
* @param string $file Language INI-file name
|
2021-05-09 15:39:07 +00:00
|
|
|
*
|
2021-05-11 20:52:40 +00:00
|
|
|
* @return void
|
2021-05-09 15:39:07 +00:00
|
|
|
*/
|
|
|
|
protected function populateLangKeys($file)
|
|
|
|
{
|
|
|
|
if (is_file($file))
|
|
|
|
{
|
2023-07-26 14:28:05 +00:00
|
|
|
$data = @parse_ini_file($file);
|
2021-05-09 15:39:07 +00:00
|
|
|
|
|
|
|
if (is_array($data))
|
|
|
|
{
|
|
|
|
$this->langKeys = array_replace($this->langKeys, $data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Reads PHP files and checks JText arguments
|
|
|
|
*
|
|
|
|
* @param string $file - The path to the file
|
|
|
|
*
|
|
|
|
* @return boolean True on success, otherwise False.
|
|
|
|
*/
|
|
|
|
protected function findJText($file)
|
|
|
|
{
|
|
|
|
$content = file_get_contents($file);
|
|
|
|
|
|
|
|
// Search for Text/JText calls
|
|
|
|
if (!preg_match_all('/\bJ?Text::(?:_|s?printf|alt|plural|script)\s*\(\s*([\'])([^\'"]+)\1\s*[\),]/', $content, $matches, PREG_OFFSET_CAPTURE))
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
$lines = explode("\n", $content);
|
|
|
|
|
|
|
|
// Check all keys exist in INI files
|
|
|
|
foreach ($matches[2] as $match)
|
|
|
|
{
|
2021-05-10 19:01:37 +00:00
|
|
|
$key = strtoupper($match[0]);
|
2021-05-09 15:39:07 +00:00
|
|
|
|
|
|
|
if (!isset($this->langKeys[$key]))
|
|
|
|
{
|
|
|
|
$lineno = substr_count($content, "\n", 0, $match[1]);
|
2023-08-13 09:22:27 +00:00
|
|
|
$this->report->addNotice($file, Text::sprintf('COM_JEDCHECKER_LANG_UNKNOWN_KEY_IN_CODE', htmlspecialchars($key)), $lineno + 1, $lines[$lineno]);
|
2021-05-09 15:39:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
2021-02-02 22:14:16 +00:00
|
|
|
}
|