BreadcrumbManager ignores cacheability when no builders apply

Created on 23 October 2024, 6 months ago

Problem/Motivation

Since 🐛 BreadcrumbBuilder::applies() mismatch with cacheability metadata Active , the cacheability of whether breadcrumb builders should apply is now gathered. That's great - but in custom cases where it's possible for no breadcrumb builder to apply, that cacheability is not merged.

Currently core's path-based breadcrumb builder service that the system module provides (system.breadcrumb.default) always applies, but it's a service that can be decorated or removed.

Steps to reproduce

1. Remove the path-based breadcrumb builder service, e.g. by decorating it with a class that doesn't always return TRUE from its applies() method, or by using a service provider's alter method.
2. Visit a path on which it, and no other builder, would apply (e.g. a 404 page).
3. The breadcrumb is now cached as empty, and with no cacheability metadata.
4. Visit another path on which other breadcrumb builders might apply - the cached empty breadcrumb is now used because there was no cache context etc on it, so core assumes it is valid to use.

Proposed resolution

Merge any cacheability immediately after calling every breadcrumb builder's applies() method.

Remaining tasks

Provide test, fix and review.

User interface changes

None

Introduced terminology

None

API changes

None

Data model changes

None

Release notes snippet

🐛 Bug report
Status

Active

Version

11.0 🔥

Component

base system

Created by

🇬🇧United Kingdom james.williams

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

Merge Requests

Comments & Activities

  • Issue created by @james.williams
  • 🇬🇧United Kingdom james.williams

    I've opened MR !9913 to demonstrate what probably needs to change; though a test would be needed anyway. Probably using a test module with a service provider to remove the path-based breadcrumb builder, or decorate it with one that won't always apply. The test would then load a page where no breadcrumb builders apply, then another where one should - an assertion should expect breadcrumbs to appear, but the test should fail without this fix as no breadcrumbs would show.

    I need to focus on some other work for now, so I reckon maybe that's something a novice could pick up?

  • Pipeline finished with Failed
    6 months ago
    Total: 3045s
    #318023
  • First commit to issue fork.
  • 🇮🇳India arunkumark Coimbatore

    @james.williams
    I made a small change in the Cache dependency for the Breadcrumb. To handle the invalid Breadcrumb exception. Kindly check is the made will resolve your issue.

    Currently moving to Needs Review.

  • Pipeline finished with Success
    5 months ago
    Total: 2406s
    #323246
  • 🇺🇸United States smustgrave

    Was previously tagged for tests which still appear to be needed.
    Please read the tests before moving to review.

  • Pipeline finished with Failed
    5 months ago
    Total: 8434s
    #323858
  • 🇬🇧United Kingdom james.williams

    The new test passes without the fix, which can't be right. I don't really understand what that new test is actually doing - I'm not sure it really represents the problem at hand.

    We've also got an issue in that existing core tests fail with the change, with this warning:

    Trying to overwrite a cache redirect with one that
    has nothing in common, old one at address "languages:language_interface,
    theme, user.permissions" was pointing to "url.path.parent,
    url.path.is_front", new one points to "route".

    Although the test itself doesn't look relevant, that does sound like we've adjusted cacheability metadata in some way that hasn't been expected. More work needed on all fronts!

  • 🇧🇪Belgium kristiaanvandeneynde Antwerp, Belgium

    I think we should leave the foreach as is, but change this bit:

          if ($breadcrumb instanceof Breadcrumb) {
            $context['builder'] = $builder;
            $breadcrumb->addCacheableDependency($cacheable_metadata);
            break;
          }
          else {
            throw new \UnexpectedValueException('Invalid breadcrumb returned by ' . get_class($builder) . '::build().');
          }
    

    To drop the addCacheableDependency() call and move that below the loop.

    The reason it's failing now is because you're adding the cacheable metadata to a breadcrumb that gets replaced by build() further down the loop.

  • Pipeline finished with Success
    5 months ago
    Total: 1688s
    #330976
  • 🇬🇧United Kingdom james.williams

    @kristiaanvandeneynde thank you for spotting that and pointing it out, yes! Next step then is to get the test demonstrating the actual problem. It should fail without the changes, but pass with them - whereas it's now passing in both situations.

  • Pipeline finished with Failed
    5 months ago
    Total: 200s
    #335358
  • Pipeline finished with Failed
    5 months ago
    Total: 2093s
    #335363
  • Pipeline finished with Failed
    5 months ago
    Total: 1160s
    #335397
  • Pipeline finished with Failed
    5 months ago
    Total: 695s
    #335407
  • Pipeline finished with Failed
    5 months ago
    Total: 159s
    #335478
  • Pipeline finished with Success
    5 months ago
    Total: 465s
    #335480
  • 🇬🇧United Kingdom james.williams

    Finally got a working test, failing without the fix to demonstrate that it is needed, and passing with it!

    Silly me for thinking that this could have been suitable for novice... I took hours fighting to get this to work! So many dark corners in the test system.... e.g. you can't assume the system module is enabled, and it's best to pass proper Url objects to $this->drupalGet(). Ah well, now I know :)

  • 🇺🇸United States smustgrave

    Left some questions/comment on MR, going to leave in review for other eyes.

  • Pipeline finished with Success
    5 months ago
    Total: 912s
    #336076
  • Pipeline finished with Failed
    5 months ago
    Total: 6701s
    #336038
  • 🇧🇪Belgium kristiaanvandeneynde Antwerp, Belgium

    Few minor nitpicks but looks good to me otherwise.

  • Pipeline finished with Success
    5 months ago
    Total: 1270s
    #336248
  • 🇬🇧United Kingdom james.williams

    Thanks for the reviews! Should be good again now 👍

  • 🇧🇪Belgium kristiaanvandeneynde Antwerp, Belgium

    LGTM

    Pipelines seem to be fine aside from the test-only one, which makes sense.

  • 🇳🇿New Zealand quietone

    All questions have been answered here and the basics are all in order. I updated credit. Leaving at RTBC.

  • 🇦🇺Australia larowlan 🇦🇺🏝.au GMT+10

    Left one comment on the MR, fine to self RTBC after adapting

  • 🇧🇪Belgium kristiaanvandeneynde Antwerp, Belgium

    Left a comment why we don't need the instanceof check. So back to RTBC

    • larowlan committed a8556f6f on 10.4.x
      Issue #3482691 by james.williams, arunkumark, kristiaanvandeneynde,...
    • larowlan committed 92e0fdbe on 10.5.x
      Issue #3482691 by james.williams, arunkumark, kristiaanvandeneynde,...
    • larowlan committed 677bd5a1 on 11.1.x
      Issue #3482691 by james.williams, arunkumark, kristiaanvandeneynde,...
    • larowlan committed 78bbe4db on 11.x
      Issue #3482691 by james.williams, arunkumark, kristiaanvandeneynde,...
    • larowlan committed 04bd5613 on 10.4.x
      Revert "Issue #3482691 by james.williams, arunkumark,...
  • Status changed to RTBC 3 months ago
  • 🇦🇺Australia larowlan 🇦🇺🏝.au GMT+10

    Committed to 11.x and backported to 10.5.x and 11.1.x

    Originally backported to 10.4.x but then reverted because this doesn't meet the requirements for maintenance minors

    Thanks all

  • Automatically closed - issue fixed for 2 weeks with no activity.

Production build 0.71.5 2024