From 1d50852a7a61981cf84e9f71049da58820cebb60 Mon Sep 17 00:00:00 2001 From: catch <catch@35733.no-reply.drupal.org> Date: Sun, 20 Sep 2015 19:37:52 +0200 Subject: [PATCH] Issue #2570107 by alexpott: Make format_plural() return a PluralTranslatableString object to remove reliance on a static, unpredictable safe list --- core/includes/common.inc | 9 +- .../PluralTranslatableString.php | 185 ++++++++++++++++++ .../TranslationInterface.php | 47 +---- .../StringTranslation/TranslationManager.php | 47 ++--- .../src/Tests/LocalePluralFormatTest.php | 18 +- .../src/Plugin/views/field/NumericField.php | 3 +- core/modules/views_ui/src/ViewEditForm.php | 6 +- .../TranslationManagerTest.php | 10 +- core/tests/Drupal/Tests/UnitTestCase.php | 7 +- 9 files changed, 231 insertions(+), 101 deletions(-) create mode 100644 core/lib/Drupal/Core/StringTranslation/PluralTranslatableString.php diff --git a/core/includes/common.inc b/core/includes/common.inc index d8e399fc1cf4..e66261e447e4 100644 --- a/core/includes/common.inc +++ b/core/includes/common.inc @@ -33,6 +33,7 @@ use Drupal\Component\Utility\NestedArray; use Drupal\Core\Datetime\DrupalDateTime; use Drupal\Core\Routing\GeneratorNotInitializedException; +use Drupal\Core\StringTranslation\PluralTranslatableString; use Drupal\Core\Template\Attribute; use Drupal\Core\Render\BubbleableMetadata; use Drupal\Core\Render\Element; @@ -142,11 +143,11 @@ /** * The delimiter used to split plural strings. * - * This is the ETX (End of text) character and is used as a minimal means to - * separate singular and plural variants in source and translation text. It - * was found to be the most compatible delimiter for the supported databases. + * @deprecated in Drupal 8.0.x-dev, will be removed before Drupal 9.0.0. + * Use \Drupal\Core\StringTranslation\PluralTranslatableString::DELIMITER + * instead. */ -const LOCALE_PLURAL_DELIMITER = "\03"; +const LOCALE_PLURAL_DELIMITER = PluralTranslatableString::DELIMITER; /** * Prepares a 'destination' URL query parameter for use with url(). diff --git a/core/lib/Drupal/Core/StringTranslation/PluralTranslatableString.php b/core/lib/Drupal/Core/StringTranslation/PluralTranslatableString.php new file mode 100644 index 000000000000..f29d6e7822cf --- /dev/null +++ b/core/lib/Drupal/Core/StringTranslation/PluralTranslatableString.php @@ -0,0 +1,185 @@ +<?php + +/** + * @file + * Contains \Drupal\Core\StringTranslation\PluralTranslatableString. + */ + +namespace Drupal\Core\StringTranslation; + +use Drupal\Component\Utility\PlaceholderTrait; +use Drupal\Component\Utility\SafeMarkup; + +/** + * A class to hold plural translatable strings. + */ +class PluralTranslatableString extends TranslatableString { + use PlaceholderTrait; + + /** + * The delimiter used to split plural strings. + * + * This is the ETX (End of text) character and is used as a minimal means to + * separate singular and plural variants in source and translation text. It + * was found to be the most compatible delimiter for the supported databases. + */ + const DELIMITER = "\03"; + + /** + * The item count to display. + * + * @var int + */ + protected $count; + + /** + * The already translated string. + * + * @var string + */ + protected $translatedString; + + /** + * A bool that statically caches whether locale_get_plural() exists. + * + * @var bool + */ + protected static $localeEnabled; + + /** + * Constructs a new PluralTranslatableString object. + * + * Parses values passed into this class through the format_plural() function + * in Drupal and handles an optional context for the string. + * + * @param int $count + * The item count to display. + * @param string $singular + * The string for the singular case. Make sure it is clear this is singular, + * to ease translation (e.g. use "1 new comment" instead of "1 new"). Do not + * use @count in the singular string. + * @param string $plural + * The string for the plural case. Make sure it is clear this is plural, to + * ease translation. Use @count in place of the item count, as in + * "@count new comments". + * @param array $args + * (optional) An associative array of replacements to make after + * translation. Instances of any key in this array are replaced with the + * corresponding value. Based on the first character of the key, the value + * is escaped and/or themed. See + * \Drupal\Component\Utility\SafeMarkup::format(). Note that you do not need + * to include @count in this array; this replacement is done automatically + * for the plural cases. + * @param array $options + * (optional) An associative array of additional options. See t() for + * allowed keys. + * @param \Drupal\Core\StringTranslation\TranslationInterface $string_translation + * (optional) The string translation service. + */ + public function __construct($count, $singular, $plural, array $args = [], array $options = [], TranslationInterface $string_translation = NULL) { + $this->count = $count; + $translatable_string = implode(static::DELIMITER, array($singular, $plural)); + parent::__construct($translatable_string, $args, $options, $string_translation); + } + + /** + * Constructs a new class instance from an already translated string. + * + * This method ensures that the string is pluralized correctly. As opposed + * to the __construct() method, this method is designed to be invoked with + * a string already translated (such as with configuration translation). + * + * @param int $count + * The item count to display. + * @param string $translated_string + * The already translated string. + * @param array $args + * An associative array of replacements to make after translation. Instances + * of any key in this array are replaced with the corresponding value. + * Based on the first character of the key, the value is escaped and/or + * themed. See \Drupal\Component\Utility\SafeMarkup::format(). Note that you + * do not need to include @count in this array; this replacement is done + * automatically for the plural cases. + * @param array $options + * An associative array of additional options. See t() for allowed keys. + * + * @return \Drupal\Core\StringTranslation\PluralTranslatableString + * A PluralTranslatableString object. + */ + public static function createFromTranslatedString($count, $translated_string, array $args = [], array $options = []) { + $safe = TRUE; + foreach (array_keys($args) as $arg_key) { + // If the string has arguments that start with '!' we consider it unsafe + // and return the translation as a string for backward compatibility + // purposes. + // @todo https://www.drupal.org/node/2570037 remove this temporary + // workaround. + if (0 === strpos($arg_key, '!') && !SafeMarkup::isSafe($args[$arg_key])) { + $safe = FALSE; + break; + } + } + $plural = new static($count, '', '', $args, $options); + $plural->translatedString = $translated_string; + return $safe ? $plural : (string) $plural; + } + + /** + * Renders the object as a string. + * + * @return string + * The translated string. + */ + public function render() { + if (!$this->translatedString) { + $this->translatedString = $this->getStringTranslation()->translateString($this); + } + if ($this->translatedString === '') { + return ''; + } + + $arguments = $this->getArguments(); + $arguments['@count'] = $this->count; + $translated_array = explode(static::DELIMITER, $this->translatedString); + + if ($this->count == 1) { + return $this->placeholderFormat($translated_array[0], $arguments); + } + + $index = $this->getPluralIndex(); + if ($index == 0) { + // Singular form. + $return = $translated_array[0]; + } + else { + if (isset($translated_array[$index])) { + // N-th plural form. + $return = $translated_array[$index]; + } + else { + // If the index cannot be computed or there's no translation, use the + // second plural form as a fallback (which allows for most flexibility + // with the replaceable @count value). + $return = $translated_array[1]; + } + } + + return $this->placeholderFormat($return, $arguments); + } + + /** + * Gets the plural index through the gettext formula. + * + * @return int + */ + protected function getPluralIndex() { + if (!isset(static::$localeEnabled)) { + static::$localeEnabled = function_exists('locale_get_plural'); + } + if (function_exists('locale_get_plural')) { + return locale_get_plural($this->count, $this->getOption('langcode')); + } + return -1; + } + +} diff --git a/core/lib/Drupal/Core/StringTranslation/TranslationInterface.php b/core/lib/Drupal/Core/StringTranslation/TranslationInterface.php index 224c1f9a7e1b..0521612e925d 100644 --- a/core/lib/Drupal/Core/StringTranslation/TranslationInterface.php +++ b/core/lib/Drupal/Core/StringTranslation/TranslationInterface.php @@ -54,9 +54,10 @@ public function translateString(TranslatableString $translated_string); /** * Formats a string containing a count of items. * - * This function ensures that the string is pluralized correctly. Since t() is - * called by this function, make sure not to pass already-localized strings to - * it. See formatPluralTranslated() for that. + * This function ensures that the string is pluralized correctly. Since + * TranslationInterface::translate() is called by this function, make sure not + * to pass already-localized strings to it. See + * PluralTranslatableString::createFromTranslatedString() for that. * * For example: * @code @@ -91,50 +92,16 @@ public function translateString(TranslatableString $translated_string); * @param array $options * An associative array of additional options. See t() for allowed keys. * - * @return string + * @return \Drupal\Core\StringTranslation\PluralTranslatableString * A translated string. * - * @see self::translate() + * @see \Drupal\Core\StringTranslation\TranslationInterface::translate() * @see t() * @see \Drupal\Component\Utility\SafeMarkup::format() - * @see self::formatPluralTranslated + * @see \Drupal\Core\StringTranslation\PluralTranslatableString::createFromTranslatedString() */ public function formatPlural($count, $singular, $plural, array $args = array(), array $options = array()); - /** - * Formats an already translated string containing a count of items. - * - * This function ensures that the string is pluralized correctly. As opposed - * to the formatPlural() method, this method is designed to be invoked with - * a string already translated (such as with configuration translation). - * - * @param int $count - * The item count to display. - * @param string $translation - * The string containing the translation of a singular/plural pair. It may - * contain any number of possible variants (depending on the language - * translated to) separated by the value of the LOCALE_PLURAL_DELIMITER - * constant. - * @param array $args - * Associative array of replacements to make in the translation. Instances - * of any key in this array are replaced with the corresponding value. - * Based on the first character of the key, the value is escaped and/or - * themed. See \Drupal\Component\Utility\SafeMarkup::format(). Note that you do - * not need to include @count in this array; this replacement is done - * automatically for the plural cases. - * @param array $options - * An associative array of additional options. The 'context' key is not - * supported because the passed string is already translated. Use the - * 'langcode' key to ensure the proper plural logic is used. - * - * @return string - * The correct substring for the given $count with $args replaced. - * - * @see self::formatPlural() - * @see \Drupal\Component\Utility\SafeMarkup::format() - */ - public function formatPluralTranslated($count, $translation, array $args = array(), array $options = array()); - /** * Returns the number of plurals supported by a given language. * diff --git a/core/lib/Drupal/Core/StringTranslation/TranslationManager.php b/core/lib/Drupal/Core/StringTranslation/TranslationManager.php index 836abd89c373..ce4c6b36c970 100644 --- a/core/lib/Drupal/Core/StringTranslation/TranslationManager.php +++ b/core/lib/Drupal/Core/StringTranslation/TranslationManager.php @@ -191,43 +191,20 @@ protected function doTranslate($string, array $options = array()) { * {@inheritdoc} */ public function formatPlural($count, $singular, $plural, array $args = array(), array $options = array()) { - $translatable_string = implode(LOCALE_PLURAL_DELIMITER, array($singular, $plural)); - $translated_strings = $this->doTranslate($translatable_string, $options); - return $this->formatPluralTranslated($count, $translated_strings, $args, $options); - } - - /** - * {@inheritdoc} - */ - public function formatPluralTranslated($count, $translation, array $args = array(), array $options = array()) { - $args['@count'] = $count; - $translated_array = explode(LOCALE_PLURAL_DELIMITER, $translation); - - if ($count == 1) { - return SafeMarkup::format($translated_array[0], $args); - } - - // Get the plural index through the gettext formula. - // @todo implement static variable to minimize function_exists() usage. - $index = (function_exists('locale_get_plural')) ? locale_get_plural($count, isset($options['langcode']) ? $options['langcode'] : NULL) : -1; - if ($index == 0) { - // Singular form. - $return = $translated_array[0]; - } - else { - if (isset($translated_array[$index])) { - // N-th plural form. - $return = $translated_array[$index]; - } - else { - // If the index cannot be computed or there's no translation, use - // the second plural form as a fallback (which allows for most flexibility - // with the replaceable @count value). - $return = $translated_array[1]; + $safe = TRUE; + foreach (array_keys($args) as $arg_key) { + // If the string has arguments that start with '!' we consider it unsafe + // and return the translation as a string for backward compatibility + // purposes. + // @todo https://www.drupal.org/node/2570037 remove this temporary + // workaround. + if (0 === strpos($arg_key, '!') && !SafeMarkup::isSafe($args[$arg_key])) { + $safe = FALSE; + break; } } - - return SafeMarkup::format($return, $args); + $plural = new PluralTranslatableString($count, $singular, $plural, $args, $options, $this); + return $safe ? $plural : (string) $plural; } /** diff --git a/core/modules/locale/src/Tests/LocalePluralFormatTest.php b/core/modules/locale/src/Tests/LocalePluralFormatTest.php index 08cdc2d8777c..386cd5d378eb 100644 --- a/core/modules/locale/src/Tests/LocalePluralFormatTest.php +++ b/core/modules/locale/src/Tests/LocalePluralFormatTest.php @@ -7,6 +7,7 @@ namespace Drupal\locale\Tests; +use Drupal\Core\StringTranslation\PluralTranslatableString; use Drupal\simpletest\WebTestBase; /** @@ -130,12 +131,15 @@ public function testGetPluralFormat() { // expected index as per the logic for translation lookups. $expected_plural_index = ($count == 1) ? 0 : $expected_plural_index; $expected_plural_string = str_replace('@count', $count, $plural_strings[$langcode][$expected_plural_index]); - $this->assertIdentical(\Drupal::translation()->formatPlural($count, '1 hour', '@count hours', array(), array('langcode' => $langcode)), $expected_plural_string, 'Plural translation of 1 hours / @count hours for count ' . $count . ' in ' . $langcode . ' is ' . $expected_plural_string); - // DO NOT use translation to pass into formatPluralTranslated() this - // way. It is designed to be used with *already* translated text like - // settings from configuration. We use PHP translation here just because - // we have the expected result data in that format. - $this->assertIdentical(\Drupal::translation()->formatPluralTranslated($count, \Drupal::translation()->translate('1 hour' . LOCALE_PLURAL_DELIMITER . '@count hours', array(), array('langcode' => $langcode)), array(), array('langcode' => $langcode)), $expected_plural_string, 'Translated plural lookup of 1 hours / @count hours for count ' . $count . ' in ' . $langcode . ' is ' . $expected_plural_string); + $this->assertIdentical(\Drupal::translation()->formatPlural($count, '1 hour', '@count hours', array(), array('langcode' => $langcode))->render(), $expected_plural_string, 'Plural translation of 1 hours / @count hours for count ' . $count . ' in ' . $langcode . ' is ' . $expected_plural_string); + // DO NOT use translation to pass translated strings into + // PluralTranslatableString::createFromTranslatedString() this way. It + // is designed to be used with *already* translated text like settings + // from configuration. We use PHP translation here just because we have + // the expected result data in that format. + $translated_string = \Drupal::translation()->translate('1 hour' . PluralTranslatableString::DELIMITER . '@count hours', array(), array('langcode' => $langcode)); + $plural = PluralTranslatableString::createFromTranslatedString($count, $translated_string, array(), array('langcode' => $langcode)); + $this->assertIdentical($plural->render(), $expected_plural_string); } } } @@ -223,7 +227,7 @@ public function testPluralEditExport() { // langcode here because the language will be English by default and will // not save our source string for performance optimization if we do not ask // specifically for a language. - \Drupal::translation()->formatPlural(1, '1 day', '@count days', array(), array('langcode' => 'fr')); + \Drupal::translation()->formatPlural(1, '1 day', '@count days', array(), array('langcode' => 'fr'))->render(); $lid = db_query("SELECT lid FROM {locales_source} WHERE source = :source AND context = ''", array(':source' => "1 day" . LOCALE_PLURAL_DELIMITER . "@count days"))->fetchField(); // Look up editing page for this plural string and check fields. $search = array( diff --git a/core/modules/views/src/Plugin/views/field/NumericField.php b/core/modules/views/src/Plugin/views/field/NumericField.php index 999a648113c2..e558de3b348d 100644 --- a/core/modules/views/src/Plugin/views/field/NumericField.php +++ b/core/modules/views/src/Plugin/views/field/NumericField.php @@ -8,6 +8,7 @@ namespace Drupal\views\Plugin\views\field; use Drupal\Core\Form\FormStateInterface; +use Drupal\Core\StringTranslation\PluralTranslatableString; use Drupal\views\ResultRow; /** @@ -174,7 +175,7 @@ public function render(ResultRow $values) { // If we should format as plural, take the (possibly) translated plural // setting and format with the current language. if (!empty($this->options['format_plural'])) { - $value = $this->formatPluralTranslated($value, $this->options['format_plural_string']); + $value = PluralTranslatableString::createFromTranslatedString($value, $this->options['format_plural_string']); } return $this->sanitizeValue($this->options['prefix'], 'xss') diff --git a/core/modules/views_ui/src/ViewEditForm.php b/core/modules/views_ui/src/ViewEditForm.php index 1b335de24af6..3b1d6afe361c 100644 --- a/core/modules/views_ui/src/ViewEditForm.php +++ b/core/modules/views_ui/src/ViewEditForm.php @@ -139,14 +139,14 @@ public function form(array $form, FormStateInterface $form_state) { '#account' => $this->entityManager->getStorage('user')->load($view->lock->owner), ); $lock_message_substitutions = array( - '!user' => drupal_render($username), - '!age' => $this->dateFormatter->formatTimeDiffSince($view->lock->updated), + '@user' => drupal_render($username), + '@age' => $this->dateFormatter->formatTimeDiffSince($view->lock->updated), '@url' => $view->url('break-lock-form'), ); $form['locked'] = array( '#type' => 'container', '#attributes' => array('class' => array('view-locked', 'messages', 'messages--warning')), - '#children' => $this->t('This view is being edited by user !user, and is therefore locked from editing by others. This lock is !age old. Click here to <a href="@url">break this lock</a>.', $lock_message_substitutions), + '#children' => $this->t('This view is being edited by user @user, and is therefore locked from editing by others. This lock is @age old. Click here to <a href="@url">break this lock</a>.', $lock_message_substitutions), '#weight' => -10, ); } diff --git a/core/tests/Drupal/Tests/Core/StringTranslation/TranslationManagerTest.php b/core/tests/Drupal/Tests/Core/StringTranslation/TranslationManagerTest.php index a19c0088656c..7eaa99dc6366 100644 --- a/core/tests/Drupal/Tests/Core/StringTranslation/TranslationManagerTest.php +++ b/core/tests/Drupal/Tests/Core/StringTranslation/TranslationManagerTest.php @@ -5,7 +5,7 @@ * Contains \Drupal\Tests\Core\StringTranslation\TranslationManagerTest. */ -namespace Drupal\Tests\Core\StringTranslation { +namespace Drupal\Tests\Core\StringTranslation; use Drupal\Component\Utility\SafeMarkup; use Drupal\Component\Utility\SafeStringInterface; @@ -106,11 +106,3 @@ public function __construct() { } } - -} - -namespace { - if (!defined('LOCALE_PLURAL_DELIMITER')) { - define('LOCALE_PLURAL_DELIMITER', "\03"); - } -} diff --git a/core/tests/Drupal/Tests/UnitTestCase.php b/core/tests/Drupal/Tests/UnitTestCase.php index 0b845d34ff12..9d805be23369 100644 --- a/core/tests/Drupal/Tests/UnitTestCase.php +++ b/core/tests/Drupal/Tests/UnitTestCase.php @@ -14,6 +14,8 @@ use Drupal\Core\DependencyInjection\ContainerBuilder; use Drupal\Component\Utility\PlaceholderTrait; use Drupal\Core\StringTranslation\TranslatableString; +use Drupal\Core\StringTranslation\PluralTranslatableString; + /** * Provides a base class and helpers for Drupal unit tests. @@ -231,8 +233,9 @@ public function getStringTranslationStub() { }); $translation->expects($this->any()) ->method('formatPlural') - ->willReturnCallback(function ($count, $singular, $plural, array $args = [], array $options = []) { - return $count === 1 ? SafeMarkup::format($singular, $args) : SafeMarkup::format($plural, $args + ['@count' => $count]); + ->willReturnCallback(function ($count, $singular, $plural, array $args = [], array $options = []) use ($translation) { + $wrapper = new PluralTranslatableString($count, $singular, $plural, $args, $options, $translation); + return $wrapper; }); return $translation; } -- GitLab