$stream_wrapper->getDirectoryPath() is used all over core, but is not defined in StreamWrapperInterface

Created on 21 June 2010, almost 15 years ago
Updated 5 February 2025, about 2 months ago

drupal_tempnam() and file_directory_path() calls $wrapper->getDirectoryPath(), but that method isn't defined in the DrupalStreamWrapperInterface interface but only in the DrupalLocalStreamWrapper abstract class.

So drupal_tempnam('foo://', 'bar') will generate a fatal error, if the foo prefix is mapped to a wrapper that is not a subclass of DrupalLocalStreamWrapper.

Options:
Remove the usage of the method from outside the streamWrapper classes
Add the method to the interface (with appropriate methods for signaling a streamWrapper does not have a local path).

πŸ› Bug report
Status

Active

Version

11.0 πŸ”₯

Component

file system

Created by

πŸ‡©πŸ‡°Denmark c960657

Live updates comments and jobs are added and updated live.
  • Needs issue summary update

    Issue summaries save everyone time if they are kept up-to-date. See Update issue summary task instructions.

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.

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

    I'm going to re-open this, the method is used in core requiring third party streamWrappers to implement even though it is not on interface.

    modules/system/src/Routing/AssetRoutes.php:    $directory_path = $this->streamWrapperManager->getViaScheme('assets')->getDirectoryPath(); // @see 3504164
    modules/image/src/Routing/ImageStyleRoutes.php:    $directory_path = $this->streamWrapperManager->getViaScheme('public')->getDirectoryPath();
    modules/image/src/PathProcessor/PathProcessorImageStyles.php:    $directory_path = $this->streamWrapperManager->getViaScheme('public')->getDirectoryPath();
    modules/image/src/Entity/ImageStyle.php:      $directory_path = $stream_wrapper_manager->getViaUri($uri)->getDirectoryPath();
    lib/Drupal/Core/Updater/Updater.php:    return \Drupal::service('stream_wrapper_manager')->getViaScheme('temporary')->getDirectoryPath();
    lib/Drupal/Core/File/FileSystem.php:      if ($filename = tempnam($wrapper->getDirectoryPath(), $prefix)) {
    modules/package_manager/src/PathExcluder/SiteFilesExcluder.php:        $path = $wrapper->getDirectoryPath(); // This has a LocalStreams check, however that makes an assumption all replacment local streamWrappers will inherit from LocalStreams which is bad assumption.
    

    Since not all streamWrappers will have a directoryPath my suggestion would instead to be that we remove the method from the public usage and find alternative solutions Core should not assume a streamWrapper path is local. This might create some initial pain, however longer term if we get into the mindset to not assume local disk access for anything expect local file paths we will be better aligned to the intended usage of streamWrappers.

    Leaving as 'file system' for component as initial decisions need to come from there, eventually though this may need to expand to 'base system' or similar.

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

    Providing an explicit link to πŸ› AssetRoutes::routes() assumes stream wrapper implements ::getDirectoryPath() Active which illustrates this well.

  • πŸ‡¦πŸ‡ΊAustralia kim.pepper πŸ„β€β™‚οΈπŸ‡¦πŸ‡ΊSydney, Australia

    Could we start by looking at how we would change \Drupal\Core\File\FileSystem::tempnam() to not use ::getDirectoryPath()?

    Do we need more strict type checks for LocalStream?

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

    Could we start by looking at how we would change \Drupal\Core\File\FileSystem::tempnam() to not use ::getDirectoryPath()?

    ::realpath() of the scheme would accomplish the same outcome.

    Do we need more strict type checks for LocalStream?

    If using ::realpath() no, just validate the return is not FALSE. This allows classes that may implement local storage, however not extend LocalStream (such as extending a 3rd party framework to interface with core) to still function.

  • πŸ‡¦πŸ‡ΊAustralia kim.pepper πŸ„β€β™‚οΈπŸ‡¦πŸ‡ΊSydney, Australia

    OK. I created an issue πŸ“Œ FileSystem::tempnam() should use realpath() instead of getDirectoryPath() Active so we can tackle that first.

Production build 0.71.5 2024