Merge branch '3.4' into 4.2
* 3.4: [Console] Fix command testing with missing inputs [Validator] Sync no/nb translation files [Translation] Added a script to display the status of translations [Validator] Added missing translations for Norwegian (\"no\") locale #30179 [Security\Guard] bump lowest version of security-core
This commit is contained in:
commit
890c2ac9f5
@ -126,7 +126,7 @@ class QuestionHelper extends Helper
|
|||||||
if (false === $ret) {
|
if (false === $ret) {
|
||||||
$ret = fgets($inputStream, 4096);
|
$ret = fgets($inputStream, 4096);
|
||||||
if (false === $ret) {
|
if (false === $ret) {
|
||||||
throw new RuntimeException('Aborted');
|
throw new RuntimeException('Aborted.');
|
||||||
}
|
}
|
||||||
$ret = trim($ret);
|
$ret = trim($ret);
|
||||||
}
|
}
|
||||||
@ -213,8 +213,10 @@ class QuestionHelper extends Helper
|
|||||||
while (!feof($inputStream)) {
|
while (!feof($inputStream)) {
|
||||||
$c = fread($inputStream, 1);
|
$c = fread($inputStream, 1);
|
||||||
|
|
||||||
// Backspace Character
|
// as opposed to fgets(), fread() returns an empty string when the stream content is empty, not false.
|
||||||
if ("\177" === $c) {
|
if (false === $c || ('' === $ret && '' === $c && null === $question->getDefault())) {
|
||||||
|
throw new RuntimeException('Aborted.');
|
||||||
|
} elseif ("\177" === $c) { // Backspace Character
|
||||||
if (0 === $numMatches && 0 !== $i) {
|
if (0 === $numMatches && 0 !== $i) {
|
||||||
--$i;
|
--$i;
|
||||||
// Move cursor backwards
|
// Move cursor backwards
|
||||||
@ -339,7 +341,7 @@ class QuestionHelper extends Helper
|
|||||||
shell_exec(sprintf('stty %s', $sttyMode));
|
shell_exec(sprintf('stty %s', $sttyMode));
|
||||||
|
|
||||||
if (false === $value) {
|
if (false === $value) {
|
||||||
throw new RuntimeException('Aborted');
|
throw new RuntimeException('Aborted.');
|
||||||
}
|
}
|
||||||
|
|
||||||
$value = trim($value);
|
$value = trim($value);
|
||||||
|
@ -60,9 +60,8 @@ class CommandTester
|
|||||||
}
|
}
|
||||||
|
|
||||||
$this->input = new ArrayInput($input);
|
$this->input = new ArrayInput($input);
|
||||||
if ($this->inputs) {
|
// Use an in-memory input stream even if no inputs are set so that QuestionHelper::ask() does not rely on the blocking STDIN.
|
||||||
$this->input->setStream(self::createStream($this->inputs));
|
$this->input->setStream(self::createStream($this->inputs));
|
||||||
}
|
|
||||||
|
|
||||||
if (isset($options['interactive'])) {
|
if (isset($options['interactive'])) {
|
||||||
$this->input->setInteractive($options['interactive']);
|
$this->input->setInteractive($options['interactive']);
|
||||||
|
@ -549,7 +549,7 @@ class QuestionHelperTest extends AbstractQuestionHelperTest
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* @expectedException \Symfony\Component\Console\Exception\RuntimeException
|
* @expectedException \Symfony\Component\Console\Exception\RuntimeException
|
||||||
* @expectedExceptionMessage Aborted
|
* @expectedExceptionMessage Aborted.
|
||||||
*/
|
*/
|
||||||
public function testAskThrowsExceptionOnMissingInput()
|
public function testAskThrowsExceptionOnMissingInput()
|
||||||
{
|
{
|
||||||
@ -559,7 +559,17 @@ class QuestionHelperTest extends AbstractQuestionHelperTest
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* @expectedException \Symfony\Component\Console\Exception\RuntimeException
|
* @expectedException \Symfony\Component\Console\Exception\RuntimeException
|
||||||
* @expectedExceptionMessage Aborted
|
* @expectedExceptionMessage Aborted.
|
||||||
|
*/
|
||||||
|
public function testAskThrowsExceptionOnMissingInputForChoiceQuestion()
|
||||||
|
{
|
||||||
|
$dialog = new QuestionHelper();
|
||||||
|
$dialog->ask($this->createStreamableInputInterfaceMock($this->getInputStream('')), $this->createOutputInterface(), new ChoiceQuestion('Choice', ['a', 'b']));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @expectedException \Symfony\Component\Console\Exception\RuntimeException
|
||||||
|
* @expectedExceptionMessage Aborted.
|
||||||
*/
|
*/
|
||||||
public function testAskThrowsExceptionOnMissingInputWithValidator()
|
public function testAskThrowsExceptionOnMissingInputWithValidator()
|
||||||
{
|
{
|
||||||
|
@ -124,7 +124,7 @@ class SymfonyQuestionHelperTest extends AbstractQuestionHelperTest
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* @expectedException \Symfony\Component\Console\Exception\RuntimeException
|
* @expectedException \Symfony\Component\Console\Exception\RuntimeException
|
||||||
* @expectedExceptionMessage Aborted
|
* @expectedExceptionMessage Aborted.
|
||||||
*/
|
*/
|
||||||
public function testAskThrowsExceptionOnMissingInput()
|
public function testAskThrowsExceptionOnMissingInput()
|
||||||
{
|
{
|
||||||
|
@ -17,6 +17,7 @@ use Symfony\Component\Console\Command\Command;
|
|||||||
use Symfony\Component\Console\Helper\HelperSet;
|
use Symfony\Component\Console\Helper\HelperSet;
|
||||||
use Symfony\Component\Console\Helper\QuestionHelper;
|
use Symfony\Component\Console\Helper\QuestionHelper;
|
||||||
use Symfony\Component\Console\Output\Output;
|
use Symfony\Component\Console\Output\Output;
|
||||||
|
use Symfony\Component\Console\Question\ChoiceQuestion;
|
||||||
use Symfony\Component\Console\Question\Question;
|
use Symfony\Component\Console\Question\Question;
|
||||||
use Symfony\Component\Console\Style\SymfonyStyle;
|
use Symfony\Component\Console\Style\SymfonyStyle;
|
||||||
use Symfony\Component\Console\Tester\CommandTester;
|
use Symfony\Component\Console\Tester\CommandTester;
|
||||||
@ -139,7 +140,7 @@ class CommandTesterTest extends TestCase
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* @expectedException \RuntimeException
|
* @expectedException \RuntimeException
|
||||||
* @expectedMessage Aborted
|
* @expectedExceptionMessage Aborted.
|
||||||
*/
|
*/
|
||||||
public function testCommandWithWrongInputsNumber()
|
public function testCommandWithWrongInputsNumber()
|
||||||
{
|
{
|
||||||
@ -153,13 +154,40 @@ class CommandTesterTest extends TestCase
|
|||||||
$command->setHelperSet(new HelperSet([new QuestionHelper()]));
|
$command->setHelperSet(new HelperSet([new QuestionHelper()]));
|
||||||
$command->setCode(function ($input, $output) use ($questions, $command) {
|
$command->setCode(function ($input, $output) use ($questions, $command) {
|
||||||
$helper = $command->getHelper('question');
|
$helper = $command->getHelper('question');
|
||||||
|
$helper->ask($input, $output, new ChoiceQuestion('choice', ['a', 'b']));
|
||||||
|
$helper->ask($input, $output, new Question($questions[0]));
|
||||||
|
$helper->ask($input, $output, new Question($questions[1]));
|
||||||
|
$helper->ask($input, $output, new Question($questions[2]));
|
||||||
|
});
|
||||||
|
|
||||||
|
$tester = new CommandTester($command);
|
||||||
|
$tester->setInputs(['a', 'Bobby', 'Fine']);
|
||||||
|
$tester->execute([]);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @expectedException \RuntimeException
|
||||||
|
* @expectedExceptionMessage Aborted.
|
||||||
|
*/
|
||||||
|
public function testCommandWithQuestionsButNoInputs()
|
||||||
|
{
|
||||||
|
$questions = [
|
||||||
|
'What\'s your name?',
|
||||||
|
'How are you?',
|
||||||
|
'Where do you come from?',
|
||||||
|
];
|
||||||
|
|
||||||
|
$command = new Command('foo');
|
||||||
|
$command->setHelperSet(new HelperSet([new QuestionHelper()]));
|
||||||
|
$command->setCode(function ($input, $output) use ($questions, $command) {
|
||||||
|
$helper = $command->getHelper('question');
|
||||||
|
$helper->ask($input, $output, new ChoiceQuestion('choice', ['a', 'b']));
|
||||||
$helper->ask($input, $output, new Question($questions[0]));
|
$helper->ask($input, $output, new Question($questions[0]));
|
||||||
$helper->ask($input, $output, new Question($questions[1]));
|
$helper->ask($input, $output, new Question($questions[1]));
|
||||||
$helper->ask($input, $output, new Question($questions[2]));
|
$helper->ask($input, $output, new Question($questions[2]));
|
||||||
});
|
});
|
||||||
|
|
||||||
$tester = new CommandTester($command);
|
$tester = new CommandTester($command);
|
||||||
$tester->setInputs(['Bobby', 'Fine']);
|
|
||||||
$tester->execute([]);
|
$tester->execute([]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -17,7 +17,7 @@
|
|||||||
],
|
],
|
||||||
"require": {
|
"require": {
|
||||||
"php": "^7.1.3",
|
"php": "^7.1.3",
|
||||||
"symfony/security-core": "~3.4|~4.0",
|
"symfony/security-core": "~3.4.22|^4.2.3",
|
||||||
"symfony/security-http": "~3.4|~4.0"
|
"symfony/security-http": "~3.4|~4.0"
|
||||||
},
|
},
|
||||||
"require-dev": {
|
"require-dev": {
|
||||||
|
@ -0,0 +1,195 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This file is part of the Symfony package.
|
||||||
|
*
|
||||||
|
* (c) Fabien Potencier <fabien@symfony.com>
|
||||||
|
*
|
||||||
|
* For the full copyright and license information, please view the LICENSE
|
||||||
|
* file that was distributed with this source code.
|
||||||
|
*/
|
||||||
|
|
||||||
|
$usageInstructions = <<<END
|
||||||
|
|
||||||
|
Usage instructions
|
||||||
|
-------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
$ cd symfony-code-root-directory/
|
||||||
|
|
||||||
|
# show the translation status of all locales
|
||||||
|
$ php translation-status.php
|
||||||
|
|
||||||
|
# show the translation status of all locales and all their missing translations
|
||||||
|
$ php translation-status.php -v
|
||||||
|
|
||||||
|
# show the status of a single locale
|
||||||
|
$ php translation-status.php fr
|
||||||
|
|
||||||
|
# show the status of a single locale and all its missing translations
|
||||||
|
$ php translation-status.php fr -v
|
||||||
|
|
||||||
|
END;
|
||||||
|
|
||||||
|
$config = [
|
||||||
|
// if TRUE, the full list of missing translations is displayed
|
||||||
|
'verbose_output' => false,
|
||||||
|
// NULL = analyze all locales
|
||||||
|
'locale_to_analyze' => null,
|
||||||
|
// the reference files all the other translations are compared to
|
||||||
|
'original_files' => [
|
||||||
|
'src/Symfony/Component/Form/Resources/translations/validators.en.xlf',
|
||||||
|
'src/Symfony/Component/Security/Core/Resources/translations/security.en.xlf',
|
||||||
|
'src/Symfony/Component/Validator/Resources/translations/validators.en.xlf',
|
||||||
|
],
|
||||||
|
];
|
||||||
|
|
||||||
|
$argc = $_SERVER['argc'];
|
||||||
|
$argv = $_SERVER['argv'];
|
||||||
|
|
||||||
|
if ($argc > 3) {
|
||||||
|
echo str_replace('translation-status.php', $argv[0], $usageInstructions);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach (array_slice($argv, 1) as $argumentOrOption) {
|
||||||
|
if (0 === strpos($argumentOrOption, '-')) {
|
||||||
|
$config['verbose_output'] = true;
|
||||||
|
} else {
|
||||||
|
$config['locale_to_analyze'] = $argumentOrOption;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($config['original_files'] as $originalFilePath) {
|
||||||
|
if (!file_exists($originalFilePath)) {
|
||||||
|
echo sprintf('The following file does not exist. Make sure that you execute this command at the root dir of the Symfony code repository.%s %s', PHP_EOL, $originalFilePath);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$totalMissingTranslations = 0;
|
||||||
|
|
||||||
|
foreach ($config['original_files'] as $originalFilePath) {
|
||||||
|
$translationFilePaths = findTranslationFiles($originalFilePath, $config['locale_to_analyze']);
|
||||||
|
$translationStatus = calculateTranslationStatus($originalFilePath, $translationFilePaths);
|
||||||
|
|
||||||
|
$totalMissingTranslations += array_sum(array_map(function ($translation) {
|
||||||
|
return \count($translation['missingKeys']);
|
||||||
|
}, array_values($translationStatus)));
|
||||||
|
|
||||||
|
printTranslationStatus($originalFilePath, $translationStatus, $config['verbose_output']);
|
||||||
|
}
|
||||||
|
|
||||||
|
exit($totalMissingTranslations > 0 ? 1 : 0);
|
||||||
|
|
||||||
|
function findTranslationFiles($originalFilePath, $localeToAnalyze)
|
||||||
|
{
|
||||||
|
$translations = [];
|
||||||
|
|
||||||
|
$translationsDir = dirname($originalFilePath);
|
||||||
|
$originalFileName = basename($originalFilePath);
|
||||||
|
$translationFileNamePattern = str_replace('.en.', '.*.', $originalFileName);
|
||||||
|
|
||||||
|
$translationFiles = glob($translationsDir.'/'.$translationFileNamePattern);
|
||||||
|
foreach ($translationFiles as $filePath) {
|
||||||
|
$locale = extractLocaleFromFilePath($filePath);
|
||||||
|
|
||||||
|
if (null !== $localeToAnalyze && $locale !== $localeToAnalyze) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$translations[$locale] = $filePath;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $translations;
|
||||||
|
}
|
||||||
|
|
||||||
|
function calculateTranslationStatus($originalFilePath, $translationFilePaths)
|
||||||
|
{
|
||||||
|
$translationStatus = [];
|
||||||
|
$allTranslationKeys = extractTranslationKeys($originalFilePath);
|
||||||
|
|
||||||
|
foreach ($translationFilePaths as $locale => $translationPath) {
|
||||||
|
$translatedKeys = extractTranslationKeys($translationPath);
|
||||||
|
$missingKeys = array_diff_key($allTranslationKeys, $translatedKeys);
|
||||||
|
|
||||||
|
$translationStatus[$locale] = [
|
||||||
|
'total' => \count($allTranslationKeys),
|
||||||
|
'translated' => \count($translatedKeys),
|
||||||
|
'missingKeys' => $missingKeys,
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
return $translationStatus;
|
||||||
|
}
|
||||||
|
|
||||||
|
function printTranslationStatus($originalFilePath, $translationStatus, $verboseOutput)
|
||||||
|
{
|
||||||
|
printTitle($originalFilePath);
|
||||||
|
printTable($translationStatus, $verboseOutput);
|
||||||
|
echo PHP_EOL.PHP_EOL;
|
||||||
|
}
|
||||||
|
|
||||||
|
function extractLocaleFromFilePath($filePath)
|
||||||
|
{
|
||||||
|
$parts = explode('.', $filePath);
|
||||||
|
|
||||||
|
return $parts[count($parts) - 2];
|
||||||
|
}
|
||||||
|
|
||||||
|
function extractTranslationKeys($filePath)
|
||||||
|
{
|
||||||
|
$translationKeys = [];
|
||||||
|
$contents = new \SimpleXMLElement(file_get_contents($filePath));
|
||||||
|
|
||||||
|
foreach ($contents->file->body->{'trans-unit'} as $translationKey) {
|
||||||
|
$translationId = (string) $translationKey['id'];
|
||||||
|
$translationKey = (string) $translationKey->source;
|
||||||
|
|
||||||
|
$translationKeys[$translationId] = $translationKey;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $translationKeys;
|
||||||
|
}
|
||||||
|
|
||||||
|
function printTitle($title)
|
||||||
|
{
|
||||||
|
echo $title.PHP_EOL;
|
||||||
|
echo str_repeat('=', strlen($title)).PHP_EOL.PHP_EOL;
|
||||||
|
}
|
||||||
|
|
||||||
|
function printTable($translations, $verboseOutput)
|
||||||
|
{
|
||||||
|
$longestLocaleNameLength = max(array_map('strlen', array_keys($translations)));
|
||||||
|
|
||||||
|
foreach ($translations as $locale => $translation) {
|
||||||
|
$isTranslationCompleted = $translation['translated'] === $translation['total'];
|
||||||
|
if ($isTranslationCompleted) {
|
||||||
|
textColorGreen();
|
||||||
|
}
|
||||||
|
|
||||||
|
echo sprintf('| Locale: %-'.$longestLocaleNameLength.'s | Translated: %d/%d', $locale, $translation['translated'], $translation['total']).PHP_EOL;
|
||||||
|
|
||||||
|
textColorNormal();
|
||||||
|
|
||||||
|
if (true === $verboseOutput && \count($translation['missingKeys']) > 0) {
|
||||||
|
echo str_repeat('-', 80).PHP_EOL;
|
||||||
|
echo '| Missing Translations:'.PHP_EOL;
|
||||||
|
|
||||||
|
foreach ($translation['missingKeys'] as $id => $content) {
|
||||||
|
echo sprintf('| (id=%s) %s', $id, $content).PHP_EOL;
|
||||||
|
}
|
||||||
|
|
||||||
|
echo str_repeat('-', 80).PHP_EOL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function textColorGreen()
|
||||||
|
{
|
||||||
|
echo "\033[32m";
|
||||||
|
}
|
||||||
|
|
||||||
|
function textColorNormal()
|
||||||
|
{
|
||||||
|
echo "\033[0m";
|
||||||
|
}
|
@ -314,6 +314,26 @@
|
|||||||
<source>This is not a valid Business Identifier Code (BIC).</source>
|
<source>This is not a valid Business Identifier Code (BIC).</source>
|
||||||
<target>Dette er ikke en gyldig BIC.</target>
|
<target>Dette er ikke en gyldig BIC.</target>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
|
<trans-unit id="82">
|
||||||
|
<source>Error</source>
|
||||||
|
<target>Feil</target>
|
||||||
|
</trans-unit>
|
||||||
|
<trans-unit id="83">
|
||||||
|
<source>This is not a valid UUID.</source>
|
||||||
|
<target>Dette er ikke en gyldig UUID.</target>
|
||||||
|
</trans-unit>
|
||||||
|
<trans-unit id="84">
|
||||||
|
<source>This value should be a multiple of {{ compared_value }}.</source>
|
||||||
|
<target>Verdien skal være flertall av {{ compared_value }}.</target>
|
||||||
|
</trans-unit>
|
||||||
|
<trans-unit id="85">
|
||||||
|
<source>This Business Identifier Code (BIC) is not associated with IBAN {{ iban }}.</source>
|
||||||
|
<target>Business Identifier Code (BIC) er ikke tilknyttet en IBAN {{ iban }}.</target>
|
||||||
|
</trans-unit>
|
||||||
|
<trans-unit id="86">
|
||||||
|
<source>This value should be valid JSON.</source>
|
||||||
|
<target>Verdien er ikke gyldig JSON.</target>
|
||||||
|
</trans-unit>
|
||||||
</body>
|
</body>
|
||||||
</file>
|
</file>
|
||||||
</xliff>
|
</xliff>
|
||||||
|
@ -314,6 +314,26 @@
|
|||||||
<source>This is not a valid Business Identifier Code (BIC).</source>
|
<source>This is not a valid Business Identifier Code (BIC).</source>
|
||||||
<target>Dette er ikke en gyldig BIC.</target>
|
<target>Dette er ikke en gyldig BIC.</target>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
|
<trans-unit id="82">
|
||||||
|
<source>Error</source>
|
||||||
|
<target>Feil</target>
|
||||||
|
</trans-unit>
|
||||||
|
<trans-unit id="83">
|
||||||
|
<source>This is not a valid UUID.</source>
|
||||||
|
<target>Dette er ikke en gyldig UUID.</target>
|
||||||
|
</trans-unit>
|
||||||
|
<trans-unit id="84">
|
||||||
|
<source>This value should be a multiple of {{ compared_value }}.</source>
|
||||||
|
<target>Verdien skal være flertall av {{ compared_value }}.</target>
|
||||||
|
</trans-unit>
|
||||||
|
<trans-unit id="85">
|
||||||
|
<source>This Business Identifier Code (BIC) is not associated with IBAN {{ iban }}.</source>
|
||||||
|
<target>Business Identifier Code (BIC) er ikke tilknyttet en IBAN {{ iban }}.</target>
|
||||||
|
</trans-unit>
|
||||||
|
<trans-unit id="86">
|
||||||
|
<source>This value should be valid JSON.</source>
|
||||||
|
<target>Verdien er ikke gyldig JSON.</target>
|
||||||
|
</trans-unit>
|
||||||
</body>
|
</body>
|
||||||
</file>
|
</file>
|
||||||
</xliff>
|
</xliff>
|
||||||
|
Reference in New Issue
Block a user