php.net |  support |  documentation |  report a bug |  advanced search |  search howto |  statistics |  random bug |  login
Bug #78296 is_file fails to detect file
Submitted: 2019-07-15 22:53 UTC Modified: 2019-11-25 11:07 UTC
Votes:1
Avg. Score:3.0 ± 0.0
Reproduced:0 of 0 (0.0%)
From: v-altruo at microsoft dot com Assigned: cmb (profile)
Status: Closed Package: Scripting Engine problem
PHP Version: 7.2Git-2019-07-16 (Git) OS: Windows
Private report: No CVE-ID: None
 [2019-07-15 22:53 UTC] v-altruo at microsoft dot com
Description:
------------
Test fails across all builds regardless of opcache on/off. Looked a bit into it, the "is_file($file)" is the part that fails. 

Failed Test Location: ext\standard\tests\file\bug75679.phpt 

Test script:
---------------
<?php
$d = __DIR__;
$Files2Report =
 [
  str_pad ($d . '/' . str_repeat (str_pad ('bug75679_path_259_characters_long_', 100, '_') . '/', 1), 259, '_') => [],
  str_pad ($d . '/' . str_repeat (str_pad ('bug75679_path_260_characters_long_', 100, '_') . '/', 1), 260, '_') => [],
  str_pad ($d . '/' . str_repeat (str_pad ('bug75679_path_261_characters_long_', 100, '_') . '/', 1), 261, '_') => [],
 ];
foreach ($Files2Report as $file => &$Report)
 {
  $Report = ['strlen' => strlen ($file), 'result' => 'nok'];

  if (! is_dir (dirname ($file))) mkdir (dirname ($file), 0777, true);
  if (copy (__FILE__, $file) && is_file ($file))
   {
    $Report['result'] = 'ok';
   }

  print_r ($Report);
 }


?>

Expected result:
----------------
Array
(
    [strlen] => 259
    [result] => ok
)
Array
(
    [strlen] => 260
    [result] => ok
)
Array
(
    [strlen] => 261
    [result] => ok
)
==DONE==

Actual result:
--------------
Array
(
    [strlen] => 259
    [result] => ok
)
Array
(
    [strlen] => 260
    [result] => nok
)
Array
(
    [strlen] => 261
    [result] => nok
)
==DONE==

Patches

Pull Requests

Pull requests:

History

AllCommentsChangesGit/SVN commitsRelated reports
 [2019-07-16 10:52 UTC] cmb@php.net
-Status: Open +Status: Verified -Assigned To: +Assigned To: cmb
 [2019-07-16 10:52 UTC] cmb@php.net
Thanks for reporting!  This is a rather interesting
issue.

At first I have not been able to reproduce the test
failure.  Then I checked the registry key LongPathsEnabled[1]
and found that it was disabled.  After enabling it, the test
failed for me as well.  It seems that the file and directory
management functions (such as CreateFileW()) don't accept
long paths prefixed by \\?\.  The following patch makes the test
pass for LongPathsEnabled set to true:

 win32/ioutil.h | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/win32/ioutil.h b/win32/ioutil.h
index 34104a3f45..7937718766 100644
--- a/win32/ioutil.h
+++ b/win32/ioutil.h
@@ -190,7 +190,7 @@ __forceinline static wchar_t *php_win32_ioutil_conv_any_to_w(const char* in, siz
 	}
 
 	/* Only prefix with long if it's needed. */
-	if (mb_len >= _MAX_PATH) {
+	if (0) {
 		size_t new_mb_len;
 
 		ret = (wchar_t *) malloc((mb_len + PHP_WIN32_IOUTIL_LONG_PATH_PREFIX_LENW + 1) * sizeof(wchar_t));


[1] <https://docs.microsoft.com/en-us/windows/win32/fileio/naming-a-file#enable-long-paths-in-windows-10-version-1607-and-later>
 [2019-07-16 12:26 UTC] cmb@php.net
-Package: Testing related +Package: Scripting Engine problem -PHP Version: 7.4.0alpha3 +PHP Version: 7.2Git-2019-07-16 (Git)
 [2019-07-16 12:26 UTC] cmb@php.net
> It seems that the file and directory management functions (such
> as CreateFileW()) don't accept long paths prefixed by \\?\.

Wrong conclusion.  Actually we're not properly normalizing the
path, if the canonicalized path has the same length as before.
 [2019-07-16 13:05 UTC] cmb@php.net
The following pull request has been associated:

Patch Name: Fix #78296: is_file fails to detect file
On GitHub:  https://github.com/php/php-src/pull/4421
Patch:      https://github.com/php/php-src/pull/4421.patch
 [2019-11-25 11:07 UTC] cmb@php.net
-Status: Verified +Status: Analyzed
 [2019-11-25 11:07 UTC] cmb@php.net
My assessment so far was not quite right.  Actually, the problem
is that[1]:

| File I/O functions in the Windows API convert "/" to "\" as part
| of converting the name to an NT-style name, except when using the
| "\\?\" prefix as detailed in the following sections.

This is already correctly handled on systems supporting
PathCchCanonicalizeEx(), which have the registry setting
LongPathsEnabled disabled, but not for other systems.

Also, mkdir() is affected by this issue if the path is between 248
and 259 characters long.

[1] <https://docs.microsoft.com/en-us/windows/win32/fileio/naming-a-file>
 [2019-11-25 11:29 UTC] cmb@php.net
The following pull request has been associated:

Patch Name: Fix #78296: is_file fails to detect file
On GitHub:  https://github.com/php/php-src/pull/4943
Patch:      https://github.com/php/php-src/pull/4943.patch
 [2019-12-02 10:30 UTC] cmb@php.net
Automatic comment on behalf of cmbecker69@gmx.de
Revision: http://git.php.net/?p=php-src.git;a=commit;h=bb735c9e9e4a2ca2686a141ffe867f60ee0053c3
Log: Fix #78296: is_file fails to detect file
 [2019-12-02 10:30 UTC] cmb@php.net
-Status: Analyzed +Status: Closed
 
PHP Copyright © 2001-2025 The PHP Group
All rights reserved.
Last updated: Wed Jan 22 19:01:31 2025 UTC