mirror of
https://github.com/Combodo/iTop.git
synced 2026-02-24 21:04:14 +01:00
# Conflicts: # tests/php-unit-tests/integration-tests/DictionariesConsistencyTest.php # tests/php-unit-tests/src/BaseTestCase/ItopTestCase.php # tests/php-unit-tests/unitary-tests/core/CMDBSource/TransactionsTest.php # tests/php-unit-tests/unitary-tests/sources/Application/TwigBase/Twig/TwigTest.php
193 lines
6.6 KiB
PHP
193 lines
6.6 KiB
PHP
<?php
|
|
/**
|
|
* Copyright (C) 2013-2023 Combodo SARL
|
|
* This file is part of iTop.
|
|
* iTop is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU Affero General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
* iTop is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU Affero General Public License for more details.
|
|
* You should have received a copy of the GNU Affero General Public License
|
|
*/
|
|
|
|
namespace Combodo\iTop\Test\UnitTest\Integration;
|
|
|
|
use Combodo\iTop\Test\UnitTest\ItopTestCase;
|
|
|
|
|
|
|
|
/**
|
|
* Wrapper to load dictionnary files without altering the main dictionnary
|
|
* Eval will be called within the current namespace (this is done by adding a "namespace" statement)
|
|
*/
|
|
class Dict
|
|
{
|
|
public static function Add($sLanguageCode, $sEnglishLanguageDesc, $sLocalizedLanguageDesc, $aEntries)
|
|
{
|
|
}
|
|
}
|
|
/**
|
|
* For tests on compiled dict files, see {@see CompiledDictionariesConsistencyTest}
|
|
* @group beforeSetup
|
|
*/
|
|
class DictionariesConsistencyTest extends ItopTestCase
|
|
{
|
|
/**
|
|
* Verify that language declarations match the file names (same language codes)
|
|
*
|
|
* @dataProvider DictionaryFileProvider
|
|
*
|
|
* @param $sDictFile
|
|
*/
|
|
public function testDictionariesLanguage($sDictFile): void
|
|
{
|
|
// In iTop the language available list is dynamically made during setup, depending on the dict files found
|
|
// Here we are using a fixed list
|
|
$aPrefixToLanguageData = array(
|
|
'cs' => array('CS CZ', 'Czech', 'Čeština'),
|
|
'da' => array('DA DA', 'Danish', 'Dansk'),
|
|
'de' => array('DE DE', 'German', 'Deutsch'),
|
|
'en' => array('EN US', 'English', 'English'),
|
|
'es_cr' => array('ES CR', 'Spanish', array(
|
|
'Español, Castellaño', // old value
|
|
'Español, Castellano', // new value since N°3635
|
|
)),
|
|
'fr' => array('FR FR', 'French', 'Français'),
|
|
'hu' => array('HU HU', 'Hungarian', 'Magyar'),
|
|
'it' => array('IT IT', 'Italian', 'Italiano'),
|
|
'ja' => array('JA JP', 'Japanese', '日本語'),
|
|
'nl' => array('NL NL', 'Dutch', 'Nederlands'),
|
|
'pl' => array('PL PL', 'Polish', 'Polski'),
|
|
'pt_br' => array('PT BR', 'Brazilian', 'Brazilian'),
|
|
'ru' => array('RU RU', 'Russian', 'Русский'),
|
|
'sk' => array('SK SK', 'Slovak', 'Slovenčina'),
|
|
'tr' => array('TR TR', 'Turkish', 'Türkçe'),
|
|
'zh_cn' => array('ZH CN', 'Chinese', '简体中文'),
|
|
);
|
|
|
|
if (!preg_match('/^(.*)\\.dict/', basename($sDictFile), $aMatches)) {
|
|
static::fail("Dictionary file '$sDictFile' not matching the naming convention");
|
|
}
|
|
|
|
$sLangPrefix = $aMatches[1];
|
|
if (!array_key_exists($sLangPrefix, $aPrefixToLanguageData)) {
|
|
static::fail("Unknown prefix '$sLangPrefix' for dictionary file '$sDictFile'");
|
|
}
|
|
|
|
[$sExpectedLanguageCode, $sExpectedEnglishLanguageDesc, $aExpectedLocalizedLanguageDesc] = $aPrefixToLanguageData[$sLangPrefix];
|
|
|
|
$sDictPHP = file_get_contents($sDictFile);
|
|
$iCount = preg_match_all("@Dict::Add\('(.*)'\s*,\s*'(.*)'\s*,\s*'(.*)'@", $sDictPHP, $aMatches);
|
|
if ($iCount === false) {
|
|
static::fail("Pattern not working");
|
|
}
|
|
if ($iCount === 0) {
|
|
// Empty dictionary, that's fine!
|
|
static::assertTrue(true);
|
|
}
|
|
foreach ($aMatches[1] as $sLanguageCode) {
|
|
static::assertSame($sExpectedLanguageCode, $sLanguageCode,
|
|
"Unexpected language code for Dict::Add in dictionary $sDictFile");
|
|
}
|
|
foreach ($aMatches[2] as $sEnglishLanguageDesc) {
|
|
static::assertSame($sExpectedEnglishLanguageDesc, $sEnglishLanguageDesc,
|
|
"Unexpected language description (english) for Dict::Add in dictionary $sDictFile");
|
|
}
|
|
foreach ($aMatches[3] as $sLocalizedLanguageDesc)
|
|
{
|
|
if (false === is_array($aExpectedLocalizedLanguageDesc)) {
|
|
$aExpectedLocalizedLanguageDesc = array($aExpectedLocalizedLanguageDesc);
|
|
}
|
|
static::assertContains($sLocalizedLanguageDesc,$aExpectedLocalizedLanguageDesc,
|
|
"Unexpected language description for Dict::Add in dictionary $sDictFile");
|
|
}
|
|
}
|
|
|
|
public function DictionaryFileProvider(): array
|
|
{
|
|
$this->setUp();
|
|
|
|
$sAppRoot = $this->GetAppRoot();
|
|
|
|
$aDictFiles = array_merge(
|
|
glob($sAppRoot.'datamodels/2.x/*/*.dict*.php'), // legacy form in modules
|
|
glob($sAppRoot.'datamodels/2.x/*/dictionaries/*.dict*.php'), // modern form in modules
|
|
glob($sAppRoot.'dictionaries/*.dict*.php') // framework
|
|
);
|
|
$aTestCases = array();
|
|
foreach ($aDictFiles as $sDictFile) {
|
|
$aTestCases[$sDictFile] = array('sDictFile' => $sDictFile);
|
|
}
|
|
|
|
return $aTestCases;
|
|
}
|
|
|
|
/**
|
|
* @dataProvider DictionaryFileProvider
|
|
*
|
|
* @param string $sDictFile
|
|
*
|
|
*
|
|
* @uses CheckDictionarySyntax
|
|
*/
|
|
public function testStandardDictionariesPhpSyntax(string $sDictFile): void
|
|
{
|
|
$this->CheckDictionarySyntax($sDictFile);
|
|
}
|
|
|
|
/**
|
|
* Checks that {@see CheckDictionarySyntax} works as expected by passing 2 test dictionaries
|
|
*
|
|
* @uses CheckDictionarySyntax
|
|
*/
|
|
public function testPlaygroundDictionariesPhpSyntax(): void
|
|
{
|
|
$this->CheckDictionarySyntax(__DIR__.'/dictionaries-test/fr.dictionary.itop.core.KO.wrong_php', false);
|
|
/** @noinspection PhpRedundantOptionalArgumentInspection */
|
|
$this->CheckDictionarySyntax(__DIR__.'/dictionaries-test/fr.dictionary.itop.core.OK.php', true);
|
|
}
|
|
|
|
/**
|
|
* @param string $sDictFile complete path for the file to check
|
|
* @param bool $bIsSyntaxValid expected assert value
|
|
*/
|
|
private function CheckDictionarySyntax(string $sDictFile, $bIsSyntaxValid = true): void
|
|
{
|
|
$sPHP = file_get_contents($sDictFile);
|
|
// Strip php tag to allow "eval"
|
|
$sPHP = substr(trim($sPHP), strlen('<?php'));
|
|
// Make sure the Dict class is the one declared in the current file
|
|
$sPHP = 'namespace '.__NAMESPACE__.";\n".$sPHP;
|
|
$iLineShift = 1; // Cope with the shift due to the namespace statement
|
|
$sPHP = str_replace(
|
|
['ITOP_APPLICATION_SHORT', 'ITOP_APPLICATION', 'ITOP_VERSION_NAME'],
|
|
['\'itop\'', '\'itop\'', '\'1.2.3\''],
|
|
$sPHP
|
|
);
|
|
try {
|
|
eval($sPHP);
|
|
// Reaching this point => No syntax error
|
|
if (!$bIsSyntaxValid) {
|
|
$this->fail("Failed to detect syntax error in dictionary `{$sDictFile}` (which is known as being INCORRECT)");
|
|
}
|
|
}
|
|
catch (\Error $e) {
|
|
if ($bIsSyntaxValid) {
|
|
$iLine = $e->getLine() - $iLineShift;
|
|
$this->fail("Invalid dictionary: {$e->getMessage()} in {$sDictFile}:{$iLine}");
|
|
}
|
|
}
|
|
catch (\Exception $e) {
|
|
if ($bIsSyntaxValid) {
|
|
$iLine = $e->getLine() - $iLineShift;
|
|
$sExceptionClass = get_class($e);
|
|
$this->fail("Exception thrown from dictionary: '$sExceptionClass: {$e->getMessage()}' in {$sDictFile}:{$iLine}");
|
|
}
|
|
}
|
|
$this->assertTrue(true);
|
|
}
|
|
}
|