mirror of
https://github.com/joomla-extensions/jedchecker.git
synced 2025-01-01 06:01:46 +00:00
Merge PR #83 with develop
This commit is contained in:
commit
3d9fc254a0
@ -91,6 +91,7 @@ COM_JEDCHECKER_DELETE_FAILED="Can't delete temporary folder"
|
||||
COM_JEDCHECKER_DELETE_SUCCESS="Temporary folder deleted!"
|
||||
COM_JEDCHECKER_EMPTY_UPLOAD_FIELD="Please, select a zipped file to be uploaded"
|
||||
<<<<<<< HEAD
|
||||
<<<<<<< HEAD
|
||||
COM_JEDCHECKER_TOOLBAR_CHECK="Check"
|
||||
COM_JEDCHECKER_TOOLBAR_CLEAR="Clear"
|
||||
COM_JEDCHECKER_CLICK_TO_VIEW_DETAILS="Click to View Details"
|
||||
@ -113,3 +114,30 @@ COM_JEDCHECKER_XML_FILES="XML Files references"
|
||||
COM_JEDCHECKER_XML_FILES_DESC="Check for incorrect files and folders references in the XML manifest"
|
||||
COM_JEDCHECKER_XML_FILES_FILE_NOT_FOUND="File not found: %s"
|
||||
COM_JEDCHECKER_XML_FILES_FOLDER_NOT_FOUND="Folder not found: %s"
|
||||
COM_JEDCHECKER_LANG="Language files"
|
||||
COM_JEDCHECKER_LANG_DESC="Validates language files"
|
||||
COM_JEDCHECKER_LANG_INCORRECT_EOL="Incorrect end-of-line character found. Convert file to Unix EOL (\n) format."
|
||||
COM_JEDCHECKER_LANG_BOM_FOUND="The byte order mark (BOM) is detected"
|
||||
COM_JEDCHECKER_LANG_INCORRECT_COMMENT="Incorrect comment character, use ';' instead"
|
||||
COM_JEDCHECKER_LANG_WRONG_LINE="Incorrect line without '=' character"
|
||||
COM_JEDCHECKER_LANG_KEY_EMPTY="Empty key name"
|
||||
COM_JEDCHECKER_LANG_KEY_WHITESPACE="Whitespace in the key is not allowed"
|
||||
COM_JEDCHECKER_LANG_KEY_INVALID_CHARACTER="Invalid character in the key name"
|
||||
COM_JEDCHECKER_LANG_KEY_RESERVED="Reserved keyword in the key name"
|
||||
COM_JEDCHECKER_LANG_KEY_NOT_ASCII="The key name is not ASCII-compatible"
|
||||
COM_JEDCHECKER_LANG_KEY_NOT_UPPERCASE="The key name is not uppercase"
|
||||
COM_JEDCHECKER_LANG_KEY_DUPLICATED="The key name was declared previously on line %d"
|
||||
COM_JEDCHECKER_LANG_TRANSLATION_ERROR="Invalid translation string"
|
||||
COM_JEDCHECKER_LANG_INVALID_UTF8="Invalid UTF8 string"
|
||||
COM_JEDCHECKER_LANG_TRANSLATION_QUOTES="All translation strings should be enclosed in double quotation marks."
|
||||
COM_JEDCHECKER_LANG_TRANSLATION_MISSED_LEFT_QUOTE="You have missed the left quote."
|
||||
COM_JEDCHECKER_LANG_TRANSLATION_MISSED_RIGTH_QUOTE="You have missed the right quote"
|
||||
COM_JEDCHECKER_LANG_TRANSLATION_EMPTY="Empty translation string"
|
||||
COM_JEDCHECKER_LANG_QQ_DEPRECATED="Usage of \"_QQ_\" is deprecated since Joomla! 3.9. Use escaped double quotes (\\\") instead"
|
||||
COM_JEDCHECKER_LANG_UNESCAPED_QUOTE="Unescaped double quotation mark found"
|
||||
COM_JEDCHECKER_LANG_VARIABLE_REF="A variable reference found in the translation string."
|
||||
COM_JEDCHECKER_LANG_INCORRECT_ARGNUM="Probably you use incorrect placeholder format (e.g. '%1s' instead of '%1$s'), see <a href='https://www.php.net/manual/en/function.sprintf.php'>printf's argnum format specification</a> for details"
|
||||
COM_JEDCHECKER_LANG_SPACES_AROUND="Spaces around the translation string are detected"
|
||||
COM_JEDCHECKER_TOOLBAR_CHECK="Check"
|
||||
COM_JEDCHECKER_TOOLBAR_CLEAR="Clear"
|
||||
COM_JEDCHECKER_CLICK_TO_VIEW_DETAILS="Click to View Details"
|
||||
|
@ -0,0 +1,331 @@
|
||||
<?php
|
||||
/**
|
||||
* @package Joomla.JEDChecker
|
||||
*
|
||||
* @copyright Copyright (C) 2021 Open Source Matters, Inc. All rights reserved.
|
||||
*
|
||||
* @license GNU General Public License version 2 or later; see LICENSE.txt
|
||||
*/
|
||||
|
||||
defined('_JEXEC') or die('Restricted access');
|
||||
|
||||
|
||||
// Include the rule base class
|
||||
require_once JPATH_COMPONENT_ADMINISTRATOR . '/models/rule.php';
|
||||
|
||||
|
||||
/**
|
||||
* class JedcheckerRulesLanguage
|
||||
*
|
||||
* This class validates language ini file
|
||||
*
|
||||
* @since 3.0
|
||||
*/
|
||||
class JedcheckerRulesLanguage extends JEDcheckerRule
|
||||
{
|
||||
/**
|
||||
* The formal ID of this rule. For example: SE1.
|
||||
*
|
||||
* @var string
|
||||
*/
|
||||
protected $id = 'LANG';
|
||||
|
||||
/**
|
||||
* The title or caption of this rule.
|
||||
*
|
||||
* @var string
|
||||
*/
|
||||
protected $title = 'COM_JEDCHECKER_LANG';
|
||||
|
||||
/**
|
||||
* The description of this rule.
|
||||
*
|
||||
* @var string
|
||||
*/
|
||||
protected $description = 'COM_JEDCHECKER_LANG_DESC';
|
||||
|
||||
/**
|
||||
* Initiates the search and check
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function check()
|
||||
{
|
||||
// Find all INI files of the extension
|
||||
$files = JFolder::files($this->basedir, '\.ini$', true, true);
|
||||
|
||||
// Iterate through all the ini files
|
||||
foreach ($files as $file)
|
||||
{
|
||||
/* 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) */
|
||||
if (preg_match('#(?:^|/)([a-z]{2,3}-[A-Z]{2})[./]\w+(?:\.sys)?\.ini$#', $file, $match))
|
||||
{
|
||||
// Try to validate the file
|
||||
$this->find($file, $match[1]);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Reads and validates an ini file
|
||||
*
|
||||
* @param string $file - The path to the file
|
||||
* @param string $tag - Language tag code
|
||||
*
|
||||
* @return boolean True on success, otherwise False.
|
||||
*/
|
||||
protected function find($file, $tag)
|
||||
{
|
||||
$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)
|
||||
{
|
||||
$this->report->addWarning($file, JText::_('COM_JEDCHECKER_LANG_INCORRECT_EOL'));
|
||||
}
|
||||
|
||||
$lines = file($file);
|
||||
|
||||
if ($lines === false)
|
||||
{
|
||||
return false;
|
||||
}
|
||||
|
||||
$nLines = count($lines);
|
||||
$keys = array();
|
||||
|
||||
// Use mb_check_encoding (if exists) to validate UTF-8
|
||||
$mbExists = function_exists('mb_check_encoding');
|
||||
|
||||
for ($lineno = 0; $lineno < $nLines; $lineno++)
|
||||
{
|
||||
$startLineno = $lineno + 1;
|
||||
$line = trim($lines[$lineno]);
|
||||
|
||||
// Check for BOM sequence
|
||||
if ($lineno === 0 && strncmp($line, "\xEF\xBB\xBF", 3) === 0)
|
||||
{
|
||||
// Report as an error if BOM is directly followed by key name (and become a part of the name)
|
||||
if (isset($line[3]) && strpos(";\n\r", $line[3]) === false)
|
||||
{
|
||||
$this->report->addError($file, JText::_('COM_JEDCHECKER_LANG_BOM_FOUND'), $startLineno);
|
||||
}
|
||||
else // Otherwise report a warning
|
||||
{
|
||||
$this->report->addWarning($file, JText::_('COM_JEDCHECKER_LANG_BOM_FOUND'), $startLineno);
|
||||
}
|
||||
|
||||
// Romeve BOM for further checks
|
||||
$line = substr($line, 3);
|
||||
}
|
||||
|
||||
// Skip empty lines, comments, and section names
|
||||
if ($line === '' || $line[0] === ';' || $line[0] === '[')
|
||||
{
|
||||
continue;
|
||||
}
|
||||
|
||||
// Report incorrect comment character
|
||||
if ($line[0] === '#')
|
||||
{
|
||||
$this->report->addError($file, JText::_('COM_JEDCHECKER_LANG_INCORRECT_COMMENT'), $startLineno, $line);
|
||||
continue;
|
||||
}
|
||||
|
||||
// Check for "=" character in the line
|
||||
if (strpos($line, '=') === false)
|
||||
{
|
||||
$this->report->addError($file, JText::_('COM_JEDCHECKER_LANG_WRONG_LINE'), $startLineno, $line);
|
||||
continue;
|
||||
}
|
||||
|
||||
// Extract key and value
|
||||
list ($key, $value) = explode('=', $line, 2);
|
||||
|
||||
// Validate key
|
||||
$key = rtrim($key);
|
||||
|
||||
// Check for empty key
|
||||
if ($key === '')
|
||||
{
|
||||
$this->report->addError($file, JText::_('COM_JEDCHECKER_LANG_KEY_EMPTY'), $startLineno, $line);
|
||||
continue;
|
||||
}
|
||||
|
||||
// Check for spaces in the key name
|
||||
if (preg_match('/\s/', $key))
|
||||
{
|
||||
$this->report->addError($file, JText::_('COM_JEDCHECKER_LANG_KEY_WHITESPACE'), $startLineno, $line);
|
||||
continue;
|
||||
}
|
||||
|
||||
// Check for invalid characters (see https://www.php.net/manual/en/function.parse-ini-file.php)
|
||||
if (strpbrk($key, '{}|&~![()^"') !== false)
|
||||
{
|
||||
$this->report->addError($file, JText::_('COM_JEDCHECKER_LANG_KEY_INVALID_CHARACTER'), $startLineno, $line);
|
||||
continue;
|
||||
}
|
||||
|
||||
// Check for invalid key names (see https://www.php.net/manual/en/function.parse-ini-file.php)
|
||||
if (in_array($key, array('null', 'yes', 'no', 'true', 'false', 'on', 'off', 'none'), true))
|
||||
{
|
||||
$this->report->addError($file, JText::_('COM_JEDCHECKER_LANG_KEY_RESERVED'), $startLineno, $line);
|
||||
continue;
|
||||
}
|
||||
|
||||
// Check key contains ASCII characters only
|
||||
if (preg_match('/[\x00-\x1F\x80-\xFF]/', $key))
|
||||
{
|
||||
$this->report->addWarning($file, JText::_('COM_JEDCHECKER_LANG_KEY_NOT_ASCII'), $startLineno, $line);
|
||||
}
|
||||
|
||||
// Check key is uppercase
|
||||
if ($key !== strtoupper($key))
|
||||
{
|
||||
$this->report->addWarning($file, JText::_('COM_JEDCHECKER_LANG_KEY_NOT_UPPERCASE'), $startLineno, $line);
|
||||
}
|
||||
|
||||
// Check for duplicated keys
|
||||
if (isset($keys[$key]))
|
||||
{
|
||||
$this->report->addWarning($file, JText::sprintf('COM_JEDCHECKER_LANG_KEY_DUPLICATED', $keys[$key]), $startLineno, $line);
|
||||
}
|
||||
else
|
||||
{
|
||||
$keys[$key] = $startLineno;
|
||||
}
|
||||
|
||||
// Validate value
|
||||
$value = ltrim($value);
|
||||
|
||||
// Parse multiline values
|
||||
while (!preg_match('/^((?>\'(?>[^\'\\\\]+|\\\\.)*\'|"(?>[^"\\\\]+|\\\\.)*"|[^\'";]+)*)(;.*)?$/', $value, $matches))
|
||||
{
|
||||
if ($lineno + 1 >= $nLines)
|
||||
{
|
||||
break;
|
||||
}
|
||||
|
||||
$lineno++;
|
||||
$chunk = "\n" . trim($lines[$lineno]);
|
||||
$line .= $chunk;
|
||||
$value .= $chunk;
|
||||
}
|
||||
|
||||
// The value doesn't match INI format
|
||||
if (!isset($matches[0]))
|
||||
{
|
||||
$this->report->addWarning($file, JText::_('COM_JEDCHECKER_LANG_TRANSLATION_ERROR'), $startLineno, $line);
|
||||
continue;
|
||||
}
|
||||
|
||||
// Get value w/o comment
|
||||
$value = trim($matches[1]);
|
||||
|
||||
// Check for empty value
|
||||
if ($value === '""')
|
||||
{
|
||||
$this->report->addInfo($file, JText::_('COM_JEDCHECKER_LANG_TRANSLATION_EMPTY'), $startLineno, $line);
|
||||
continue;
|
||||
}
|
||||
|
||||
// Check it's a valid UTF-8 string
|
||||
$validUTF8 = $mbExists ? mb_check_encoding($value, 'UTF-8') : preg_match('//u', $value);
|
||||
|
||||
if (!$validUTF8)
|
||||
{
|
||||
$this->report->addWarning($file, JText::_('COM_JEDCHECKER_LANG_INVALID_UTF8'), $startLineno, $line);
|
||||
}
|
||||
|
||||
// Check for unquoted values
|
||||
if (strlen($value) < 2 || ($value[0] !== '"' && substr($value, -1) !== '"'))
|
||||
{
|
||||
$this->report->addWarning($file, JText::_('COM_JEDCHECKER_LANG_TRANSLATION_QUOTES'), $startLineno, $line);
|
||||
continue;
|
||||
}
|
||||
|
||||
if ($value[0] !== '"')
|
||||
{
|
||||
$msg = JText::_('COM_JEDCHECKER_LANG_TRANSLATION_QUOTES') . ' ' . JText::_('COM_JEDCHECKER_LANG_TRANSLATION_MISSED_LEFT_QUOTE');
|
||||
$this->report->addWarning($file, $msg, $startLineno, $line);
|
||||
continue;
|
||||
}
|
||||
|
||||
if (substr($value, -1) !== '"')
|
||||
{
|
||||
$msg = JText::_('COM_JEDCHECKER_LANG_TRANSLATION_QUOTES') . ' ' . JText::_('COM_JEDCHECKER_LANG_TRANSLATION_MISSED_RIGHT_QUOTE');
|
||||
$this->report->addWarning($file, $msg, $startLineno, $line);
|
||||
continue;
|
||||
}
|
||||
|
||||
// Remove quotes around
|
||||
$value = substr($value, 1, -1);
|
||||
|
||||
// Check for legacy "_QQ_" code (deprecated since Joomla! 3.9 in favour of escaped double quote \"; removed in Joomla! 4)
|
||||
if (strpos($value, '"_QQ_"') !== false)
|
||||
{
|
||||
$this->report->addCompat($file, JText::_('COM_JEDCHECKER_LANG_QQ_DEPRECATED'), $startLineno, $line);
|
||||
}
|
||||
|
||||
// Convert "_QQ_" to escaped quotes for further analysis
|
||||
$value = str_replace('"_QQ_"', '\"', $value);
|
||||
|
||||
// Check for unescaped quote
|
||||
if (preg_match('/[^\\\\]"/', $value))
|
||||
{
|
||||
$this->report->addWarning($file, JText::_('COM_JEDCHECKER_LANG_UNESCAPED_QUOTE'), $startLineno, $line);
|
||||
}
|
||||
|
||||
// Check for value interpolation (see https://www.php.net/manual/en/function.parse-ini-file.php for details)
|
||||
if (strpos($value, '${') !== false)
|
||||
{
|
||||
$this->report->addWarning($file, JText::_('COM_JEDCHECKER_LANG_VARIABLE_REF'), $startLineno, $line);
|
||||
}
|
||||
|
||||
// The code below detects incorrect format of numbered placeholders (e.g. "%1s" instead of "%1$s")
|
||||
|
||||
// Count numbered placeholders in the string (e.g. "%1s")
|
||||
$count = preg_match_all('/(?<=^|[^%])%(\d+)\w/', $value, $matches);
|
||||
|
||||
if ($count)
|
||||
{
|
||||
// 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)
|
||||
{
|
||||
$this->report->addWarning($file, JText::_('COM_JEDCHECKER_LANG_INCORRECT_ARGNUM'), $startLineno, $line);
|
||||
}
|
||||
}
|
||||
|
||||
// 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))
|
||||
{
|
||||
$this->report->addInfo($file, JText::_('COM_JEDCHECKER_LANG_SPACES_AROUND'), $startLineno, $line);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// All checks passed. Return true
|
||||
return true;
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue
Block a user