Form state storage changes in #process callbacks are not cached during form rebuild.

Created on 20 October 2020, almost 4 years ago
Updated 22 August 2024, 22 days ago

Problem/Motivation

This issue originally surfaced in Entity Browser β†’ where, in certain configurations used with Inline Entity Form β†’ or Layout Builder, there can be unexpected results:

#2764889: Entity Browser widget loses selected images in inline entity form β†’
#3046416: Remove button conflict wrong triggering element β†’
#3104901: Entity Browser used in a entity referenced field of a layout builder custom block is not working β†’

But it seems the underlying issue may actually be a core FormBuilder bug.

Comment #7 β†’ in #2764889: Entity Browser widget loses selected images in inline entity form β†’ documents what's happening:

I have investigated further what makes inconsistent form state -> and I have ended in IEF module.

So what is going on (in "short"): for example when remove button is pressed without IEF

  1. in processing of action (FormBuilder::processForm()), rebuild is triggered -> FormBuilder::rebuildForm() and then FormBuilder::retrieveForm() -> that will execute EntityReferenceBrowserWidget::formElement()
  2. setting of new state is done in EntityReferenceBrowserWidget::formElement()
  3. and after FormBuilder::retrieveForm() in FormBuilder::rebuildForm() form state will be saved (cached)
  4. so we end with correct form state (when I say form state I mean form state relevant for Entity Browser)

Other case: for example when remove button is pressed within IEF

  1. in processing of action (FormBuilder::processForm()), rebuild is triggered -> FormBuilder::rebuildForm() and then FormBuilder::retrieveForm() -> that will execute creating of IEF form and -> it will generate #process callback for inner form
  2. and after FormBuilder::retrieveForm() in FormBuilder::rebuildForm() form state will be saved (cached)
  3. after saving of form state inside FormBuilder::rebuildForm() -> FormBuilder::doBuildForm() will be triggered. That will pick up all defined #process callbacks and execute them
  4. execution of defined #process callback will execute EntityReferenceBrowserWidget::formElement() and new form state will be set
  5. But!!! Problem is that form state is already saved (cached) and form state set during execution of #process callback will not be saved and actions after that will have wrong form state

Inline block forms in Layout Builder work similarly to IEF, in that the entity form is built in a #process callback (see \Drupal\layout_builder\Plugin\Block\InlineBlock::blockForm()

The root issue seems to be that in \Drupal\Core\Form\FormBuilder::rebuildForm(), the form and form state is cached before the call to doBuildForm, where the #process callbacks are run. If we compare how the form cache is set in rebuildform() to processForm, we can see that in the latter method, caching is done after the doBuildForm() call, using a copy of the form array created before the doBuildForm().

While the identified issues are specific to Entity Browser, this could affect any code where form state storage is updated in #process callbacks.

Steps to reproduce

See #13 πŸ› Form state storage changes in #process callbacks are not cached during form rebuild. Needs review

My setup is Drupal 10.1.7 with PHP 8.1.

The patch solved this bug:
1. Create a custom block with a required reference field, e.g. for nodes
2. Configure its form display to use an entity browser to display the field. Display the Remove button.
3. Configure the entity browser to use modal windows.
4. Edit the layout of a basic page and add the custom block.
5. Use the entity browser to add a couple of nodes.
6. Remove one of the nodes.
7. Remove another one. Instead of removing the node, the modal window to add more nodes opens.

It also solves this bug:
1. Create a custom block with a required reference field, e.g. for nodes
2. Configure its form display to use an entity browser to display the field.
3. Configure the entity browser to use modal windows.
4. Edit the layout of a basic page and add the custom block.
5. Use the entity browser to add a couple of nodes.
6. Try to save the block, but provoke a validation error, e.g. by saving the block without a title. The Layout Builder is reloaded, but the previously selected nodes are not displayed anymore. They are back again if you add another node or if you simply save the block without seeing them.

Proposed resolution

