Plural Formula gets ignored for Singular Case for languages where Singular Form is not found in the first index

Created on 27 December 2024, 3 months ago

Problem/Motivation

Plural formula handling in PluralTranslatableMarkup::render() function ignores the Plural Formula Rule and hardcode the case for $count == 1 to use first index.

https://git.drupalcode.org/project/drupal/-/blob/11.x/core/lib/Drupal/Co...

As you can see from the latest code, It is ignoring the getPluralIndex for case where count is 1.

     $index = $this->getPluralIndex();
    if ($this->count == 1 || $index == 0 || count($translated_array) == 1) {
      // Singular form.
      $return = $translated_array[0];
    }

This leads to a problem where Plural Form Rules for languages where case for "count = 1" uses any other index but 0, leads to wrong Plural Form shown to user.

Example,

I was testing Plural Formula for Arabic, where singular form has index 1.

"Plural-Forms: nplurals=6; plural=(n==0 ? 0 : n==1 ? 1 : n==2 ? 2 : n%100>=3 && n%100<=10 ? 3 : n%100>=11 && n%100<=99 ? 4 : 5);\n

Steps to reproduce

- On Standard Drupal installation enable locale module
- Add and enable "Arabic" language
- Upload a Arabic PO file with translation for "@count items", this way you can set the rules for Arabic.
- Update Plural Translation for "1 comment" by visiting Translation edit for "/admin/config/regional/translate" to something as follows,

Notice that the labels provided for translators are already wrong here, since the first item is pointing to form for "zero" and second item points to "Singular" form.
- Run following code to display translation for singular form in Arabic,

ddev drush eval "print_r(\Drupal::translation()->formatPlural(1, '1 comment', '@count comments', array(), array('langcode' => 'ar')) . PHP_EOL);"

Result
1 - zero
Expected Result
1 - one

Proposed resolution

TBD

Remaining tasks

TBD

User interface changes

TBD

Introduced terminology

TBD

API changes

TBD

Data model changes

TBD

Release notes snippet

TBD

πŸ› Bug report
Status

Active

Version

11.0 πŸ”₯

Component

language system

Created by

πŸ‡©πŸ‡ͺGermany D34dMan Hamburg

Live updates comments and jobs are added and updated live.
Sign in to follow issues

Merge Requests

Comments & Activities

  • Issue created by @D34dMan
  • πŸ‡©πŸ‡ͺGermany D34dMan Hamburg

    If anybody shed light on why the conditional check for $this->count == 1 was included, it would be super helpful.

  • πŸ‡¨πŸ‡¦Canada Charlie ChX Negyesi 🍁Canada

    While I can answer your question it is not helpful at all: the code was added in 1831e1b690 and there was an attempt to fix this in #12096: fix bug with plural forms for some languages β†’ -- both were twenty years ago. The fix apparently was not enough. Locale code is ancient.

  • πŸ‡©πŸ‡ͺGermany D34dMan Hamburg

    It seems to me the following simplification can be made as a step forward:

    I have updated the MR with suggested changes.

    >... precomputing 0-199

    @ghost of drupal past, so we have a bug due to what seems to be a performance optimization. Thanks for the issues links which are quite helpful.

    ---

    P.S. It seems like some comment/s are vanishing, like ghost comments! Is it Drupal.Orgs way of keeping discussion between a dead guy and a ghost spooky Β―\_(ツ)_/Β―

  • πŸ‡ΊπŸ‡ΈUnited States smustgrave

    Don't see an MR. but it should have test coverage when opened please

  • Pipeline finished with Failed
    3 months ago
    Total: 612s
    #380782
  • πŸ‡©πŸ‡ͺGermany D34dMan Hamburg

    I am happy to write some test, but would like to clarify few things,

    Question A:
    Am not sure about core policy regarding test, I think we would need a kernel test for this one, Is that assumption correct?

    Question B:
    I see testing this is tricky. This refactor touches part which needs locale module to be enabled and configured in a certain way. So should this test be part of Locale Module testing scope? i.e. test be written under namespace Drupal\KernelTests\Core\Locale instead of Drupal\KernelTests\Core\StringTranslation

  • πŸ‡¨πŸ‡¦Canada Charlie ChX Negyesi 🍁Canada

    Yeah, tests fail... here's a slightly better version, with comments:

        $translated_array = explode(PoItem::DELIMITER, $this->translatedString);
        // No need to check plural formulas if only one translation is provided.
        $index = count($translated_array) === 1 ? 0 : $this->getPluralIndex();
        // If the translation is not found, it's guesswork, and the best guess
        // is the most common case where the first plural form is for n = 1 and
        // the second is for everything else. It's not always correct but
        // Β―\_(ツ)_/Β―
        $return = $translated_array[$index] ?? $translated_array[$this->count != 1];
    

    This does bring back the faulty assumption but it's delegated to a fallback case where I don't really have a better idea.

  • πŸ‡©πŸ‡ͺGermany D34dMan Hamburg

    @ghost of drupal past,

    I am afraid, we are increasing the scope of the fix.

    As far as the scope of this bug report is about, it feels removing the hardcoded condition for "$this->count == 1" should be enough. Trying to find a sane default for cases under following scenario should be followed up in a separate issue (or issues),

    - Case A: Translation data is corrupted, i.e. Message array doesn't contain the string at a particular index.
    - Case B: Translation index formula can be wrong because of using default plural formula for language where it doesn't make sense.
    - Case C: Translation index formula can be wrong due to corrupted Plural Form header in the PO file.

    The main source of above scenario would be content/configuration done by User. And as pointed out in comment #8, the outcome has too many variations to handle.

    As reported in this issue, despite having configured messages and Plural Formula correctly, Drupal Core fails to use the correct index, and hence that needs fixing.

  • πŸ‡¨πŸ‡¦Canada Charlie ChX Negyesi 🍁Canada

    Sure, it might be too much I was trying to optimize but it was just an idea.

  • First commit to issue fork.
  • Pipeline finished with Failed
    about 2 months ago
    Total: 444s
    #424698
  • Pipeline finished with Failed
    about 2 months ago
    Total: 603s
    #425282
  • Pipeline finished with Failed
    about 2 months ago
    Total: 507s
    #425355
Production build 0.71.5 2024