If a form element mapped to a #config_target has multiple constraint violations, only the last one is displayed

Created on 16 November 2023, over 1 year ago

Problem/Motivation

Try this:

In system.schema.yml, add the following constraints to system.site:page.front:

          constraints: 
            ValidPath: []
            Regex:
              pattern: '/^\//'
              message: 'This path must start with a slash.'

Now, in \Drupal\system\Form\SiteInformationForm::buildForm(), map $form['front_page']['site_frontpage'] to system.site:page.front using #config_target:

    $form['front_page']['site_frontpage'] = [
      '#type' => 'textfield',
      '#title' => $this->t('Default front page'),
      '#config_target' => new ConfigTarget(
        'system.site',
        'page.front',
        fromConfig: $this->aliasManager->getAliasByPath(...),
      ),
      '#required' => TRUE,
      '#size' => 40,
      '#description' => $this->t('Specify a relative URL to display as the front page.'),
      '#field_prefix' => $this->requestContext->getCompleteBaseUrl(),
    ];

Remove the lines from submitForm() and validateForm() which deal with this field as well.

Now then! If you visit that form and try to submit a non-existent path in that field which does not begin with a slash, the only error you'll get for it is the one coming from whichever the last constraint was -- in this case, Regex, even though the ValidPath constraint failed too.

The problem is the logic in <code>\Drupal\Core\Form\ConfigFormBase::validateForm(), which always sets the last error message on $index = 0, even if $index should not be used because the value being checked is not a sequence. Whoops!

      foreach ($violations as $violation) {
        $property_path = $violation->getPropertyPath();
        // Default to index 0.
        $index = 0;

        if (!isset($map[$config_name][$property_path]) && preg_match("/.*\.(\d+)$/", $property_path, $matches) === 1) {
          $index = intval($matches[1]);
          // The property path as known in the config key-to-form element map
          // will not have the sequence index in it.
          $property_path = rtrim($property_path, '0123456789.');
        }

        if (isset($map[$config_name][$property_path])) {
          $config_target = ConfigTarget::fromForm($map[$config_name][$property_path], $form);
          $form_element_name = implode('][', $config_target->elementParents);
        }
        else {
          $form_element_name = '';
        }
        $violations_per_form_element[$form_element_name][$index] = $violation;
      }

Proposed resolution

I'm not entirely sure yet, but really, $index needs to be advanced in accordance with the foreach() loop, unless the value is a sequence, in which case the handling needs to be a tad different.

Remaining tasks

We'll need test coverage for sure. Then we can make the fix.

🐛 Bug report
Status

Active

Version

10.2

Component
Form 

Last updated about 15 hours ago

Created by

🇺🇸United States phenaproxima Massachusetts

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

