33
2
mirror of https://github.com/joomla-extensions/jedchecker.git synced 2025-01-25 22:58:25 +00:00

350 lines
7.2 KiB
PHP
Raw Normal View History

2021-02-02 18:58:29 +03:00
<?php
/**
* @package Joomla.JEDChecker
*
2021-03-11 15:56:27 +03:00
* @copyright Copyright (C) 2021 Open Source Matters, Inc. All rights reserved.
2021-02-02 18:58:29 +03:00
*
* @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 JedcheckerRulesXMLManifest
*
2021-04-04 13:59:50 +03:00
* This class validates all XML manifests
2021-02-02 18:58:29 +03:00
*
2021-02-02 19:10:59 +03:00
* @since 2.3
2021-02-02 18:58:29 +03:00
*/
class JedcheckerRulesXMLManifest extends JEDcheckerRule
{
/**
* The formal ID of this rule. For example: SE1.
*
* @var string
*/
protected $id = 'MANIFEST';
/**
* The title or caption of this rule.
*
* @var string
*/
protected $title = 'COM_JEDCHECKER_MANIFEST';
/**
* The description of this rule.
*
* @var string
*/
protected $description = 'COM_JEDCHECKER_MANIFEST_DESC';
/**
* List of errors.
*
* @var string[]
*/
protected $errors;
/**
* List of warnings.
*
* @var string[]
*/
protected $warnings;
/**
* List of infos.
*
* @var string[]
*/
protected $infos;
2021-02-02 18:58:29 +03:00
/**
* Rules for XML nodes
* ? - single, optional
* = - single, required, warning if missed
* ! - single, required, error if missed
* * - multiple, optional
* @var array
*/
protected $DTDNodeRules;
/**
* Rules for attributes
* (list of allowed attributes)
* @var array
*/
protected $DTDAttrRules;
2021-04-04 13:59:50 +03:00
/**
* List of extension types
*
* @var string[]
*/
protected $joomlaTypes = array(
2021-04-04 13:59:50 +03:00
'component', 'file', 'language', 'library',
'module', 'package', 'plugin', 'template'
2021-02-02 18:58:29 +03:00
);
/**
* List of JED extension types
*
* @var string[]
*/
protected $jedTypes = array(
'component', 'module', 'package', 'plugin'
);
2021-02-02 18:58:29 +03:00
/**
* Initiates the search and check
*
* @return void
*/
public function check()
{
// Find all XML files of the extension
2021-02-13 23:53:36 +03:00
$files = JFolder::files($this->basedir, '\.xml$', true, true);
2021-02-02 18:58:29 +03:00
// Iterate through all the xml files
foreach ($files as $file)
{
// Try to check the file
$this->find($file);
}
}
/**
* Reads a file and validate XML manifest
*
* @param string $file - The path to the file
*
* @return boolean True if the manifest file was found, otherwise False.
*/
protected function find($file)
{
2021-02-13 23:53:09 +03:00
$xml = simplexml_load_file($file);
2021-02-02 18:58:29 +03:00
// Failed to parse the xml file.
// Assume that this is not a extension manifest
if (!$xml)
{
return false;
}
// Check if this is an extension manifest
if ($xml->getName() !== 'extension')
{
return false;
}
2021-04-04 13:59:50 +03:00
// Check extension type
2021-02-02 18:58:29 +03:00
$type = (string) $xml['type'];
2021-04-04 13:59:50 +03:00
if (!in_array($type, $this->joomlaTypes, true))
2021-02-02 18:58:29 +03:00
{
$this->report->addError($file, JText::sprintf('COM_JEDCHECKER_MANIFEST_UNKNOWN_TYPE', $type));
2021-04-04 13:59:50 +03:00
2021-02-02 18:58:29 +03:00
return true;
}
// JED allows components, modules, plugins, and packages (as a container) only
if (!in_array($type, $this->jedTypes, true))
{
$this->report->addError($file, JText::sprintf('COM_JEDCHECKER_MANIFEST_TYPE_NOT_ACCEPTED', $type));
}
2021-04-04 13:59:50 +03:00
// Load DTD-like data for this extension type
2021-02-24 00:27:50 +03:00
$json_filename = __DIR__ . '/xmlmanifest/dtd_' . $type . '.json';
2021-04-04 13:59:50 +03:00
2021-02-02 18:58:29 +03:00
if (!is_file($json_filename))
{
return true;
}
2021-04-04 13:59:50 +03:00
2021-02-02 18:58:29 +03:00
$data = json_decode(file_get_contents($json_filename), true);
$this->DTDNodeRules = $data['nodes'];
$this->DTDAttrRules = $data['attributes'];
$this->errors = array();
$this->warnings = array();
$this->infos = array();
2021-02-02 18:58:29 +03:00
2021-04-04 13:59:50 +03:00
// Validate manifest
2021-02-02 18:58:29 +03:00
$this->validateXml($xml, 'extension');
if (count($this->errors))
{
$this->report->addError($file, implode('<br />', $this->errors));
}
if (count($this->warnings))
{
$this->report->addWarning($file, implode('<br />', $this->warnings));
}
if (count($this->infos))
{
$this->report->addInfo($file, implode('<br />', $this->infos));
}
2021-02-02 18:58:29 +03:00
// All checks passed. Return true
return true;
}
/**
2021-04-04 13:59:50 +03:00
* @param SimpleXMLElement $node XML node object
* @param string $name XML node name
*
* @return void
2021-02-02 18:58:29 +03:00
*/
protected function validateXml($node, $name)
{
// Check attributes
$DTDattributes = isset($this->DTDAttrRules[$name]) ? $this->DTDAttrRules[$name] : array();
if (isset($DTDattributes[0]) && $DTDattributes[0] !== '*')
2021-02-02 18:58:29 +03:00
{
foreach ($node->attributes() as $attr)
2021-02-02 18:58:29 +03:00
{
$attrName = (string) $attr->getName();
2021-04-04 13:59:50 +03:00
if (!in_array($attrName, $DTDattributes, true))
{
2021-03-11 01:15:13 +03:00
// The node has unknown attribute
$this->infos[] = JText::sprintf('COM_JEDCHECKER_MANIFEST_UNKNOWN_ATTRIBUTE', $name, $attrName);
}
2021-02-02 18:58:29 +03:00
}
}
// Check children nodes
if (!isset($this->DTDNodeRules[$name]))
{
// No children
if ($node->count() > 0)
{
$this->infos[] = JText::sprintf('COM_JEDCHECKER_MANIFEST_UNKNOWN_CHILDREN', $name);
2021-02-02 18:58:29 +03:00
}
}
elseif (!isset($this->DTDNodeRules[$name]['*']))
{
2021-02-02 18:58:29 +03:00
$DTDchildren = $this->DTDNodeRules[$name];
// 1) check required single elements
foreach ($DTDchildren as $child => $mode)
{
$count = $node->$child->count();
2021-04-04 13:59:50 +03:00
2021-02-02 18:58:29 +03:00
switch ($mode)
{
case '!':
$errors =& $this->errors;
break;
case '=':
$errors =& $this->warnings;
break;
default:
continue 2;
}
2021-04-04 13:59:50 +03:00
2021-02-02 18:58:29 +03:00
if ($count === 0)
{
$errors[] = JText::sprintf('COM_JEDCHECKER_MANIFEST_MISSED_REQUIRED', $name, $child);
}
elseif ($count > 1)
{
$errors[] = JText::sprintf('COM_JEDCHECKER_MANIFEST_MULTIPLE_FOUND', $name, $child);
}
2021-04-04 13:59:50 +03:00
2021-02-02 18:58:29 +03:00
unset($errors);
}
// 2) check unknown/multiple elements
2021-04-04 13:59:50 +03:00
// Collect unique child node names
$childNames = array();
2021-02-02 18:58:29 +03:00
foreach ($node as $child)
{
2021-04-04 13:59:50 +03:00
$childNames[$child->getName()] = 1;
2021-02-02 18:58:29 +03:00
}
2021-04-04 13:59:50 +03:00
$childNames = array_keys($childNames);
2021-02-02 18:58:29 +03:00
2021-04-04 13:59:50 +03:00
foreach ($childNames as $child)
2021-02-02 18:58:29 +03:00
{
if (!isset($DTDchildren[$child]))
{
$this->infos[] = JText::sprintf('COM_JEDCHECKER_MANIFEST_UNKNOWN_CHILD', $name, $child);
2021-02-02 18:58:29 +03:00
}
else
{
if ($DTDchildren[$child] === '?' && $node->$child->count() > 1)
{
$this->errors[] = JText::sprintf('COM_JEDCHECKER_MANIFEST_MULTIPLE_FOUND', $name, $child);
}
}
}
2021-02-24 14:57:11 +03:00
// 3) check empty elements
foreach ($node as $child)
{
if ($child->count() === 0 && $child->attributes()->count() === 0 && (string) $child === '')
2021-02-24 14:57:11 +03:00
{
$this->infos[] = JText::sprintf('COM_JEDCHECKER_MANIFEST_EMPTY_CHILD', $child->getName());
}
}
2021-02-02 18:58:29 +03:00
}
// Extra checks (if exist)
$method = 'validateXml' . $name;
2021-04-04 13:59:50 +03:00
2021-02-02 18:58:29 +03:00
if (method_exists($this, $method))
{
$this->$method($node);
}
// Recursion
foreach ($node as $child)
{
2021-04-04 13:59:50 +03:00
$childName = $child->getName();
if (isset($this->DTDNodeRules[$childName]))
{
$this->validateXml($child, $childName);
2021-02-02 18:58:29 +03:00
}
}
}
/**
* Extra check for menu nodes
2021-04-04 13:59:50 +03:00
* @param SimpleXMLElement $node XML node
*
* @return void
2021-02-02 18:58:29 +03:00
*/
protected function validateXmlMenu($node)
{
if (isset($node['link']))
{
2021-03-11 01:15:13 +03:00
// The "link" attribute overrides any other link-related attributes (warn if they present)
2021-04-04 13:59:50 +03:00
$skipAttrs = array('act', 'controller', 'layout', 'sub', 'task', 'view');
2021-02-02 18:58:29 +03:00
foreach ($node->attributes() as $attr)
{
2021-04-04 13:59:50 +03:00
$attrName = $attr->getName();
if (in_array($attrName, $skipAttrs, true))
2021-02-02 18:58:29 +03:00
{
2021-04-04 13:59:50 +03:00
$this->warnings[] = JText::sprintf('COM_JEDCHECKER_MANIFEST_MENU_UNUSED_ATTRIBUTE', $attrName);
2021-02-02 18:58:29 +03:00
}
}
}
}
}