Super user access policy and the installer

Created on 23 April 2024, 9 months ago
Updated 9 May 2024, 8 months ago

Problem/Motivation

There’s a very interesting thing we need to work out about the super user access policy… I think it has to apply during the installer. Otherwise we’ve got all sorts of problems with creating content etc during the installer because there’s only ever going to be a user 0 and a user 1 during the install … and there might well be no administrator roles.

I also think that the code in standard.install that assigns an admin role to user 1 should be generic. I.e. regardless of install profile, the installer will try to assign user 1 a role with the administrator flag.

This is a major bug because it makes applying recipes that add content during the installer hard.

Steps to reproduce

Install a site with super user access policy off and use the default content module during the installer... no fun.

Proposed resolution

  • Override the security.enable_super_user container param in the installer kernel
  • Add an install task to add an admin role to user 1 (if such a role exists - what do we do if such a role does not?).
  • Remove code that does this from standard.install

Remaining tasks

User interface changes

API changes

Data model changes

Release notes snippet

πŸ› Bug report
Status

Fixed

Version

10.3 ✨

Component
InstallΒ  β†’

Last updated 17 days ago

No maintainer
Created by

πŸ‡¬πŸ‡§United Kingdom alexpott πŸ‡ͺπŸ‡ΊπŸŒ

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

Merge Requests