Comments & Activities

  • Issue created by @phenaproxima
  • @phenaproxima opened merge request.
  • Status changed to Needs work over 1 year ago
  • 🇺🇸United States phenaproxima Massachusetts

    OK, test coverage is written! Now to fix the bug...

  • Status changed to Needs review over 1 year ago
  • 🇺🇸United States phenaproxima Massachusetts
  • 🇺🇸United States smustgrave

    Ran the test-only feature

    There was 1 failure:
    1) Drupal\Tests\system\Functional\Form\ConfigTargetTest::testMultipleViolations
    Element matching xpath "//div[@data-drupal-messages]//div[(contains(@aria-label, "Error message") or contains(@aria-labelledby, "error")) and contains(., "Absolutely not. Durians are the worst.")]" not found.
    /builds/issue/drupal-3402228/core/tests/Drupal/Tests/WebAssert.php:830
    /builds/issue/drupal-3402228/core/modules/system/tests/src/Functional/Form/ConfigTargetTest.php:160
    /builds/issue/drupal-3402228/vendor/phpunit/phpunit/src/Framework/TestResult.php:728
    FAILURES!
    Tests: 4, Assertions: 18, Failures: 1.
    

    Which shows the issue of the first error not showing.

    The fix provided seems to work, if that's the one you want to go forward with could you update the proposed solution section please.

  • Status changed to Needs work over 1 year ago
  • 🇧🇪Belgium wim leers Ghent 🇧🇪🇪🇺

    We need to ensure this works consistently with how >1 constraint violation on a single content entity field item property is presented. So we need to research that first.

  • 🇺🇸United States phenaproxima Massachusetts

    I'm trying to figure out how to get a content entity even set up in such a way that I could violate two conditions of a particular field type. It's not easy.

    But, a review of the code reveals this, in \Drupal\Core\Field\WidgetBase::flagErrors()

              if (is_numeric($delta)) {
                $violations_by_delta[$delta][] = $violation;
              }
              // Violations at the ItemList level are not associated to any delta.
              else {
                $item_list_violations[] = $violation;
              }
    

    That is very, very similar to what I'm proposing here.

  • 🇺🇸United States phenaproxima Massachusetts

    OK, so I had to contrive this a bit, but I tested how it works on content entity forms.

    Short answer: If there are multiple violations on a single field, only one appears to be displayed. That is terrible UX, a perfect way to make people hate Drupal, and should be fixed ASAP in another issue.

    How I tested

    I changed \Drupal\Core\Field\Plugin\Field\FieldType\StringItem::getConstraints() so that it both checks the maximum length of the input, and prevents the input value from starting with a number. Here was the code I used for that:

          $constraints[] = $constraint_manager->create('ComplexData', [
            'value' => [
              'Length' => [
                'max' => $max_length,
                'maxMessage' => $this->t('%name: may not be longer than @max characters.', ['%name' => $this->getFieldDefinition()->getLabel(), '@max' => $max_length]),
              ],
              'Regex' => [
                'pattern' => '/^\d/',
                'match' => false,
                'message' => $this->t('Cannot start with a number.'),
              ],
            ],
          ]);
    

    Then I disabled the #maxlength property in the form system, and in the string_textfield widget, by commenting out a line or two in \Drupal\Core\Form\FormValidator::performRequiredValidation(), \Drupal\Core\Render\Element\Textfield::getInfo(), and \Drupal\Core\Field\Plugin\Field\FieldWidget\StringTextfieldWidget::formElement().

    I then went to /node/add/article (on a standard install) and entered this value for the title of the node, which is a string_textfield (bear in mind I had to use the inspector to remove the maxlength attribute added by default):

    99 red balloons;99 red balloons;99 red balloons;99 red balloons;99 red balloons;99 red balloons;99 red balloons;99 red balloons;99 red balloons;99 red balloons;99 red balloons;99 red balloons;99 red balloons;99 red balloons;99 red balloons;99 red balloons;

    Here's what I saw when I tried to submit the form:

    There should have been two errors, not one, because that input string is longer than the field's default maximum length of 128
    characters. I disabled the length validation at the form level, you see, but not at the constraint (data) level.

    So it would appear that my supposition in #7 was incorrect.

    Conclusion

    Look...if you want as many people as possible to hate Drupal as much as possible, we should keep this UX exactly the way it is. There is nothing, N O T H I N G, more frustrating than getting one error on a field, only to fix it and discover that there was, in fact, another error underneath it that the system didn't tell you about.

    This should be fixed for both content entity forms and #config_target-using config forms in this issue.

  • 🇨🇦Canada gapple

    The current MR would address multiple violations on a non-sequence value, but if an item within a sequence has multiple violations it looks like only the last one will be kept.

    ConfigFormBase::formatMultipleViolationsMessage() also assumes it's being called on a sequence, so multiple constraint violations on a non-sequence value would still be formatted like "Entry 0: Constraint 1 message \n Entry 1: Constraint 2 message".
    Forms can override the method to change the output based on the form element name in this case though.

  • 🇮🇳India mdsohaib4242

    To simplify the solution, we can focus directly on collecting error messages for each field without using an index. Instead, we’ll gather each violation message into an array for the field and then concatenate them when setting the error.

    • We skip indexing and directly store each violation message under $form_element_name.
    • Each field's error messages are combined into a single string in one step with implode(' ', $messages).
Production build 0.71.5 2024