Create BC redirects for children of changed paths

Created on 2 April 2023, almost 2 years ago
Updated 8 June 2023, over 1 year ago

Problem/Motivation

In Drupal 10.1, we are changing paths and menu navigation for several pages in the Block and Block content modules. In particular,

When we change paths, we provide redirects from the old ones to the new ones. See 🐛 Create a redirect for the new Block types path Fixed . That issue and #2317981 create redirects for the primary paths and also for child paths defined in the block_content module.

This issue is about creating redirect for the child paths provided by other modules, such as content_translation, config_translation, and user. We also want to make it easy for contributed modules to do the same.

Steps to reproduce

  1. Install Drupal 10.1.x with the Standard or Umami profile.
  2. Visit the path /admin/structure/block/block-content/types/manage/basic/permissions
  3. Admire your site's 404 page.

The expected behavior is to be redirected to the new path, /admin/structure/block-content/manage/basic/permissions, with warning messages in the messages area and in the logs.

Proposed resolution

  1. Add the helper class Drupal\Core\Routing\PathChangedHelper so that controller classes can create BC redirects conveniently and in a standard way.
  2. Update existing redirects to use the new helper class.
  3. Add a BC route for entity.block_content_type.edit_form, since that is the Field UI base path.
  4. Use a new route subscriber to add routes to redirect from old child paths to new ones.

Remaining tasks

User interface changes

Redirect to the new path with a warning message instead of giving a 404 (page not found) response.

API changes

Does adding a new helper class count as an API change? Or a new route subscriber?

Data model changes

None

🐛 Bug report
Status

Fixed

Version

10.1

Component
Block content 

Last updated 3 days ago

Created by

🇺🇸United States benjifisher Boston area

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

Comments & Activities