Comments & Activities

  • Issue created by @alexpott
  • πŸ‡¬πŸ‡§United Kingdom alexpott πŸ‡ͺπŸ‡ΊπŸŒ
  • Pipeline finished with Success
    9 months ago
    Total: 959s
    #154916
  • πŸ‡§πŸ‡ͺBelgium kristiaanvandeneynde Antwerp, Belgium

    Okay so after a lengthy Slack conversation, a few things were brought up:

    Summary

    Whatever solution we come up with, we know that the super user access policy is one day going away and so will the accompanying security.enable_super_user parameter on the container. So whatever we decide to cook up here should preferably not rely on that parameter because it's more code we later on have to deprecate and/or remove again.

    A counterpoint was that the installer will always have to run as a super user but not everyone agreed on that. However, if we do decide to keep that behavior, we could still remove the super user access policy, but move it's logic to an @internal access policy that is only turned on during the installer.

    Another disputed point was also made that any content import, whether via default_content, migration, etc.) failing entity validation without user 1 should probably be fixed in the code that handles said import. We can see an example of this in the migrate module's EntityContentBase destination's validateEntity() method, which uses the account switcher before validating content.

    Finally, the option was raised that the proper solution could be to set importing content to isSyncing(TRUE) and make sure that any import-supporting code such as the migrate module takes said flag into account. This would take time, tough, so for the time being we'd still need a solution for this issue, even if temporary.

    Also keep in mind there still doesn't seem to be a documented consensus on what isSyncing means for content entities, see: πŸ“Œ [PP-2] Better describe how SynchronizableInterface should be used for content entities Needs work

    Personal opinion

    I still as if we should consider not changing any code right now. We advise people using content_import, migrate et al to not disable the super user access policy during their content import. For D11 we change the default behavior for contrib tests to not run with the super user access policy, which core tests already do, meaning modules such as default_content will now start seeing failing tests.

    We then encourage these modules to fix their code and tests in the D11 lifecycle and then we remove the super user access policy altogether in D12, provided we addressed everything in 🌱 [Meta] Plan for deprecating and eventually removing the super user access policy Active by then.

    This way, we don't have to touch the installer at all, we buy at least two more years to work on the isSyncing stuff and contrib gets test fails starting from D11, telling them their code only works because they rely on UID 1.

    All in all, I still feel we can one day have a Drupal core where relying on a super user is a thing of the past. Relying on an admin role being perfectly fine, though.

  • πŸ‡¬πŸ‡§United Kingdom alexpott πŸ‡ͺπŸ‡ΊπŸŒ

    @kristiaanvandeneynde we can't hold off on tackling this - people have test pipelines and development set ups that install from configuration.

  • πŸ‡¬πŸ‡§United Kingdom alexpott πŸ‡ͺπŸ‡ΊπŸŒ

    Addendum to point 5... and such setups need to be able to set the super user access policy to false to be like production.

  • πŸ‡§πŸ‡ͺBelgium kristiaanvandeneynde Antwerp, Belgium

    Having said all of the above, one thing I do very much like about this MR is the removal of the admin role from standard_install() and demo_umami_install() to move it someplace central. All installations should get this admin role out of the box and it should become an opt-out rather than the opt-in it currently is; i.e.: Profiles currently have to remember to ship with an admin role, which feels like poor DX.

    Making it so all installations get an admin role would also be a way of somehow keeping the super user behavior during install: Simply assign said admin role to the current user being used for the install, which should almost always be UID 1.

    Purely based on this the MR might need work because demo_umami_install() still has the same code as standard_install(). Also, both standard and demo_umami also still have the config entity in their config/install folder, we might need to remove those too then.

  • πŸ‡§πŸ‡ͺBelgium kristiaanvandeneynde Antwerp, Belgium

    Okay, how about a second container parameter: security.enable_super_user_during_install

    So you can keep it on during install, but off after that. We change SuperUserAccessPolicyPass to account for it as such:

      public function process(ContainerBuilder $container): void {
        if ($container->getParameter('drupal.is_installing') === TRUE) {
          $enable_policy = $container->getParameter('security.enable_super_user_during_install');
        }
        else {
          $enable_policy = $container->getParameter('security.enable_super_user');
        }
        if ($enable_policy === FALSE) {
          $container->removeDefinition('access_policy.super_user');
          $container->removeAlias('Drupal\Core\Session\SuperUserAccessPolicy');
        }
      }
    
    

    Then in core/lib/Drupal/Core/Installer/NormalInstallerServiceProvider.php, rather than setting security.enable_super_user to TRUE, you set drupal.is_installing to TRUE.

    The bonus with this approach is that we could still make it so contrib tests run with security.enable_super_user_during_install disabled for D11, but keep it enabled in D10.3

  • Pipeline finished with Success
    9 months ago
    Total: 965s
    #155223
  • Status changed to Needs review 9 months ago
  • πŸ‡¬πŸ‡§United Kingdom alexpott πŸ‡ͺπŸ‡ΊπŸŒ

    Profiles currently have to remember to ship with an admin role, which feels like poor DX.

    Well once everything is using recipes they can use standard's admin role recipe for this.

    The MR kind of implements #8 - but no additional container parameter - just an even more limited access policy that only works during the installer.

    The MR has test coverage and re-instates the 10.2.x installer behaviour of having a user 1 with admin access. It consolidates add the admin role to user 1 if an admin role exists.

  • Pipeline finished with Success
    9 months ago
    Total: 987s
    #155244
  • Status changed to Needs work 9 months ago
  • πŸ‡§πŸ‡ͺBelgium kristiaanvandeneynde Antwerp, Belgium

    Well once everything is using recipes they can use standard's admin role recipe for this.

    Oh, that would be perfect.

    I had a quick review and the cache contexts on the new access policy aren't 100% correct. However, when introducing a new access policy, you need to make sure there are no cache entries, so this wouldn't be an issue. Seeing as after the installer the access policy goes away and caches should be cleared (right?), there is also no risk of polluting the cache with the wrong value.

    But from a developer example point of view, we should make sure all of core's access policies show the right cache contexts. Setting to NW for that reason, but could be convinced otherwise.

    The tests look really nice too, but I'd stick to one test and use a data provider to set the code, flag and expected results.
    E.g.:

    $cases['no-super-user-access'] = [
      'install_code' => <<<PHP
          <?php
          function {$this->profile}_install() {
            \$user = \Drupal\user\Entity\User::load(1);
            \Drupal::state()->set('admin_permission_in_installer', \$user->hasPermission('administer software updates'));
          }
          PHP,
      'super-user-policy' => FALSE,
      'expect_user_has_permission_after_install' => FALSE,
      'expect_user_has_permission_during_install' => TRUE,
      'expect_page_shows_no_access_warning' => TRUE,
    ];
    
  • πŸ‡¬πŸ‡§United Kingdom alexpott πŸ‡ͺπŸ‡ΊπŸŒ

    Dataproviders and the InstallerTestBase - too much happens in setUp() and buried in there.

    Cache contexts in the installer are moot - we only use a memory cache for everything - there's no permanent caching.

  • πŸ‡§πŸ‡ͺBelgium kristiaanvandeneynde Antwerp, Belgium

    Alex just informed me there is no cache in the installer so I'm not mega-fussed about the missing cache context. On the one hand it sets a bad example, on the other it's @internal and I don't see us adding caches to the installer soon.

    We should probably also document the plans for the future on 🌱 [Meta] Plan for deprecating and eventually removing the super user access policy Active .

    Some things we discussed were to remove the super user from regular Drupal but keep it in installer until we find a way to enable developers to update their code. This could be further exploring the isSyncing() route discussed in #4. Either way, if we commit the MR here we need to make sure it's kept track of in that meta issue.

  • Status changed to Needs review 9 months ago
  • πŸ‡§πŸ‡ͺBelgium kristiaanvandeneynde Antwerp, Belgium

    Fine, setting to NR again. From my POV it's +1 now, but @catch was also involved in the discussion so we should perhaps have him take a look at this too.

  • Pipeline finished with Success
    9 months ago
    Total: 1170s
    #155297
  • πŸ‡§πŸ‡ͺBelgium kristiaanvandeneynde Antwerp, Belgium

    Looks great to me now. Leaving at NR for:

    I think we need to improve this message. Maybe point to a page of Drupal.org about administering your site with no admin role.

    But no other objections.

  • Pipeline finished with Success
    9 months ago
    Total: 991s
    #155473
  • Status changed to RTBC 9 months ago
  • πŸ‡§πŸ‡ͺBelgium kristiaanvandeneynde Antwerp, Belgium

    Looks good to me. Ubernit on a grammar mistake, but that can be applied on commit.

  • πŸ‡¬πŸ‡§United Kingdom alexpott πŸ‡ͺπŸ‡ΊπŸŒ

    Crediting @Taran2l for a suggestion in slack.

  • πŸ‡¬πŸ‡§United Kingdom longwave UK

    Does this also need a separate change record to tell other install profiles that they don't need to explicitly set a user 1 role any more?

  • πŸ‡¬πŸ‡§United Kingdom catch
  • πŸ‡¬πŸ‡§United Kingdom alexpott πŸ‡ͺπŸ‡ΊπŸŒ

    @longwave great point about the CR - created https://www.drupal.org/node/3443437 β†’

  • Pipeline finished with Success
    9 months ago
    #156250
  • Pipeline finished with Failed
    9 months ago
    Total: 347s
    #156278
  • Pipeline finished with Success
    9 months ago
    Total: 517s
    #156300
    • catch β†’ committed aba2631b on 10.3.x
      Issue #3443037 by alexpott, catch, kristiaanvandeneynde, Taran2L,...
    • catch β†’ committed a2af0aa1 on 11.x
      Issue #3443037 by alexpott, catch, kristiaanvandeneynde, Taran2L,...
  • Status changed to Fixed 9 months ago
  • πŸ‡¬πŸ‡§United Kingdom catch

    Gave this another look over, all the nits I found are in, small enough changes I'm going to go ahead. Committed/pushed to 11.x and cherry-picked to 10.3.x, thanks! Nice way to solve this, simplifies install profiles, doesn't box us in for later.

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

Production build 0.71.5 2024