PoStreamReader::readLine() throws an error on module install

Created on 2 August 2022, over 2 years ago
Updated 17 January 2023, about 2 years ago

Problem/Motivation

I've downaloaded a new module and want to enable it, but get fatal error. I've tried to enable several different modules and every time get this error:

$ drush en honeypot 
>  [notice] Checked fr translation for honeypot.
>  [notice] Checked nl translation for honeypot.
>  [error]  The specified file 'temporary://fileie34b1' could not be copied because the destination directory 'translations://' is not properly configured. This may be caused by a problem with file or directory permissions. 
>  [error]  Unable to download translation file https://ftp.drupal.org/files/translations/all/honeypot/honeypot-2.1.1.fr.po. 
>  [warning] fopen(translations://honeypot-2.1.1.fr.po): Failed to open stream: "Drupal\locale\StreamWrapper\TranslationsStream::stream_open" call failed PoStreamReader.php:154
>  [error]  TypeError: fgets(): Argument #1 ($stream) must be of type resource, bool given in fgets() (line 248 of /app/docroot/core/lib/Drupal/Component/Gettext/PoStreamReader.php) #0 /app/docroot/core/lib/Drupal/Component/Gettext/PoStreamReader.php(248): fgets(false)
> #1 /app/docroot/core/lib/Drupal/Component/Gettext/PoStreamReader.php(186): Drupal\Component\Gettext\PoStreamReader->readLine()
> #2 /app/docroot/core/lib/Drupal/Component/Gettext/PoStreamReader.php(217): Drupal\Component\Gettext\PoStreamReader->readItem()
> #3 /app/docroot/core/lib/Drupal/Component/Gettext/PoStreamReader.php(155): Drupal\Component\Gettext\PoStreamReader->readHeader()
> #4 /app/docroot/core/modules/locale/src/Gettext.php(56): Drupal\Component\Gettext\PoStreamReader->open()
> #5 /app/docroot/core/modules/locale/locale.bulk.inc(213): Drupal\locale\Gettext::fileToDatabase(Object(stdClass), Array)
> #6 /app/docroot/core/modules/locale/locale.batch.inc(191): locale_translate_batch_import(Object(stdClass), Array, Object(DrushBatchContext))
> #7 /app/vendor/drush/drush/includes/batch.inc(261): locale_translation_batch_fetch_import('honeypot', 'fr', Array, Object(DrushBatchContext))
> #8 /app/vendor/drush/drush/includes/batch.inc(206): _drush_batch_worker()
> #9 /app/vendor/drush/drush/includes/batch.inc(100): _drush_batch_command('414')
> #10 /app/vendor/drush/drush/src/Drupal/Commands/core/BatchCommands.php(22): drush_batch_command('414')
> #11 [internal function]: Drush\Drupal\Commands\core\BatchCommands->process('414', Array)
> #12 /app/vendor/consolidation/annotated-command/src/CommandProcessor.php(257): call_user_func_array(Array, Array)
> #13 /app/vendor/consolidation/annotated-command/src/CommandProcessor.php(212): Consolidation\AnnotatedCommand\CommandProcessor->runCommandCallback(Array, Object(Consolidation\AnnotatedCommand\CommandData))
> #14 /app/vendor/consolidation/annotated-command/src/CommandProcessor.php(176): Consolidation\AnnotatedCommand\CommandProcessor->validateRunAndAlter(Array, Array, Object(Consolidation\AnnotatedCommand\CommandData))
> #15 /app/vendor/consolidation/annotated-command/src/AnnotatedCommand.php(350): Consolidation\AnnotatedCommand\CommandProcessor->process(Object(Symfony\Component\Console\Output\ConsoleOutput), Array, Array, Object(Consolidation\AnnotatedCommand\CommandData))
> #16 /app/vendor/symfony/console/Command/Command.php(255): Consolidation\AnnotatedCommand\AnnotatedCommand->execute(Object(Drush\Symfony\DrushArgvInput), Object(Symfony\Component\Console\Output\ConsoleOutput))
> #17 /app/vendor/symfony/console/Application.php(1027): Symfony\Component\Console\Command\Command->run(Object(Drush\Symfony\DrushArgvInput), Object(Symfony\Component\Console\Output\ConsoleOutput))
> #18 /app/vendor/symfony/console/Application.php(273): Symfony\Component\Console\Application->doRunCommand(Object(Consolidation\AnnotatedCommand\AnnotatedCommand), Object(Drush\Symfony\DrushArgvInput), Object(Symfony\Component\Console\Output\ConsoleOutput))
> #19 /app/vendor/symfony/console/Application.php(149): Symfony\Component\Console\Application->doRun(Object(Drush\Symfony\DrushArgvInput), Object(Symfony\Component\Console\Output\ConsoleOutput))
> #20 /app/vendor/drush/drush/src/Runtime/Runtime.php(118): Symfony\Component\Console\Application->run(Object(Drush\Symfony\DrushArgvInput), Object(Symfony\Component\Console\Output\ConsoleOutput))
> #21 /app/vendor/drush/drush/src/Runtime/Runtime.php(48): Drush\Runtime\Runtime->doRun(Array, Object(Symfony\Component\Console\Output\ConsoleOutput))
> #22 /app/vendor/drush/drush/drush.php(72): Drush\Runtime\Runtime->run(Array)
> #23 /app/vendor/drush/drush/drush(4): require('/app/vendor/dru...')
> #24 /app/vendor/bin/drush(120): include('/app/vendor/dru...')
> #25 {main}. 

Steps to reproduce

  1. You should have a multilingual site.
  2. Add a new module, e.g: composer install drupal/honeypot.
  3. Enable it: drush en honeypot.

Expected result: module is successfully enabled, no errors.

Actual result: module is enabled, but there was a php error on translations import during module install.

Proposed resolution

After some investigation it turned out that we can modify just 1 line in PoStreamReader::readLine() to fix the issue, see a patch below.

๐Ÿ› Bug report
Status

Postponed: needs info

Version

9.5

Component
Baseย  โ†’

Last updated about 5 hours ago

Created by

๐Ÿ‡ท๐Ÿ‡บRussia i-grou

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

Merge Requests

Comments & Activities

Not all content is available!

It's likely this issue predates Contrib.social: some issue and comment data are missing.

  • Status changed to Needs review about 2 years ago
  • ๐Ÿ‡ง๐Ÿ‡ชBelgium kaduk

    Patch in #7 works for me on Drupal 9.5.2!

  • Status changed to Needs work about 2 years ago
  • ๐Ÿ‡บ๐Ÿ‡ธUnited States smustgrave

    As a bug this will need a test case to show the error.

  • ๐Ÿ‡บ๐Ÿ‡ฆUkraine rollins

    Patch #7 resolved bug for 9.5.7 core version.
    thank you @iperiba92

  • ivnish Kazakhstan

    I had the same problem. I updated translations again here /admin/reports/translations and the problem is gone

  • Thank you for patch #7!

    I started getting this error recently, preventing module installation. The patch allowed me to install again as normal without fatal errors.

  • ๐Ÿ‡ธ๐Ÿ‡ฎSlovenia joco_sp

    In our case it happened on core 9.5.10. The temporary folder was not configured correctly. After adding

    $settings['file_temp_path'] = 'tmp';

    the problem went away.
    So, it's worth checking if your tmp folder is correctly configured and if it has the correct permissions before applying a patch.

  • ๐Ÿ‡ฉ๐Ÿ‡ชGermany Anybody Porta Westfalica

    I just ran into this in a local ddev enviroment and #16 worked. Perhaps it's a file / folder permission issue?

  • ๐Ÿ‡ง๐Ÿ‡ชBelgium flyke

    For me the solution was:
    - visit /admin/config/media/file-system
    - make sure a translations folder is set up there, like: 'sites/default/files/translations'.
    - Save the page

    Now it is fixed, if I enable another module, or drush pmu and then drush en my module again, all works good.
    I should note that when I visited /admin/config/media/file-system, a translations folder was already set up there.
    But the actual folder that is set up there was missing on my local dev environment.
    By saving that page without making any changes, Drupal created the translations folder for me.

  • ๐Ÿ‡จ๐Ÿ‡ฆCanada mahde Vancouver

    I faced this issue after upgrading from Drupal 9.5.11 to Drupal 10.1.5.
    The translations directory exists and the permissions are fine.
    The temporary directory is configured correctly.
    It works only after applying the patch!

  • ๐Ÿ‡ฒ๐Ÿ‡ฆMorocco h_kac

    I faced this issue with drupal 9.3.22 when we upgraded from PHP 7.3 to php 8.1. The comment #21 worked for me. I just saved the page without making any changes and the error was gone. When I do "drush locale-update" no more error are shown.

  • ๐Ÿ‡ซ๐Ÿ‡ทFrance mupsi

    Upgrading from 9.5.11 to 10.1.7, I had the same issue. What fixed it for me was a mix of #21 ๐Ÿ› PoStreamReader::readLine() throws an error on module install Needs work and other things:

    • visiting /admin/config/media/file-system
    • making sure a translations folder is set up there, like: 'sites/default/files/translations'.
    • saving the form
    • running drush locale:check then drush locale:umpdate
    • finally, in /admin/reports/status, clicking on the run cron button, I get an error message saying that it failed. I clicked again and now it's working

    Hope that helps. No patch was needed for me.

  • ๐Ÿ‡ช๐Ÿ‡ธSpain rcodina Barcelona

    Patch on #7 works for me on Drupal 10.1.7. In my case, the error showed up while performing a "drush deploy" in local DDEV environment. So I think it's important to show the error message but avoid interrupting the configuration import process only due to translations import failures.

  • ๐Ÿ‡บ๐Ÿ‡ฆUkraine _shy Ukraine, Lutsk ๐Ÿ‡บ๐Ÿ‡ฆ

    Patch in #7 works for me on Drupal 10.2.3.

  • ๐Ÿ‡ฆ๐Ÿ‡ทArgentina matiasmiranda Yerba Buena

    The patch works, but I don't think it's the correct approach here, we're simply ignoring an error and/or misconfiguration issue.
    In my opinion this should be fixed by configuring correctly the file-system options and directory permissions.

  • ๐Ÿ‡ฌ๐Ÿ‡งUnited Kingdom joachim

    Agree with #28 -- this is not the right fix.

    The right fix would be for the problem to produce a warning that tells users that they need to configure the temp folder.

  • ๐Ÿ‡ง๐Ÿ‡ฌBulgaria pfrenssen Sofia

    Yeah this is not a bug, but a configuration error on the user's behalf. I hit this on a CI environment because the translations folder was missing.

    I guess we can improve the error message.

  • ๐Ÿ‡ง๐Ÿ‡ฌBulgaria pfrenssen Sofia
  • ๐Ÿ‡ง๐Ÿ‡ฌBulgaria pfrenssen Sofia
  • ๐Ÿ‡ฉ๐Ÿ‡ชGermany Anybody Porta Westfalica

    #30 is right and the way to go. Exactly. Still I think for the results it might be seen as (at least ux) bug (WSOD)

    Most important is, that this doesn't cause a WSOD and non-technical users are entirely blocked...

  • ๐Ÿ‡ช๐Ÿ‡ธSpain rcodina Barcelona

    In my case this error blocks the deploy pipeline when installing new modules. I agree it's a server miss configuration, but it's not enough reason to block the deployment. So, apart from improving the error message, we should keep changes from current patch.

  • ๐Ÿ‡ณ๐Ÿ‡ฑNetherlands groendijk

    Came across this issue and used the patch. Not because translation directory wasn't configured properly but just because it couldn't download with a drush locale:update:

    ddev drush locale:update
    >  [error]  Unable to download translation file https://ftp.drupal.org/files/translations/all/admin_toolbar/admin_toolbar-3.5.1.nl.po.
    >  [warning] filesize(): stat failed for //admin_toolbar-3.5.1.nl.po locale.bulk.inc:203
    >  [warning] fopen(translations://admin_toolbar-3.5.1.nl.po): Failed to open stream: "Drupal\locale\StreamWrapper\TranslationsStream::stream_open" call failed PoStreamReader.php:154
    >  [error]  TypeError: fgets(): Argument #1 ($stream) must be of type resource, bool given in fgets() (line 248 of /var/www/html/web/core/lib/Drupal/Component/Gettext/PoStreamReader.php) #0 /var/www/html/web/core/lib/Drupal/Component/Gettext/PoStreamReader.php(248): fgets(false)...

    Patch saved my day. Rather have a broken translation than trying to figure out why it's unable to download translation. Agree with #34.

  • ๐Ÿ‡ช๐Ÿ‡ธSpain guiu.rocafort.ferrer Barcelona

    guiu.rocafort.ferrer โ†’ made their first commit to this issueโ€™s fork.

  • ๐Ÿ‡ช๐Ÿ‡ธSpain guiu.rocafort.ferrer Barcelona

    I created a new branch from the targeted Drupal version 11.x in the issue and applied the previously commited change in the old branch by @immaculate.x. I reproduced the issue with the change, and now the command "drush pm:enable paragraphs" succeeds, but it gives the following errors:

    > [notice] Checked es translation for paragraphs.
    > [notice] Checked it translation for paragraphs.
    > [notice] Checked de translation for paragraphs.
    > [error] Unable to download translation file https://ftp.drupal.org/files/translations/all/paragraphs/paragraphs-8.x-....
    > [warning] filesize(): stat failed for /var/www/html/web/sites/default/files/translations/paragraphs-8.x-1.18.es.po locale.bulk.inc:204
    > [warning] fopen(translations://paragraphs-8.x-1.18.es.po): Failed to open stream: "Drupal\locale\StreamWrapper\TranslationsStream::stream_open" call failed PoStreamReader.php:154
    > [notice] Unable to import translations file: translations://paragraphs-8.x-1.18.es.po
    > [error] Unable to download translation file https://ftp.drupal.org/files/translations/all/paragraphs/paragraphs-8.x-....
    > [warning] filesize(): stat failed for /var/www/html/web/sites/default/files/translations/paragraphs-8.x-1.18.it.po locale.bulk.inc:204
    > [warning] fopen(translations://paragraphs-8.x-1.18.it.po): Failed to open stream: "Drupal\locale\StreamWrapper\TranslationsStream::stream_open" call failed PoStreamReader.php:154
    > [notice] Unable to import translations file: translations://paragraphs-8.x-1.18.it.po
    > [error] Unable to download translation file https://ftp.drupal.org/files/translations/all/paragraphs/paragraphs-8.x-....
    > [warning] filesize(): stat failed for /var/www/html/web/sites/default/files/translations/paragraphs-8.x-1.18.de.po locale.bulk.inc:204
    > [warning] fopen(translations://paragraphs-8.x-1.18.de.po): Failed to open stream: "Drupal\locale\StreamWrapper\TranslationsStream::stream_open" call failed PoStreamReader.php:154
    > [notice] Unable to import translations file: translations://paragraphs-8.x-1.18.de.po
    > [notice] The configuration was successfully updated. 370 configuration objects updated.
    > [error] Message: Failed to save file due to error "/The specified file
    > 'temporary://fileSRCiNi' could not be copied because the destination
    > directory 'translations://' is not properly configured. This may be caused by
    > a problem with file or directory permissions./"
    >
    > [error] Message: Failed to save file due to error "/The specified file
    > 'temporary://filePFV9Fz' could not be copied because the destination
    > directory 'translations://' is not properly configured. This may be caused by
    > a problem with file or directory permissions./"
    >
    > [error] Message: Failed to save file due to error "/The specified file
    > 'temporary://fileWiy0wK' could not be copied because the destination
    > directory 'translations://' is not properly configured. This may be caused by
    > a problem with file or directory permissions./"
    >
    > [error] Message: 6 translation files could not be imported. See the log for details.
    >
    > [notice] Message: The configuration was successfully updated. There are /370/ configuration
    > objects updated.

    Notice that the errors already says that the directory translations might be not properly configured, so i think this should be good enough to not generate a WSOD, and also display the appropiate error messages to the user about the translation folder not being correctly configured.

  • Pipeline finished with Failed
    2 months ago
    Total: 531s
    #410075
  • ๐Ÿ‡บ๐Ÿ‡ธUnited States smustgrave

    Thanks for continuing this forward. Was previously tagged for test coverage which still appear to be needed, so NW for that.

  • Pipeline finished with Failed
    about 2 months ago
    Total: 242s
    #414516
  • Pipeline finished with Success
    about 2 months ago
    Total: 541s
    #414524
  • ๐Ÿ‡ง๐Ÿ‡ฌBulgaria pfrenssen Sofia

    pfrenssen โ†’ changed the visibility of the branch 3301239-postreamreaderreadline-throws-an to hidden.

  • ๐Ÿ‡ง๐Ÿ‡ฌBulgaria pfrenssen Sofia

    pfrenssen โ†’ changed the visibility of the branch 11.x to hidden.

  • ๐Ÿ‡ง๐Ÿ‡ฌBulgaria pfrenssen Sofia

    I've hidden the outdated branches, the branch from @guiu.rocafort.ferrer is the current focus :) Thanks a lot working on this!

  • ๐Ÿ‡ช๐Ÿ‡ธSpain guiu.rocafort.ferrer Barcelona

    Hi @pfrenssen, thanks for your comments.

    I think a good approach would be to throw an exception in open when fopen fails to open the stream.

    Regarding readItem, i would leave the readLine fix still there, in case someone calls the method accidentally. The method might already return NULL if the file has some errors in it or if it is empty, so i think there is no warm in leaving it for now.

    I will add the exception to the open method, and also leave the already implemented fix in readLine, and add a new test to validate the open exception.

    When this issue is closed, we can create a separate issue to review the whole PoStreamReader class and add more exhaustive unit testing to all the methods.

  • Pipeline finished with Failed
    about 2 months ago
    Total: 121s
    #415627
  • Pipeline finished with Failed
    about 2 months ago
    Total: 409s
    #415629
  • ๐Ÿ‡ช๐Ÿ‡ธSpain guiu.rocafort.ferrer Barcelona

    One of the tests is failing because php is throwing a warning message, and it makes phpunit treat fail the test...

  • Pipeline finished with Success
    about 2 months ago
    Total: 348s
    #415814
  • ๐Ÿ‡ช๐Ÿ‡ธSpain guiu.rocafort.ferrer Barcelona

    Thanks @pfrenssen for fixing the failing test.

    I've tried again enabling a module with the translations folder having the wrong permissions and it outputs the following messages:

    >  [notice] Checked de translation for webform.
    >  [notice] Checked es translation for webform.
    >  [notice] Checked ca translation for webform.
    >  [error]  Unable to download translation file https://ftp.drupal.org/files/translations/all/webform/webform-6.3.0-beta1.de.po. 
    >  [warning] filesize(): stat failed for /var/www/html/web/sites/default/files/translations/webform-6.3.0-beta1.de.po locale.bulk.inc:204
    >  [notice] Unable to import translations file: translations://webform-6.3.0-beta1.de.po
    >  [error]  Unable to download translation file https://ftp.drupal.org/files/translations/all/webform/webform-6.3.0-beta1.es.po. 
    >  [warning] filesize(): stat failed for /var/www/html/web/sites/default/files/translations/webform-6.3.0-beta1.es.po locale.bulk.inc:204
    >  [notice] Unable to import translations file: translations://webform-6.3.0-beta1.es.po
    >  [error]  Unable to download translation file https://ftp.drupal.org/files/translations/all/webform/webform-6.3.0-beta1.ca.po. 
    >  [warning] filesize(): stat failed for /var/www/html/web/sites/default/files/translations/webform-6.3.0-beta1.ca.po locale.bulk.inc:204
    >  [notice] Unable to import translations file: translations://webform-6.3.0-beta1.ca.po
    >  [notice] The configuration was successfully updated. 398 configuration objects updated.
    >  [error]  Message: Failed to save file due to error "/The specified file  
    > 'temporary://fileB2wo3R' could not be copied because the destination  
    > directory 'translations://' is not properly configured. This may be caused by  
    > a problem with file or directory permissions./"
    >  
    >  [error]  Message: Failed to save file due to error "/The specified file  
    > 'temporary://fileSQbYRB' could not be copied because the destination  
    > directory 'translations://' is not properly configured. This may be caused by  
    > a problem with file or directory permissions./"
    >  
    >  [error]  Message: Failed to save file due to error "/The specified file  
    > 'temporary://fileUBLW8Q' could not be copied because the destination  
    > directory 'translations://' is not properly configured. This may be caused by  
    > a problem with file or directory permissions./"
    >  
    >  [error]  Message: 6 translation files could not be imported. See the log for details.
    >  
    >  [notice] Message: The configuration was successfully updated. There are /398/ configuration  
    > objects updated.
    > 
    

    The command does not fail, but it also displays enough error messages to point up to the user there is a problem with the translations folder permissions, there are also test for validating the changes made, so i am marking it as needs review.

  • First commit to issue fork.
  • Pipeline finished with Running
    about 2 months ago
    #416838
  • Pipeline finished with Failed
    about 2 months ago
    Total: 102s
    #416858
  • ๐Ÿ‡ง๐Ÿ‡ฌBulgaria pfrenssen Sofia

    Thanks for the updates @VladimirAus, I am really happy with how this is shaping up. Not only is the PHP error solved in a clean way, but we also have a helpful exception that can be caught by calling code and gives useful information about the underlying problem.

    I would like to remove the second test because I think it is unnecessary and would make future maintenance of the class harder. And we have a PHPStan warning. I left comments on the MR.

  • ๐Ÿ‡ฉ๐Ÿ‡ชGermany Anybody Porta Westfalica

    GREAT work @vladimiraus and @pfrenssen!! Thank you! Happy to see this landing soon :)

  • ๐Ÿ‡ช๐Ÿ‡ธSpain guiu.rocafort.ferrer Barcelona

    @vladimiraus, i appreciate the refactor and improving the code, it looks way better this way, but please, next time try not to introduce validation errors, it was passing all the tests before the change. I fixed it already.

    @pfrenssen I guess that now that the open method returns an exception when it cannot open the file, it is way less likely that readItem is called, but it is still not impossible. I think it would be a good idea to allow readItem to throw an exception when the fd is not valid, but i am worried that this might break existing code which is not expecting the method to be thrown.

  • Pipeline finished with Failed
    about 2 months ago
    Total: 453s
    #418279
  • First commit to issue fork.
  • Pipeline finished with Success
    20 days ago
    Total: 436s
    #447620
  • ๐Ÿ‡น๐Ÿ‡ทTurkey orkut murat yฤฑlmaz Istanbul

    When I try to run cron or update the translations, I receive the error below:

    TypeError: fgets(): Argument #1 ($stream) must be of type resource, false given in fgets() (line 248 of /var/www/html/web/core/lib/Drupal/Component/Gettext/PoStreamReader.php).

    I guess, it is related to this error.

  • ๐Ÿ‡บ๐Ÿ‡ฆUkraine v.dovhaliuk

    The patch file based on MR has been provided.

Production build 0.71.5 2024