Harden the use of unserialize in InlineBlock via allowed classes

Created on 3 October 2023, over 1 year ago

Problem/Motivation

This has been cleared with the security team and approved for public hardening issue due to a lack of exploit path.
InlineBlock uses block_serialized to store the unsaved state of a block content entity whilst a layout is in the temp store.
When the layout is saved this gets unserialized back to a block content entity and saved. This is to prevent updating the block content entity while the layout changes are still in an unsaved state.
It uses unserialize without the allowed_classes option.
This could lead to a gadget chain attack.
There is no exploit path for this in core at the moment as we don't put user data in this field, it is only from a form submission handler.
In πŸ“Œ [PP-1] Expose Layout Builder data to REST and JSON:API Postponed we're looking to add normalizer/serializer support for layout sections. This would permit submission of layout data over POST and that could in turn contain untrusted user input.
As a result this is a hard blocker for that issue.

Steps to reproduce

Proposed resolution

Use the allowed classes option.
We can use the entity type manager to get the class of the block content entity based on the bundle (which is part of the plugin definition)

Remaining tasks

User interface changes

API changes

Data model changes

Release notes snippet

πŸ› Bug report
Status

Active

Version

11.0 πŸ”₯

Component
Layout builderΒ  β†’

Last updated about 6 hours ago

Created by

πŸ‡¦πŸ‡ΊAustralia larowlan πŸ‡¦πŸ‡ΊπŸ.au GMT+10

Live updates comments and jobs are added and updated live.
  • Security improvements

    It makes Drupal less vulnerable to abuse or misuse. Note, this is the preferred tag, though the Security tag has a large body of issues tagged to it. Do NOT publicly disclose security vulnerabilities; contact the security team instead. Anyone (whether security team or not) can apply this tag to security improvements that do not directly present a vulnerability e.g. hardening an API to add filtering to reduce a common mistake in contributed modules.

Sign in to follow issues

Merge Requests

