Field preprocess makes broad assumptions about data structure, triggering fatal errors in PHP 8

Created on 30 January 2025, 2 months ago

Problem/Motivation

The blazy.module and blazy_layout.module use a generic hook_preprocess_field() processor to determine whether the field formatter used by the field contains the string "blazy." This can lead to problems, as there are assumptions made about not only machine name matching, but also, what the $element['#formatter'] consists of. In PHP 8, using strpos() in this loose way can trigger a PHP fatal error, TypeError: strpos(): Argument #1 ($haystack) must be of type string

For example, a custom field type ( https://www.drupal.org/docs/creating-custom-modules/creating-custom-fiel... ) might apply a delta to the formatter, causing the $element['#formatter'] to be an array.

Proposed resolution

Ensure that the strpos() check is comparing strings to avoid potential fatal errors in PHP 8.

🐛 Bug report
Status

Active

Version

3.0

Component

Code

Created by

🇺🇸United States mark_fullmer Tucson

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

Merge Requests

Comments & Activities

  • Issue created by @mark_fullmer
  • 🇺🇸United States mark_fullmer Tucson
  • Pipeline finished with Success
    2 months ago
    Total: 233s
    #409772
  • 🇮🇩Indonesia gausarts

    Interesting points, thank you.

    Makes no sense to me, unless you help me further with proper documentation.

    > might apply a delta to the formatter, ..
    What you said sounded a personal prediction or assumption to me.

    Mind quoting and pointing it to an exact line of documentation? With proper and documented reasonings, not predictions or assumptions, of course.

    AFAIK, Plugin ID, hence Formatters ID, is always a String, never Integer, as clearly noted here:
    https://git.drupalcode.org/project/drupal/-/blob/11.x/core/lib/Drupal/Co...

    If you were using an integer like deltas, you violated the contract, and integers also makes no sense for plugin IDs as numbers will easily conflict.

    Unless I missed the obvious, of course.

  • 🇮🇩Indonesia gausarts

    For doc purposes, the plugin ID is called here:
    https://git.drupalcode.org/project/drupal/-/blob/11.x/core/lib/Drupal/Co...

    '#formatter' => $this->getPluginId(),

    The contract is set in stone. Any violation of it in custom codes is possible, but not my responsibility.

  • 🇺🇸United States mark_fullmer Tucson

    '#formatter' => $this->getPluginId(),

    The contract is set in stone. Any violation of it in custom codes is possible, but not my responsibility. If you were using an integer like deltas, you violated the contract, and integers also makes no sense for plugin IDs as numbers will easily conflict.

    Thanks for the additional perspective. I was beginning to come to the same conclusion -- that this was likely a problem in custom code. On the surface, it had seemed that the Blazy module's use of strpos() to check if the formatter was of the Blazy family of formatters was an imprecise method -- i.e., why not use a Drupal API method to get the field definition formatter's pluginID, and then check if that formatter extends BlazyMediaFormatterBase? But maybe that seemed like overengineering?

    Thanks for your continuing contribution to the Drupal contrib module ecosystem!

  • 🇮🇩Indonesia gausarts

    > to check if the formatter was of the Blazy family of formatters was an imprecise method...
    String check is not only faster but also more efficient than class check for reasons:

    • We have 6, or so, formatters (blazy_text, blazy_image, blazy_media, etc.) with different base classes. One string "blazy" check is inevitably better than multiple base class lookups or imports.
    • Each base class may be extended by sub-modules, I avoided and was worried to leak Blazy specific features into sub-modules last time. I haven't recapped if such worries are still reasonable, though.

    No Drupal APIs are involved for both checks, IMHO. They are both native.

    Thank you for appreciation :)

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

Production build 0.71.5 2024