Literal "_none" value saved for seelct fields

Created on 4 December 2024, 4 months ago

Problem/Motivation

#3384140 introduced a regression of sorts; filtering in the NameWidget uses array_search() only returns the first key:

If needle is found in haystack more than once, the first matching key is returned. To return the keys for all matching values, use array_keys() with the optional filter_value parameter instead.

https://www.php.net/manual/en/function.array-search.php

So, if we have both title and generational in the form and the user leaves both blank, only one of these is going to get massaged to remove the "_none".

Steps to reproduce

Use a name field with multiple selects (e.g. title, generational) and leave both unassigned. When saving, only one of the values will be massaged and the other will have the literal "_none" persisted to the database.

Proposed resolution

Proposed patch attached

🐛 Bug report
Status

Active

Version

1.0

Component

Code

Created by

🇺🇸United States tkiehne

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

Merge Requests

Comments & Activities

  • Issue created by @tkiehne
  • 🇺🇸United States tkiehne
  • 🇺🇸United States tkiehne
  • First commit to issue fork.
  • Pipeline finished with Success
    4 months ago
    Total: 243s
    #358643
  • Attaching screenshot for reference.

  • heddn Nicaragua

    Feedback posted on the MR.

  • Pipeline finished with Success
    4 months ago
    Total: 353s
    #359891
  • Pipeline finished with Success
    4 months ago
    Total: 242s
    #359935
  • 🇦🇺Australia alan d.

    I cranked up my email account linked to drupal.org and saw this thread...

    I can't remember the sequence of the field widget submission processing, but can someone manually check that the validation in a required textfield component still triggers if you enter _blank for that component? If it does, the this minor fix may need a tweak to only target option based components.

  • heddn Nicaragua

    Feedback posted on the MR.

  • Pipeline finished with Success
    4 months ago
    Total: 214s
    #364126
  • @heddn In required textfield components i have tested that when we write _none then validation error is not triggering and _none is getting replaced by "" (empty strings) . When we write _blank then also there is no validation error and the field is having _blank value.

  • Pipeline finished with Success
    4 months ago
    Total: 225s
    #364136
  • heddn Nicaragua

    I think this could still add some more edge cases to the tests to address #11.

  • First commit to issue fork.
  • 🇳🇱Netherlands megachriz

    To address #11, I've added test coverage for using "_none" as value for text fields. And the tests are failing. If you for example enter "_none" for the field "Family" an empty value gets saved for that field. If the name field is required you get the more cryptic error message "This value should not be null.".

  • Pipeline finished with Failed
    3 months ago
    Total: 235s
    #384804
  • 🇳🇱Netherlands megachriz

    I've added a possible fix for the case from #11.

  • Pipeline finished with Success
    3 months ago
    Total: 238s
    #384933
  • 🇮🇳India sayan_k_dutta

    Reviewed the MR. Checked all the changes made.
    1. Checked that on submitting more than one empty components for the name field, all the "_none" values are replaced by empty strings.
    2. Checked that the validation does not trigger if we give "_blank" or "_none" as input.
    3. Checked all the tests written and all of them passed.

    Hence moving it to RTBC.

  • 🇪🇸Spain penyaskito Seville 💃, Spain 🇪🇸, UTC+2 🇪🇺

    Had this same issue when leaving title and generational components empty (only if BOTH were empty).
    This MR fixed the issue.

    As I'm using the name output as the entity title, had to write an upgrade path in my project. If it helps, customize your entity type id and field name:

    function mymodule_update_8001(&$sandbox): void {
      $nids = Database::getConnection()->select('node_revision__FIELD_NAME', 'name')
        ->fields('name', ['entity_id'])
        ->condition('FIELD_NAME_generational', '_none')
        ->execute()->fetchAll(\PDO::FETCH_COLUMN, 0);
      Database::getConnection()->update('node__FIELD_NAME')
        ->fields(['FIELD_NAME_generational' => ''])
        ->condition('FIELD_NAME_generational', '_none')
        ->execute();
      Database::getConnection()->update('node_revision__FIELD_NAME')
        ->fields(['FIELD_NAME_generational' => ''])
        ->condition('FIELD_NAME_generational', '_none')
        ->execute();
      \Drupal::entityTypeManager()->getStorage('node')->resetCache($nids);
    }
    
    
  • Status changed to RTBC about 2 months ago
  • 🇺🇸United States brad.bulger

    I am testing this using the MR diff as a patch on Name 1.0. We have code that runs in an ajax callback function that builds a fullname string to store in another field. $form_state->getValue('field_name') is returning "_none" in the array of components. So the name.formatter service's format() function treats those like strings - we get "_none John Doe _none".

    Is this the intended behavior? Anyone else who was counting on empty components returning empty strings is going to have a similar incompatibility problem.

  • 🇪🇸Spain penyaskito Seville 💃, Spain 🇪🇸, UTC+2 🇪🇺

    @brad.bulger Check my last comment where I posted a workaround for a similar usecase. Check if _none already hit the database before adding this patch.

  • 🇺🇸United States brad.bulger

    This is an AJAX callback in the form that is getting the name field value at runtime. Nothing is being written to the db. The relevant part is that $form_state->getValue('field_mynamefield') returns "_none" as the value for select list components.

    What I'm not clear on is if that's intended. Does the change here only acts on submit or something like that. Should the value be "_none" or "" for field validation? For an entity constraint on the field?

    I guess the change from blank to "_none" already went out in 1.0, I don't see any notes about the incompatibility change.

  • 🇪🇸Spain penyaskito Seville 💃, Spain 🇪🇸, UTC+2 🇪🇺

    I'd say that's expected from $form_state. You'd need to access the widget and call extractFormValues.

  • 🇨🇦Canada Liam Morland Ontario, CA 🇨🇦
  • 🇺🇸United States texas-bronius

    Thank you! Did not extensively test but confirmed patch from https://git.drupalcode.org/project/name/-/merge_requests/24 working in my use case.

  • 🇺🇸United States bkosborne New Jersey, USA

    Bumping to critical as this can lead to data issues. The MR works well for me.

  • 🇸🇬Singapore clairero

    Thanks, this MR resolves the issue for me too.

  • First commit to issue fork.
  • Pipeline finished with Success
    about 1 month ago
    Total: 255s
    #442304
  • 🇺🇸United States bluegeek9

    bluegeek9 changed the visibility of the branch 8.x-1.x to hidden.

Production build 0.71.5 2024