Not all content is available!

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

  • Issue created by @benjifisher
  • 🇺🇸United States benjifisher Boston area

    I am adding one item to the proposed resolution: update existing redirects to use the new trait.

    I considered postponing this issue on 📌 [PP1] Move "Block layout" from Structure to Appearance Postponed , but it would not be hard to copy the helper function from the MR on that issue, so I think the two issues can be done in either order.

  • 🇺🇸United States benjifisher Boston area

    I am adding the issues mentioned in the issue summary as related issues.

  • 🇺🇸United States benjifisher Boston area

    How should we handle the destination query parameter?

    1. Leave it alone.
    2. Fix it (replacing old path with new path).
    3. Remove it.

    I do not think that (2) is viable. For example, the Configure link (before 📌 [PP1] Move "Block layout" from Structure to Appearance Postponed , using the Umami profile) is /en/admin/structure/block/manage/claro_page_title?destination=/en/admin/structure/block/list/claro. The "old path" is /admin/structure/block/manage/{plugin_id} and we need to update "structure" to "appearance".

    If we use (2), then we have to hope that we created a redirect for the destination, if it is also updated. Assuming we did, then there will be another warning in the admin UI.

    I vote for (3).

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

    Yeah 3 sounds best

  • @benjifisher opened merge request.
  • Status changed to Needs work almost 2 years ago
  • 🇺🇸United States benjifisher Boston area

    I started thinking about the single-responsibility principle, and I realized that the helper function in 📌 [PP1] Move "Block layout" from Structure to Appearance Postponed is an abomination. Instead of a trait, we should have a helper class and use it in a controller like this:

      $change_record = 'https://www.drupal.org/node/3320855';
      $helper = new PathChangedHelper($route_match, $request, $change_record);
      $params = $helper->messageParameters();
      $this->logger->warning('A user was redirected from %old_path. This redirect will be removed in a future version of Drupal. Update links, shortcuts, and bookmarks to use %new_path. See %change_record for more information.', $params);
      $this->messenger->addWarning('You have been redirected from %old_path. Update links, shortcuts, and bookmarks to use %new_path.', $params));
      return $helper->redirect();
    

    I have a first draft of the helper function in the MR, so I am setting the issue status to NW. I am updating the first part of the Proposed resolution in the issue summary.

  • 🇺🇸United States benjifisher Boston area

    I think it does not make sense to pass the URL for the change record to the helper function only to have it sent back as one of the parameters. It is also not a great idea for the helper function to decide the placeholder names. So the new version is used like this:

      $helper = new PathChangedHelper($route_match, $request);
      $params = [
        '%old_path' => $helper->oldPath(),
        '%new_path' => $helper->newPath(),
        '%change_record' => 'https://www.drupal.org/node/3320855',
       ];
      $this->logger->warning('A user was redirected from %old_path. This redirect will be removed in a future version of Drupal. Update links, shortcuts, and bookmarks to use %new_path. See %change_record for more information.', $params);
      $this->messenger->addWarning('You have been redirected from %old_path. Update links, shortcuts, and bookmarks to use %new_path.', $params));
      return $helper->redirect();
    

    I wonder whether, for consistency, I should replace the redirect() method with one that just returns the full URL as a string.

  • 🇺🇸United States benjifisher Boston area

    I made the changes suggested on the MR and implemented Step 2 in the proposed resolution.

    The new function bodies are the same number of lines as the old ones, only slightly simpler. The doc blocks and parameter lists are more complicated, since they have to declare the route match and request objects. At this point, it is not clear that the helper class (and the extra level of indirection) is worth it. But the existing controllers are as simple as possible: no arguments, no query parameters. In 📌 [PP1] Move "Block layout" from Structure to Appearance Postponed , we have to deal with both, and I think it is worth handling arguments and query parameters (and stripping destination) in one place.

    One change is that the new code uses Url::getInternalPath() instead of Url::toString(). That means the warning messages will show paths like admin/content/block-content. The old message included at least a leading /; a language prefix like /en on Umami; a directory if Drupal is installed in a subdirectory. Is this a bug or a feature?

    The issue status is still NW for the last step in the proposed resolution and for tests.

  • 🇬🇧United Kingdom AaronMcHale Edinburgh, Scotland

    Grat work on this, thanks Benji!

    I wonder if it's worth bringing the log and warning messages into this helper class. My thinking is that it will help us enforce uniformity and reduce code duplication across core. This is effectively a new pattern that we've established for the admin UI and I'd love to (eventually, when I have time) document this in the user interface standards (or somewhere similar).

    So thinking like that, to me it makes sense to bring as much of the "boilerplate" code that we can into the helper class to help enforce a uniform pattern across routes that use this.

    Maybe even going as far as provide a standard controller class that BC routes can simply point to, rather than each route having to create basically the same controller over and over again.

  • 🇺🇸United States benjifisher Boston area

    @AaronMcHale:

    Have you looked at the helper function in 📌 [PP1] Move "Block layout" from Structure to Appearance Postponed ? It does pretty much what you suggest.

    I think the individual controller has to be responsible for the messages: deprecation, log, user. We may need a warning or an info or none, depending on which path is being changed and the permissions typically required for that path. I think that sort of variation was discussed on Move Custom block library to Content Fixed . Managing that variation in the helper function is clunky. As I said in #7 on this issue, I used the single-responsibility principle as a guide.

    Once the messages are in the individual controller, I think it makes sense to let that function decide what placeholders to use. But if we want to save a few lines of boilerplate, then we could go back to providing a method messageParameters() as in Comment #7.

    Follow-up to my comment in #9: even though it is the same number of lines, the advantage of the current version is that it does not hard-code the route name. That means the same controller can be used by different routes. This will be important when we get to the last step in the proposed resolution.

  • 🇬🇧United Kingdom AaronMcHale Edinburgh, Scotland

    @benjifisher

    On reflection I agree with you that the messages would vary a lot between implementations so it's probably not worth trying to abstract them away, as you said it could get clunky.

  • Open in Jenkins → Open on Drupal.org →
    Environment: PHP 8.1 & MySQL 5.7
    last update almost 2 years ago
    Custom Commands Failed
  • Status changed to Needs review almost 2 years ago
  • 🇺🇸United States benjifisher Boston area

    @AaronMcHale:

    I am glad we agree!

    I rebased on the current 10.1.x, and added three commits (tiny, small, large).

    Adding redirect routes for all the child paths was more work than I expected. I am sure there is room for improvement. It all makes sense to me now, but if it could use some code comments, I am happy to cooperate.

    I am setting the status to NR. On the theory that no good deed goes unpunished, I am sure that someone will point out that it needs tests.

    I am removing this question from the issue summary:

    To be decided: should the new routes be defined in the block_content module, so that all the related BC routes are in the same place? Or should the routes be defined in the modules that create the original routes, as a model for contributed modules to follow?

    The routes for the child paths are defined in a new route subscriber for the block_content module. This is a model that other modules can follow.

  • 🇺🇸United States benjifisher Boston area

    I may not have time to update the MR today, but after sleeping on it, I think there is a much more reliable way to generate the routes that need BC variants. Here is a little test script:

    use Drupal\Core\Routing\RouteBuildEvent;
    use Drupal\field_ui\Routing\RouteSubscriber;
    use Symfony\Component\Routing\RouteCollection;
    
    $router = \Drupal::service('router');
    $collection = $router->getRouteCollection();
    $test_collection = new RouteCollection();
    $base_route = $collection->get('entity.block_content_type.edit_form');
    $test_collection->add('entity.block_content_type.edit_form', $base_route );
    foreach ($test_collection as $r) { echo $r->getPath() . PHP_EOL; }
    $event = new RouteBuildEvent($test_collection);
    $etm = Drupal::entityTypeManager();
    $subscriber = new RouteSubscriber($etm);
    $subscriber->onAlterRoutes($event);
    foreach ($test_collection as $r) { echo $r->getPath() . PHP_EOL; }
    
  • 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 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.

  • Status changed to RTBC almost 2 years ago
  • 🇺🇸United States smustgrave

    Seems @larowlan has already done a review and all threads have been addressed.

    Do wonder if it deserves it's own change record for how contrib can leverage.

  • Open in Jenkins → Open on Drupal.org →
    Environment: PHP 8.1 & MySQL 5.7
    last update almost 2 years ago
    29,283 pass
  • Open in Jenkins → Open on Drupal.org →
    Environment: PHP 8.1 & MySQL 5.7
    last update over 1 year ago
    Build Successful
  • Status changed to Needs review over 1 year ago
  • 🇺🇸United States benjifisher Boston area

    @smustgrave:

    Thanks for the review, but I think it is premature to call this issue RTBC. There have been changes since @larowlan last commented on the MR, and I just now implemented the idea from #14. (It ends up being a few lines longer, but I still think it is better than hard-coding the list as in the earlier versions.)

    I am setting the status back to NR. To be honest, it should probably be NW for tests.

    I made one change to the config_translation module. I will add a note to the proposed resolution explaining that.

  • Status changed to Needs work over 1 year ago
  • 🇺🇸United States smustgrave

    Seems the new changes did cause some failures in the tests.

  • Open in Jenkins → Open on Drupal.org →
    Environment: PHP 8.1 & MySQL 5.7
    last update over 1 year ago
    29,253 pass, 14 fail
  • Status changed to Needs review over 1 year ago
  • 🇺🇸United States benjifisher Boston area

    This should fix at least some of the failing tests. It is the config_translation module again.

  • Status changed to Needs work over 1 year ago
  • 🇺🇸United States smustgrave

    Progress! Still some failures but think it's closer.

  • Open in Jenkins → Open on Drupal.org →
    Environment: PHP 8.1 & MySQL 5.7
    last update over 1 year ago
    29,302 pass
  • Status changed to Needs review over 1 year ago
  • 🇺🇸United States benjifisher Boston area

    I still blame the config_translation module. Instead of fighting it, I am going back to an explicit list of routes provided by that module and reverting the changes I made to that module.

  • Status changed to Needs work over 1 year ago
  • The Needs Review Queue Bot tested this issue. It no longer applies to Drupal core. Therefore, this issue status is now "Needs work".

    This does not mean that the patch 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.

  • Open in Jenkins → Open on Drupal.org →
    Environment: PHP 8.1 & MySQL 5.7
    last update over 1 year ago
    29,371 pass
  • Status changed to Needs review over 1 year ago
  • 🇺🇸United States benjifisher Boston area

    I rebased the MR on the current 10.1.x and resolved the merge conflict. (It was a simple add/add conflict.)

    I also added two unit tests and a kernel test for the helper class. The second unit test and the kernel test are redundant, and I would like a second opinion on which one to keep. I think that the unit test is too brittle, so it is worth the extra 510 ms to run the kernel test.

    Back to NR.

  • Open in Jenkins → Open on Drupal.org →
    Environment: PHP 8.1 & MySQL 5.7
    last update over 1 year ago
    29,370 pass, 2 fail
  • Status changed to Needs work over 1 year ago
  • 🇺🇸United States smustgrave

    Can't seem to rerun the tests. Can you take a look at that failure?

  • Open in Jenkins → Open on Drupal.org →
    Environment: PHP 8.1 & MySQL 5.7
    last update over 1 year ago
    29,356 pass, 2 fail
  • Status changed to Needs review over 1 year ago
  • 🇺🇸United States benjifisher Boston area

    I thought I tested when I made fc02d5d553c268b52cff2586852bb83304107100, but I must have been too tired. I just reverted that commit, and now the test passes locally.

  • Status changed to Needs work over 1 year ago
  • 🇺🇸United States smustgrave

    All threads appear to still be answered.

    Applied patch locally and tested some paths /admin/structure/block/block-content and /admin/structure/block/block-content/types and I still get the warning message as expected.

    Failure appears to be random ckeditor5 and unrelated to this issue.

    Can we update the change record to include the changes here?

    Then think good to mark RTBC.

  • Open in Jenkins → Open on Drupal.org →
    Environment: PHP 8.1 & MySQL 5.7
    last update over 1 year ago
    29,375 pass, 1 fail
  • Open in Jenkins → Open on Drupal.org →
    Environment: PHP 8.1 & MySQL 5.7
    last update over 1 year ago
    29,379 pass
  • 🇺🇸United States benjifisher Boston area

    @smustgrave:

    Thanks for testing.

    Did you consider the question about tests in #24? Do we want to keep the second unit test or the kernel test?

    I updated the change record. While I was doing that, I noticed that one of the redirect controllers referred to 📌 Move block content edit and delete routes under admin/content/block to improve IA for editors and fix breadcrumbs Fixed instead of the change record, so I updated the controller and the related test.

  • Status changed to Needs review over 1 year ago
  • 🇺🇸United States benjifisher Boston area
  • 🇺🇸United States smustgrave

    Personally for me I'm more a fan of kernel tests. So if we are voting that would be mine.

  • Status changed to Needs work over 1 year ago
  • 🇺🇸United States smustgrave

    Just to remove 2nd unit test per #28. Can RTBC after that.

  • Open in Jenkins → Open on Drupal.org →
    Environment: PHP 8.1 & MySQL 5.7
    last update over 1 year ago
    Custom Commands Failed
  • Status changed to RTBC over 1 year ago
  • 🇺🇸United States benjifisher Boston area

    I removed the second unit test. Based on #31, I am setting the status to RTBC.

  • 🇺🇸United States benjifisher Boston area

    When I created this issue, I thought that 📌 [PP1] Move "Block layout" from Structure to Appearance Postponed would be fixed first. That seems unlikely now, so I am rewriting the proposed resolution: this issue adds a new helper class, but does not replace what would have been added by that issue.

  • Open in Jenkins → Open on Drupal.org →
    Environment: PHP 8.1 & MySQL 5.7
    last update over 1 year ago
    Custom Commands Failed
  • Status changed to Needs work over 1 year ago
  • 🇦🇺Australia larowlan 🇦🇺🏝.au GMT+10

    Looks like the MR is failing with PHPCS checks

  • First commit to issue fork.
  • Open in Jenkins → Open on Drupal.org →
    Environment: PHP 8.1 & MySQL 5.7
    last update over 1 year ago
    Custom Commands Failed
  • Open in Jenkins → Open on Drupal.org →
    Environment: PHP 8.1 & MySQL 5.7
    last update over 1 year ago
    29,363 pass, 1 fail
  • Status changed to RTBC over 1 year ago
  • 🇺🇸United States benjifisher Boston area

    @Rassoni:

    Thanks for fixing the CodeSniffer issues. Back again to RTBC, based on #31. That should trigger a retest.

  • 🇺🇸United States benjifisher Boston area

    The failing test is Drupal\BuildTests\Composer\Component\ComponentsIsolatedBuildTest. It does not seem related, and it passes locally, so let's keep the status at RTBC.

  • Open in Jenkins → Open on Drupal.org →
    Environment: PHP 8.1 & MySQL 5.7
    last update over 1 year ago
    29,392 pass
  • Open in Jenkins → Open on Drupal.org →
    Environment: PHP 8.1 & MySQL 5.7
    last update over 1 year ago
    29,392 pass
  • Open in Jenkins → Open on Drupal.org →
    Environment: PHP 8.1 & MySQL 5.7
    last update over 1 year ago
    29,392 pass
  • Open in Jenkins → Open on Drupal.org →
    Environment: PHP 8.1 & MySQL 5.7
    last update over 1 year ago
    29,392 pass
  • Open in Jenkins → Open on Drupal.org →
    Environment: PHP 8.1 & MySQL 5.7
    last update over 1 year ago
    29,392 pass
  • Open in Jenkins → Open on Drupal.org →
    Environment: PHP 8.1 & MySQL 5.7
    last update over 1 year ago
    29,392 pass
  • Open in Jenkins → Open on Drupal.org →
    Environment: PHP 8.1 & MySQL 5.7
    44:08
    42:54
    Running
  • Open in Jenkins → Open on Drupal.org →
    Environment: PHP 8.1 & MySQL 5.7
    last update over 1 year ago
    29,403 pass
  • Open in Jenkins → Open on Drupal.org →
    Environment: PHP 8.1 & MySQL 5.7
    last update over 1 year ago
    29,403 pass
  • Open in Jenkins → Open on Drupal.org →
    Environment: PHP 8.1 & MySQL 5.7
    last update over 1 year ago
    29,404 pass
  • Open in Jenkins → Open on Drupal.org →
    Environment: PHP 8.1 & MySQL 5.7
    last update over 1 year ago
    29,413 pass
  • Open in Jenkins → Open on Drupal.org →
    Environment: PHP 8.1 & MySQL 5.7
    last update over 1 year ago
    29,413 pass
  • Status changed to Needs work over 1 year ago
  • 🇦🇺Australia larowlan 🇦🇺🏝.au GMT+10

    Left some comments on the MR, I think we can push the 'auto detect child paths' idea to a follow up and get this in.

    Fine to self RTBC

    • larowlan committed 8cb28259 on 10.1.x
      Issue #3351750 by benjifisher, Rassoni, smustgrave, larowlan,...
    • larowlan committed 970abeb4 on 11.x
      Issue #3351750 by benjifisher, Rassoni, smustgrave, larowlan,...
  • Status changed to Fixed over 1 year ago
  • 🇦🇺Australia larowlan 🇦🇺🏝.au GMT+10

    On second thoughts, just going to get this in by making the change myself at commit time

    Committed to 11.x and backported to 10.1.x

    Fixed on commit:

    diff --git a/core/lib/Drupal/Core/Routing/PathChangedHelper.php b/core/lib/Drupal/Core/Routing/PathChangedHelper.php
    index 650c110f27f..e8fb996680c 100644
    --- a/core/lib/Drupal/Core/Routing/PathChangedHelper.php
    +++ b/core/lib/Drupal/Core/Routing/PathChangedHelper.php
    @@ -14,7 +14,7 @@
      * compatibility (BC). If the original route is example.route, then the BC route
      * should be named example.route.bc.
      *
    - * The controller for the BC route should have an @deprecated annotation, a
    + * The controller for the BC route should have an deprecated annotation, a
      * deprecation error, and type declarations for any parameters that are required
      * for access checking. Then the body of the controller can use the methods
      * provided by this class:
    
    

    Opened 📌 Attempt to auto-detect child paths of deprecated routes and add redirects Active for follow-up

  • 🇦🇺Australia larowlan 🇦🇺🏝.au GMT+10
  • 🇺🇸United States benjifisher Boston area

    @larowlan:

    I am glad to see this get in!

    Can you add a link to the follow-up issue you created?

    I think we need to include $block_content in the parameter list, with its type hint, so that we can have the same access checks on the BC route as on the original. In other words, it is used implicitly.

    OTOH, maybe we should not do access checks at all on the BC route. (shrug)

    The doc block now says "an deprecated". Oh, well.

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

    Follow-up is in #42 📌 Attempt to auto-detect child paths of deprecated routes and add redirects Active

    I'll hotfix the deprecated, I assume the `an` was for reading it as `an at deprecated` and now that makes no sense without the `at`

    • larowlan committed b258fd36 on 11.x
      Issue #3351750 by benjifisher, Rassoni, smustgrave, larowlan,...
    • larowlan committed 34d09bf2 on 10.1.x
      Issue #3351750 by benjifisher, Rassoni, smustgrave, larowlan,...
  • 🇦🇺Australia larowlan 🇦🇺🏝.au GMT+10

    I hotfixed the an > a

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

Production build 0.71.5 2024