Improve performance of the user.permissions.js script running in /admin/people/permissions.

Created on 29 July 2024, 8 months ago
Updated 18 September 2024, 7 months ago

Problem/Motivation

The permissions page in Drupal (path /admin/people/permissions) has both backend and frontend performance problems that have been documented over the years on multiple tickets (e.g.: πŸ› Improve performance of /admin/people/permissions Needs work , #1203766: With large number of permissions /admin/people/permissions becomes unusable β†’ , #1565704: Core interfaces can go over max_input_vars β†’ among others one might find out there, both closed and open).

I recently came up with an improvement on this issue queue ( πŸ› Improve performance of /admin/people/permissions Needs work ) but upon suggestion by colleagues @nicxvan and @Scott Sawyer over the #contribute Slack channel, I am moving the proposed resolution into a new, separate issue queue. This way we can narrow the scope of the changes being introduced to the front-end realm only. This issue should not clash with any back-end improvements users might come up with in separate tickets.

Here is a brief description of what is being done within that file currently:

  • Detach the entire permissions table from DOM.
  • Manipulate it to create and apply one dummy checkbox per permission, per role.
  • Re-insert the entire table back to the original container.

Although detaching/re-inserting is probably an improvement over the previous version, it still causes considerable lag when loading the permissions page in the context of multiple roles and modules, where potentially thousands of checkboxes are manipulated by the script upon page load. It turns out, though, the heavy process of manipulating all those checkboxes at once is unnecessary and resource-intensive for the browser.

Steps to reproduce

--

Proposed resolution

I am offering a new approach that activates checkbox replacements only when necessary by using IntersectionObserver; the new script does the bare minimum to make sure the expected results are still delivered; here is an overview of what is being delivered:

  • Grab all table rows and find those carrying a checked checkbox in the Authenticated user column.
  • Attach an IntersectionObserver to each one of these rows, and then process them on demand, as they become visible on the page; it may also happen that they will never become visible on the page, thus saving unnecessary client processing to run all those replacements. The script is immediately triggered upon page load to affect only the visible rows for the user.

I ran a very simple test on a local installation of Drupal 11 with console.time() and console.timeEnd() applied to the start and end of the Drupal.behaviors.permissions. This measures how long the script takes to run. Considering a website with 190 modules installed (all included, both core and contrib), and 15 user roles, there was a massive performance gain: I could drop the time it takes for the script to run from ~400ms to ~12ms.

Remaining tasks

A proposed solution is already available to be tested by the community.

User interface changes

There are no user interface changes: the script changes should deliver a positive impact on the loading time of the permissions page with no visual changes for the end-user. The time it took for the script to run dropped from ~ 400ms to ~10ms on Firefox.

Introduced terminology

--

API changes

--

Data model changes

--

Release notes snippet

--

πŸ“Œ Task
Status

Fixed

Version

10.3 ✨

Component
User moduleΒ  β†’

Last updated 1 day ago

Created by

πŸ‡§πŸ‡·Brazil mabho Rio de Janeiro, RJ

Live updates comments and jobs are added and updated live.
  • Performance

    It affects performance. It is often combined with the Needs profiling tag.

Sign in to follow issues

Merge Requests

Comments & Activities

  • Issue created by @mabho
  • This is great.

    I am just updating tags according to the issue tag guidelines β†’ .

  • πŸ‡§πŸ‡·Brazil mabho Rio de Janeiro, RJ
  • πŸ‡§πŸ‡·Brazil mabho Rio de Janeiro, RJ
  • Status changed to Needs review 8 months ago
  • πŸ‡§πŸ‡·Brazil mabho Rio de Janeiro, RJ
  • Pipeline finished with Failed
    8 months ago
    Total: 708s
    #237713
  • Pipeline finished with Success
    8 months ago
    Total: 608s
    #237744
  • πŸ‡§πŸ‡·Brazil mabho Rio de Janeiro, RJ
  • πŸ‡§πŸ‡·Brazil mabho Rio de Janeiro, RJ

    I just attached screenshots of my Firefox browser running the project I mentioned with the Javascript performance audit script, both on the original code (403ms) and the new code (12ms).

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

    I did some manual testing on this.

    I created 15 custom roles and added all permissions to authenticated (lol).
    Stock 11.x branch running a console.time on the permissions page.

    Range 55-75ms (most often was between 66 and 72)

    Checking out this fork and same same script timed.

    Range 36-46 (most often was between 42 and 46 ms)

    Worst case performance improvement in this case is 30% while the best case improvement is: 50%

    Code looks good. I wonder if requestAnimationFrame warrants a comment on what you're doing there.

  • Issue was unassigned.
  • πŸ‡ΊπŸ‡ΈUnited States nicxvan
  • πŸ‡ΊπŸ‡ΈUnited States nicxvan

    I think someone that is more familiar with JS needs to review to approve this.

    Two quick questions:

    requestAnimationFrame does that allow each checkbox in a row to be processed before repainting?
    on the intersectionObserver what happens if you save before it finishes processing? I assume nothing since the checked state still exists and this is just setting up the dummy checkboxes for authenticated right?

  • πŸ‡§πŸ‡·Brazil mabho Rio de Janeiro, RJ

    @nicxvan: when the permissions page is loaded, I am immediately running the script to activate the dummy checkboxes on the checked columns of Authenticated user.

    This will for sure cause the script to slow down the page in a situation where you are checking all checkboxes in the Authenticated user column because the code will run more or less like the previous version where all rows were being processed upon page load.

    In my experience, though, that is something we don't see in real life (authenticated users being granted most of the permissions).

    You bring up a point that makes me think further: why should I replace those checkboxes when the page is loaded? Why not simplify the script even further and let IntersectionObserver do the magic for all rows, including the pre-checked rows?

    I think I will remove this preprocessing of the authenticated rows, and use a one-rule-fits-all using IntersectionObserver. The further performance gain in this case will be minimal, but the code will be even leaner.

  • πŸ‡§πŸ‡·Brazil mabho Rio de Janeiro, RJ

    requestAnimationFrame will 'schedule' the repaint associated with activating/deactivating the checkboxes to the next repaint.

  • Pipeline finished with Failed
    8 months ago
    Total: 114s
    #238480
  • Pipeline finished with Success
    8 months ago
    Total: 434s
    #238489
  • Status changed to RTBC 8 months ago
  • πŸ‡ΊπŸ‡ΈUnited States nicxvan

    Changes look great to me!

  • πŸ‡§πŸ‡·Brazil mabho Rio de Janeiro, RJ

    Updated the issue queue description to reflect the latest changes I had applied to simplify the code even further.

  • Status changed to Needs work 7 months ago
  • πŸ‡«πŸ‡·France nod_ Lille

    Thanks it looks very promising, a few things in the MR, I'm attaching a piece of code for inspiration, it's not cleaned up and we can remove some nesting but it's to give some direction on where I see this going.

  • πŸ‡§πŸ‡·Brazil joaopauloc.dev

    I tested the patch and the performance has been improved a lot.

    Without the patch the script take 51.7ms to run.

    With the patch became 1.4ms.

    Nice!

  • Status changed to Needs review 7 months ago
  • πŸ‡«πŸ‡·France nod_ Lille

    Went with some CSS to manage show/hide of the checkboxes

  • πŸ‡§πŸ‡·Brazil mabho Rio de Janeiro, RJ

    Thank you @_nod and @joaopauloc.dev for the updates and tests on this ticket, I am happy to see my initial effort seems fruitful. I can't test the latest changes by @_nod right now, but will try to do so next week.

  • Pipeline finished with Failed
    7 months ago
    Total: 936s
    #263122
  • πŸ‡ΊπŸ‡ΈUnited States nicxvan

    I fixed the linting issue there is a failure here:

    1)
    Drupal\Tests\user\FunctionalJavascript\UserPermissionsTest::testPermissionCheckboxes
    Error: Call to a member function isVisible() on null

    /builds/issue/drupal-3464530/core/modules/user/tests/src/FunctionalJavascript/UserPermissionsTest.php:66

    ERRORS!
    Tests: 1, Assertions: 5, Errors: 1.

  • πŸ‡«πŸ‡·France nod_ Lille

    I'm guessing the isVisible method has some trouble with the CSS managing the visibility. Might need to write some JS that checks "manually" the visibility of the element.

  • Pipeline finished with Success
    7 months ago
    Total: 921s
    #263130
  • Status changed to RTBC 7 months ago
  • πŸ‡ΊπŸ‡ΈUnited States nicxvan

    Change look great! Intersection observer and css were both great ideas.

    I tested again and it's a huge performance improvement.

    I like the rename from dummy to fake as well.

    My only code contribution was removing an extra line so that tests would run so I think it's ok for me to mark RTBC.

  • πŸ‡§πŸ‡·Brazil mabho Rio de Janeiro, RJ

    @_nod, the changes you are introducing are brilliant, they improve the code even further. My only stylistic comment has to do with the name of the IntersectionObserver variable. Since we are now observing checkboxes, not table rows, I think the variable should be renamed accordingly.

  • Status changed to Needs work 7 months ago
  • πŸ‡ΊπŸ‡ΈUnited States nicxvan
  • Status changed to RTBC 7 months ago
  • πŸ‡«πŸ‡·France nod_ Lille

    renamed the variable and reduced the overall nesting depth

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

    I tested again with just a console time with the same conditions (15 roles).

    Stock 11.x the timer gets called 5 times:

    test: 62ms - timer ended
    test: 1ms - timer ended
    test: 0ms - timer ended
    test: 0ms - timer ended
    test: 1ms - timer ended
    

    This fork it gets called 4 times, but says the timer exists 3 times:

    test: 3ms - timer ended
    Timer β€œtest” already exists.
    Timer β€œtest” already exists.
    Timer β€œtest” already exists.
    

    Also just checking the page load speed it saves about 250ms to load the page entirely.
    1.05 to 1.11seconds on 11.x
    830 to 910ms with this fork

    Assuming the concurrent calls are ok then I think RTBC is the correct status unless we want to rename: CheckedCheckboxObserver to checkedCheckboxObserver.

  • πŸ‡«πŸ‡·France nod_ Lille

    renamed the variable, thanks for the performance tests :)

  • Pipeline finished with Success
    7 months ago
    Total: 556s
    #266153
  • πŸ‡ΊπŸ‡ΈUnited States nicxvan

    I think it's good to go, I confirmed all the comments have been taken care of.

  • πŸ‡§πŸ‡·Brazil cassioalmeida

    Hi,

    This is a great improvement! I did a test with a website with 4 roles and some modules. The permissions table has 535 rows.

    The results are:

    Before: 91.7ms
    After: 1.7ms

    Hopefully this will be merge as soon as possible.

  • πŸ‡§πŸ‡·Brazil mabho Rio de Janeiro, RJ
  • πŸ‡―πŸ‡΄Jordan Qusai Taha Amman

    Thank you, this is working!
    I have created a patch file from the MR.

  • πŸ‡ΊπŸ‡ΈUnited States AaronBauman Philadelphia
  • πŸ‡¬πŸ‡§United Kingdom catch
  • πŸ‡©πŸ‡ͺGermany sun Karlsruhe
  • πŸ‡«πŸ‡·France nod_ Lille

    Porting credit from πŸ› Improve performance of /admin/people/permissions Needs work

    • nod_ β†’ committed 3d9ddf67 on 10.3.x
      Issue #3464530 by nod_, mabho, nicxvan, joaopauloc.dev, cassioalmeida,...
    • nod_ β†’ committed 04d9ff13 on 10.4.x
      Issue #3464530 by nod_, mabho, nicxvan, joaopauloc.dev, cassioalmeida,...
    • nod_ β†’ committed b586e902 on 11.0.x
      Issue #3464530 by nod_, mabho, nicxvan, joaopauloc.dev, cassioalmeida,...
    • nod_ β†’ committed b80bfda1 on 11.x
      Issue #3464530 by nod_, mabho, nicxvan, joaopauloc.dev, cassioalmeida,...
  • Status changed to Fixed 7 months ago
  • πŸ‡«πŸ‡·France nod_ Lille

    Worked on this but we're low on JS reviewers so I'm committing this one.

    Backporting to 10.3.x because of the potential performance improvements

    Committed and pushed b80bfda19d to 11.x and b586e9027c to 11.0.x and 04d9ff136f to 10.4.x and 3d9ddf678d to 10.3.x. Thanks!

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

Production build 0.71.5 2024