POC to redirect fragments/anchors

Created on 4 August 2020, over 4 years ago
Updated 17 July 2023, over 1 year ago

I know this request has been covered and closed in other issues e.g. https://www.drupal.org/project/redirect/issues/2396065 β†’ - this is just a POC for this functionality I thought I'd share in case anyone else finds it useful.

The use case was that I had a government client with a legacy site containing long pages (lots of regulatory content, bylaws etc) that were previously navigated by anchors, that were getting split into multiple shorter pages for better usability. There were lots of anchor links (same/other pages) within content and likely many users who had bookmarked anchor links. The client wanted to implement redirects for these.

It can be done via Javascript - a little hacky since it involves a second page load for the redirect, and won't serve any 3xx headers. It also involves an additional LIKE query by raw source path since (as stated in previous issues) the server can't obtain the requested fragment and so the generated hash in the request subscriber won't match any source path with a fragment. This additional query only runs if a non-fragment redirect is not found. (NB fragment redirects won't work if a non-fragment redirect exists, the latter will always take precedence).

This is a very rough pass and I have no idea what might happen re: loops and any other tests/validation.

✨ Feature request
Status

Needs work

Version

1.0

Component

Code

Created by

πŸ‡¨πŸ‡¦Canada bgilhome Victoria

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.

  • πŸ‡¨πŸ‡¦Canada bgilhome Victoria

    Re-roll from latest 1.x-dev/1.8 and for D9 (update renamed 'user.private_tempstore' service).

  • πŸ‡¨πŸ‡¦Canada bgilhome Victoria

    Improved to use only temp storage and not need to set a 'fragment_redirects' flag in the query string.

  • Open in Jenkins β†’ Open on Drupal.org β†’
    Core: 10.1.x + Environment: PHP 8.1 & MySQL 5.7
    last update over 1 year ago
    Patch Failed to Apply
  • πŸ‡¬πŸ‡§United Kingdom John_B London (UK), Worthing (UK), Innsbruck (Tirol)

    I find no branch where the recent three patches apply.

  • πŸ‡¬πŸ‡§United Kingdom John_B London (UK), Worthing (UK), Innsbruck (Tirol)
  • Open in Jenkins β†’ Open on Drupal.org β†’
    Core: 10.1.x + Environment: PHP 8.1 & MySQL 5.7
    last update over 1 year ago
    PHPLint Failed
  • πŸ‡¬πŸ‡§United Kingdom John_B London (UK), Worthing (UK), Innsbruck (Tirol)

    Another reroll of patch in #7, fixing syntax errors.

  • Open in Jenkins β†’ Open on Drupal.org β†’
    Core: 10.1.x + Environment: PHP 8.1 & MySQL 5.7
    last update over 1 year ago
    36 pass, 18 fail
  • πŸ‡¬πŸ‡§United Kingdom John_B London (UK), Worthing (UK), Innsbruck (Tirol)

    This patch now depends on the fragment being stored in the hash, and this only works if "Retain query string through redirect." is off. If it is on the fragment is not stored. Se πŸ› Redirect hashes created when not retaining the query string RTBC

  • πŸ‡¬πŸ‡§United Kingdom John_B London (UK), Worthing (UK), Innsbruck (Tirol)

    This is a good clean solution to redirecting fragments. It is far better than making a separate module. If the fragment reaches the server we should be able to use it. I see it is a major feature request, and have changed it to "Normal".

    On the other hand, module maintainers are often reluctant to include additional features. New features may mean more work for them. If it is not included in a published release, a solution needing a large patch on multiple files causes a maintenance headache. I'd like to see some positive reaction from the maintainers before putting more work into it.

  • πŸ‡ΊπŸ‡ΈUnited States j_ten_man

    This patch is not currently working. The problem is that the install hook simply updates the database, but it doesn't update the schema definition. I have an update update hook that works for my use case, but it's not the correct solution, I'm sure. So we're going to need to correctly handle this.

    Here's my custom update hook that creates a backup table, drops the old redirect source column, add the new redirect source column, and then update the new column with the backup table.

    /**
     * Add source fragment column.
     */
    function redirect_update_8110() {
      $database = \Drupal::database();
      $database->query('CREATE TABLE {redirect_bak} SELECT * FROM {redirect}');
      $field_storage_definition = \Drupal::entityDefinitionUpdateManager()->getFieldStorageDefinition('redirect_source', 'redirect');
      \Drupal::entityDefinitionUpdateManager()->uninstallFieldStorageDefinition($field_storage_definition);
      $field_storage_definition = \Drupal\Core\Field\BaseFieldDefinition::create('redirect_source')
        ->setLabel(t('From'))
        ->setDescription(t("Enter an internal Drupal path or path alias to redirect (e.g. %example1 or %example2). Fragment anchors (e.g. %anchor) are <strong>not</strong> allowed.", ['%example1' => 'node/123', '%example2' => 'taxonomy/term/123', '%anchor' => '#anchor']))
        ->setRequired(TRUE)
        ->setTranslatable(FALSE)
        ->setDisplayOptions('form', [
          'type' => 'redirect_link',
          'weight' => -5,
        ])
        ->setDisplayConfigurable('form', TRUE);
      \Drupal::entityDefinitionUpdateManager()->installFieldStorageDefinition('redirect_source', 'redirect', 'redirect', $field_storage_definition);
      $database->query('UPDATE {redirect} r JOIN {redirect_bak} rb ON r.rid = rb.rid SET r.redirect_source__path = rb.redirect_source__path, r.redirect_source__query = rb.redirect_source__query');
      $database->query('DROP TABLE {redirect_bak}');
    }
    
Production build 0.71.5 2024