Comments & Activities

  • Issue created by @larowlan
  • last update over 1 year ago
    30,371 pass
  • Status changed to Needs review over 1 year ago
  • Status changed to Needs work over 1 year ago
  • πŸ‡ΊπŸ‡ΈUnited States smustgrave

    With security improvement not sure if the tests could be pushed to a followup but for now tagging for tests.

  • πŸ‡·πŸ‡ΊRussia ilya.no

    ilya.no β†’ made their first commit to this issue’s fork.

  • Status changed to Needs review over 1 year ago
  • πŸ‡·πŸ‡ΊRussia ilya.no

    I've added test, but it's not shown in https://git.drupalcode.org/project/drupal/-/merge_requests/4926/diffs section. I guess, it's due to serialized lines.
    Also I've found, that function getEntity() is called mainly when we create/edit inline block via UI. But there is another place, when unserialize() function is called and it's saveBlockContent(), which is called on presave hook. So I've added separate function in order to avoid code duplication. In test I create node where component in field layout_builder__layout field, which contains serialized data.

  • Pipeline finished with Success
    over 1 year ago
    Total: 16243s
    #69121
  • Status changed to Needs work over 1 year ago
  • πŸ‡ΊπŸ‡ΈUnited States smustgrave

    Went to test this but appears there's conflict applying

    error: missing binary patch data for 'core/modules/layout_builder/tests/src/Kernel/InlineBlockSerializationTest.php'
    error: binary patch does not apply to 'core/modules/layout_builder/tests/src/Kernel/InlineBlockSerializationTest.php'

  • Pipeline finished with Success
    over 1 year ago
    Total: 594s
    #70796
  • Pipeline finished with Canceled
    about 1 year ago
    Total: 80s
    #76262
  • Status changed to Needs review about 1 year ago
  • πŸ‡·πŸ‡ΊRussia ilya.no

    I've updated code, so there is no binary data anymore.

  • Pipeline finished with Success
    about 1 year ago
    Total: 540562s
    #76265
  • Status changed to Needs work about 1 year ago
  • πŸ‡ΊπŸ‡ΈUnited States smustgrave

    Re-ran tests-only feature and it passes without the change to core/modules/layout_builder/src/Plugin/Block/InlineBlock.php

  • πŸ‡«πŸ‡·France andypost

    I think it needs change record as of behavior change - unserialize with allowed classes may fail for existing sites which has a bet on existing way to unserialize. Test surely needs improvement

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

    If you use Paragraphs on a Block Type in Layout Builder, you will get a fatal error:

    Error: Cannot use object of type __PHP_Incomplete_Class as array in Drupal\entity_reference_revisions\Plugin\DataType\EntityReferenceRevisions->setValue() (line 116 of modules/contrib/entity_reference_revisions/src/Plugin/DataType/EntityReferenceRevisions.php).
    

    I traced it back to this patch.

    Unfortunately the Paragraph class never makes it into the allowed_classes list. When it is listed, it works. Example:

      /**
       * Processes serialized block correctly.
       *
       * @return mixed
       *   Result of unserialize() function.
       */
      protected function getUnserializedBlock(): mixed {
        $base_class = $this->entityTypeManager->getDefinition('block_content')->getClass();
        [, $bundle] = explode(PluginBase::DERIVATIVE_SEPARATOR, $this->getPluginId());
        $bundle_class = $this->entityTypeManager->getStorage('block_content')->getEntityClass($bundle);
    
        return unserialize($this->configuration['block_serialized'], ['allowed_classes' => [$base_class, $bundle_class, 'Drupal\paragraphs\Entity\Paragraph']]);
      }
    

    Although I am unaware at the moment how to get "Paragraph" into that list. The block type is block_content with an EntityReferenceRevision field on it pointing at a Paragraph bundle.

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

    Well that fixes the reading, but saving is broken with Paragraphs.

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

    Disregard last comment, the saving issue was due to a change with Layout Builder iFrame Modal.

  • πŸ‡ΊπŸ‡ΈUnited States bradjones1 Digital Nomad Life

    Adding an explicit pointer here to πŸ“Œ Allow field types to control how properties are mapped to and from storage Needs work which would potentially provide a path to not needing \unserialize() and deprecating serialization of PHP objects.

  • First commit to issue fork.
  • Pipeline finished with Failed
    5 months ago
    Total: 152s
    #324446
  • Pipeline finished with Failed
    5 months ago
    Total: 599s
    #324502
  • πŸ‡¬πŸ‡§United Kingdom oily Greater London

    Changed hard-coded uuid's in the kernel test with generated uuid's.

  • πŸ‡¬πŸ‡§United Kingdom oily Greater London

    Test can be reviewed. Set to 'Needs review'.

  • 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 necessarily 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.

  • Pipeline finished with Failed
    5 months ago
    Total: 230s
    #328812
  • πŸ‡¬πŸ‡§United Kingdom oily Greater London

    Fixed PHPCBF. Looking at the comment on the MR, the function is used twice, not once, so seems okay. That leaves a comment that needs to be improved. There is also a CSpell lint error from pipeline.

  • πŸ‡¬πŸ‡§United Kingdom oily Greater London

    Remove 'Needs tests' tag.

  • Pipeline finished with Failed
    5 months ago
    Total: 3364s
    #332603
  • Pipeline finished with Failed
    5 months ago
    Total: 198s
    #332622
  • Pipeline finished with Failed
    5 months ago
    Total: 1315s
    #332627
  • Pipeline finished with Failed
    5 months ago
    Total: 238s
    #332641
  • Pipeline finished with Failed
    5 months ago
    Total: 683s
    #332645
  • πŸ‡¨πŸ‡¦Canada Liam Morland Ontario, CA πŸ‡¨πŸ‡¦

    liam morland β†’ made their first commit to this issue’s fork.

  • πŸ‡¨πŸ‡¦Canada Liam Morland Ontario, CA πŸ‡¨πŸ‡¦

    Code comment improved.

  • Pipeline finished with Failed
    1 day ago
    #463496
Production build 0.71.5 2024