Use similar form caching code in rebuildForm() to what is done in processForm():

  1. Save a copy of the form array before doBuildForm() call
  2. Call doBuildForm()
  3. Cache if necessary with unprocessed copy of form array saved before doBuildForm()
  4. return built and processed form array

Remaining tasks

Write patch and tests.

User interface changes

API changes

Data model changes

Release notes snippet

πŸ› Bug report
Status

RTBC

Version

11.0 πŸ”₯

Component
FormΒ  β†’

Last updated 26 minutes ago

Created by

Live updates comments and jobs are added and updated live.
  • Needs framework manager review

    It is used to alert the framework manager core committer(s) that an issue significantly impacts (or has the potential to impact) multiple subsystems or represents a significant change or addition in architecture or public APIs, and their signoff is needed (see the governance policy draft for more information). If an issue significantly impacts only one subsystem, use Needs subsystem maintainer review instead, and make sure the issue component is set to the correct subsystem.

Sign in to follow issues

Merge Requests

Comments & Activities

Not all content is available!

It's likely this issue predates Contrib.social: some issue and comment data are missing.

  • πŸ‡©πŸ‡ͺGermany Ammaletu Bonn, Germany

    I just tested the patch from #2 and all kinds of bugs went away. It's hard to imagine how much headscratching and confusion could have been avoided with this being included in core three years ago!

    My setup is Drupal 10.1.7 with PHP 8.1.

    The patch solved this bug:
    1. Create a custom block with a required reference field, e.g. for nodes
    2. Configure its form display to use an entity browser to display the field. Display the Remove button.
    3. Configure the entity browser to use modal windows.
    4. Edit the layout of a basic page and add the custom block.
    5. Use the entity browser to add a couple of nodes.
    6. Remove one of the nodes.
    7. Remove another one. Instead of removing the node, the modal window to add more nodes opens.

    It also solves this bug:
    1. Create a custom block with a required reference field, e.g. for nodes
    2. Configure its form display to use an entity browser to display the field.
    3. Configure the entity browser to use modal windows.
    4. Edit the layout of a basic page and add the custom block.
    5. Use the entity browser to add a couple of nodes.
    6. Try to save the block, but provoke a validation error, e.g. by saving the block without a title. The Layout Builder is reloaded, but the previously selected nodes are not displayed anymore. They are back again if you add another node or if you simply save the block without seeing them.

    It also apparently solves a bug we had where a cached form_state was reused when different anonymous users opened the form at the same time. Only the fastest of these users then could save the form, the other's got a "duplicate UUID" error.

    So, how do we add a test for this in a generic way, independent of the entity browser module? And are we sure that the approach in the patch is the right way? Is there a way to get a core committer to have a look at this? To me, this seems like quite a fundamental issue, causing numerous seemingly unrelated and hard to debug bugs. Not some minor issue that can easily be patched by community members.

  • effulgentsia is a Form API subsystem maintainer, and from his comments a couple years ago, he was hoping to see a test to determine next steps.

    I was in the process of creating an automated test for this back then, but the test I wrote was not working as expected, and then I moved to other project. I happened to re-visit this a couple weeks ago, but the challenge is that there are no forms in core that I can find that sets form storage after the cache happens, and it's non-trivial trying to devise a test form that does.

    This issue only has 10 followers, so I'm guessing there isn't a lot of awareness of how it affects entity browser or similar other contrib, but perhaps if there were more eyes on this issue, things could move forward.

  • πŸ‡§πŸ‡ͺBelgium Wim Leers Ghent πŸ‡§πŸ‡ͺπŸ‡ͺπŸ‡Ί
  • Pipeline finished with Failed
    9 months ago
    Total: 170s
    #68534
  • Pipeline finished with Failed
    9 months ago
    #68535
  • Pipeline finished with Success
    9 months ago
    Total: 552s
    #68537
  • Status changed to Needs review 9 months ago
  • Added a test and created an MR. Test doesn't really demonstrate a use case, but it does demonstrate that form state changes in #process callbacks aren't cached during rebuild.

  • It also apparently solves a bug we had where a cached form_state was reused when different anonymous users opened the form at the same time. Only the fastest of these users then could save the form, the other's got a "duplicate UUID" error.

    @Ammaletu Is the bug reproducible without contrib or custom code? If so, can you provide steps to reproduce?

  • πŸ‡§πŸ‡ͺBelgium Wim Leers Ghent πŸ‡§πŸ‡ͺπŸ‡ͺπŸ‡Ί

    Very interesting change! 🧐 πŸ‘

    This indeed needs sign-off from a Form API maintainer, because it has significant potential repercussions. Pinged @tim.plunkett.

  • Pipeline finished with Success
    8 months ago
    Total: 618s
    #75243
  • πŸ‡©πŸ‡ͺGermany Ammaletu Bonn, Germany

    @Ammaletu Is the bug reproducible without contrib or custom code? If so, can you provide steps to reproduce?

    I'm afraid not. I had another look, and we got the "duplicate UUID" problem because we are caching the form state (to solve yet another bug). Invoking the cache kill switch on the pages with the form solved this for guest users, except for one form which is displayed via AJAX. For this last form with the "duplicate UUID" problem, the patch from this issue helped. I'am afraid I don't have the time and expertise to boil this down into a generic use case.

  • I'm afraid not. I had another look, and we got the "duplicate UUID" problem because we are caching the form state (to solve yet another bug).

    OK, thanks for checking. Mostly I was asking because there wasn't context on what type of form/use case was surfacing the "duplicate UUID" issue, but I'm assuming from your last comment that anonymous users have access to an entity form using entity browser.

  • Status changed to Needs work 7 months ago
  • The Needs Review Queue Bot β†’ tested this issue. It fails the Drupal core commit checks. Therefore, this issue status is now "Needs work".

    This does not mean that the patch necessarily needs to be re-rolled or the MR rebased. Read the Issue Summary, the issue tags and the latest discussion here to determine what needs to be done.

    Consult the Drupal Contributor Guide β†’ to find step-by-step guides for working with issues.

  • Pipeline finished with Failed
    7 months ago
    Total: 170s
    #106070
  • Pipeline finished with Success
    7 months ago
    #106081
  • Status changed to Needs review 7 months ago
  • Rebased and addressed new code sniff issue.

  • Pipeline finished with Success
    5 months ago
    Total: 5361s
    #136918
  • πŸ‡ΊπŸ‡ΈUnited States smustgrave

    Going to bump this to framework manager.

  • Pipeline finished with Success
    4 months ago
    Total: 1730s
    #163766
  • Status changed to RTBC about 1 month ago
  • πŸ‡ΊπŸ‡ΈUnited States smustgrave

    Going to take a leap on this one.

    1) Drupal\Tests\system\Functional\Form\FormStatePersistTest::testFormStatePersistence
    Element matching xpath "//div[@data-drupal-messages]//div[contains(., "Rebuild state cached.")]" not found.
    /builds/issue/drupal-3177977/core/tests/Drupal/Tests/WebAssert.php:856
    /builds/issue/drupal-3177977/core/modules/system/tests/src/Functional/Form/FormStatePersistTest.php:53
    /builds/issue/drupal-3177977/vendor/phpunit/phpunit/src/Framework/TestResult.php:729
    FAILURES!
    Tests: 1, Assertions: 5, Failures: 1.
    

    Shows the test coverage for the change.

    Reviewing the code I don't see anything glaring wrong.

    Issue summary appears complete.

    Fingers crossed.

  • πŸ‡³πŸ‡ΏNew Zealand quietone New Zealand

    @godotislate, thanks for the issue summary updated and the test.

    I read the IS, comments and MR (but I have a headache - so really a skim). Of note is the analysis in #5 πŸ› Form state storage changes in #process callbacks are not cached during form rebuild. Needs review . Leaving at RTBC

Production build 0.71.5 2024