|
php.net | support | documentation | report a bug | advanced search | search howto | statistics | random bug | login |
[2017-01-06 20:12 UTC] kulakov74 at yandex dot ru
Description:
------------
On my Windows PC, I scanned the whole directory structure in C:\ (both from browsers and in CLI mode) and found out that for all of the junctions / symlinks that had been created by the system is_dir() returns false.
According to the manual, "If filename is a symbolic or hard link then the link will be resolved and checked." but this is only the case with the junctions / symlinks I created manually for testing. It might seem the problem is with permissions but I run the script as admin and also I can see the fact that the items in question are junctions by using plain console (cmd.exe) even not as admin, for ex.:
c:\Users>dir /a:l
...
22.08.2013 17:45 <SYMLINKD> All Users [C:\ProgramData]
22.08.2013 17:45 <JUNCTION> Default User [C:\Users\Default]
...
BTW, In Windows, junctions and symlinks are almost the same.
Test script:
---------------
$Dir="C:\Users\All Users";
if (is_dir($Dir)) echo("dir"); else echo("not a dir");
Expected result:
----------------
dir
Actual result:
--------------
not a dir
PatchesPull RequestsHistoryAllCommentsChangesGit/SVN commits
|
|||||||||||||||||||||||||||
Copyright © 2001-2025 The PHP GroupAll rights reserved. |
Last updated: Sat Nov 01 13:00:01 2025 UTC |
Thanks for the effort so far. Maybe I expressed myself wrong. I was talking about the NTFS permissions. Either in GUI by viewing advanced security, or by icacls or similar tools. Here is it on my side, with c:\users\all users php.exe -n -r "$p = 'c:\users\all users'; var_dump(is_dir($p), readlink($p));" bool(true) string(14) "C:\ProgramData" $ icacls "c:\Users\All Users" c:\Users\All Users NT AUTHORITY\SYSTEM:(OI)(CI)(F) BUILTIN\Administrators:(OI)(CI)(F) CREATOR OWNER:(OI)(CI)(IO)(F) BUILTIN\Users:(OI)(CI)(RX) BUILTIN\Users:(CI)(WD,AD,WEA,WA) With c:\users\default user - i cannot even view it in the explorer, "access denied" popup. So that's what it gives php.exe -n -r "$p = 'c:\users\default user'; var_dump(is_dir($p), readlink($p));" Warning: readlink(): readlink failed to read the symbolic link (c:\users\default user), error 5) in Command line code on line 1 bool(true) bool(false) And that's not a wonder, as $ icacls "c:\Users\Default User" c:\Users\Default User Everyone:(DENY)(S,RD) Everyone:(RX) NT AUTHORITY\SYSTEM:(F) BUILTIN\Administrators:(F) Given you say, that a custom junction works as expected, but some particular path don't - it still looks like a permission issue. I might be still wrong, though, but really need a reliable reproducer for the case. Which other tools do you mean? Best way would be to have a synthetic case, that creates an fs object with exact access privileges and shows the issue so then it's repeatable on any system. Thanks.I did many tests and I finally came up with the conclusion the problem is at least in Cyrillic (Russian) characters in directory names (even though I have the latest PHP where the UTF problem has been resolved). I'll give the last lines of my tests for you to make it clear. I'm using russian letters in my tests so make sure you see them. I have created 2 directories for tests: C:\Temp\Dir C:\Temp\Кат (this one is rus) I'm giving 4 combinations for eng/rus: //1. eng -> eng: OK c:\Temp>mklink /j linktoeng C:\Temp\Dir Junction created for linktoeng <<===>> C:\Temp\Dir c:\Temp>php.exe -n -r "$p = 'c:\Temp\linktoeng'; var_dump(is_dir($p), readlink($p));" bool(true) string(11) "C:\Temp\Dir" //2. rus -> eng: OK! c:\Temp>mklink /j рус-ссылка-наангл C:\Temp\Dir Junction created for рус-ссылка-наангл <<===>> C:\Temp\Dir c:\Temp>php.exe -n -r "$p = 'c:\Temp\рус-ссылка-наангл'; var_dump(is_dir($p), readlink($p));" bool(true) string(11) "C:\Temp\Dir" //3. eng -> rus: NO c:\Temp>mklink /j linktorus C:\Temp\Кат Junction created for linktorus <<===>> C:\Temp\Кат c:\Temp>php.exe -n -r "$p = 'c:\Temp\linktorus'; var_dump(is_dir($p), readlink($p));" bool(false) string(14) "C:\Temp\Кат" In this case note that readlink() does perfectly well, also, if I try c:\Temp>php.exe -n -r "$p = 'C:\Temp\Кат'; var_dump(is_dir($p), readlink($p));" bool(true) string(14) "C:\Temp\Кат" is_dir() is correct, so it's is_dir()'s only bug and it only shows with links, cause when I give it the russian dir directly it knows it's a dir! //4. rus -> rus: NO (this is formal, we know it won't work :) c:\Temp>mklink /j рус-ссылка C:\Temp\Кат Junction created for рус-ссылка <<===>> C:\Temp\Кат c:\Temp>php.exe -n -r "$p = 'c:\Temp\рус-ссылка'; var_dump(is_dir($p), readlink($p));" bool(false) string(14) "C:\Temp\Кат" And because I have a russian user name all of the links in my C:\Users fail with PHP. //----------------- As for permissions (ACLs), when you run icacls "c:\Users\All Users" as "All Users" is a link icacls displays ACLs for the target, not for the link. If you try icacls "c:\programdata" you'll see the same ACLs, while in fact "All Users" has the same Deny rule you mentioned (I must say all the built-in links have this Deny rule). So you should use icacls "c:\Users\All Users" /L >>With c:\users\default user - i cannot even view it in the explorer, "access denied" popup The same with me :) BUT if I open console (cmd.exe) I can do this: dir /A:L C:\Users ... <JUNCTION> Default User [C:\Users\Default] and then I can do c:\Users>cd "Default User" c:\Users\Default User> So, Explorer behaves different with links that have equal ACLs, so this is probably where junctions and symlinks differ. It looks like PHP follows the logic used by Explorer and I don't see why Explorer can't access the objects other tools can. Other tools: cmd.exe, Total Commander.Thanks for the further investigation. I still not reproduce the cases 3. and 4. from your post. Here $ mkdir bugs\bug73884\Кат $ icacls bugs\bug73884\Кат bugs\bug73884\Кат BUILTIN\Administrators:(I)(OI)(CI)(F) NT AUTHORITY\SYSTEM:(I)(OI)(CI)(F) BUILTIN\Users:(I)(OI)(CI)(RX) NT AUTHORITY\Authenticated Users:(I)(M) NT AUTHORITY\Authenticated Users:(I)(OI)(CI)(IO)(M) $ mklink /j bugs\bug73884\linktorus bugs\bug73884\Кат Junction created for bugs\bug73884\linktorus <<===>> bugs\bug73884\Кат $ icacls bugs\bug73884\linktorus /L bugs\bug73884\linktorus BUILTIN\Administrators:(I)(OI)(CI)(F) NT AUTHORITY\SYSTEM:(I)(OI)(CI)(F) BUILTIN\Users:(I)(OI)(CI)(RX) NT AUTHORITY\Authenticated Users:(I)(M) NT AUTHORITY\Authenticated Users:(I)(OI)(CI)(IO)(M) $ x64\Release\php.exe -n -r "$p = 'bugs\bug73884\linktorus'; var_dump(is_dir($p), readlink($p));" bool(true) string(54) "C:\php-sdk\php71\vc14\x64\php-src\bugs\bug73884\Кат" $ mklink /j bugs\bug73884\рус-ссылка bugs\bug73884\Кат Junction created for bugs\bug73884\рус-ссылка <<===>> bugs\bug73884\Кат $ icacls bugs\bug73884\рус-ссылка /L bugs\bug73884\рус-ссылка BUILTIN\Administrators:(I)(OI)(CI)(F) NT AUTHORITY\SYSTEM:(I)(OI)(CI)(F) BUILTIN\Users:(I)(OI)(CI)(RX) NT AUTHORITY\Authenticated Users:(I)(M) NT AUTHORITY\Authenticated Users:(I)(OI)(CI)(IO)(M) $ x64\Release\php.exe -n -r "$p = 'bugs\bug73884\рус-ссылка'; var_dump(is_dir($p), readlink($p));" bool(true) string(54) "C:\php-sdk\php71\vc14\x64\php-src\bugs\bug73884\Кат" Used the current dev tree, of course. Hopefully you used some snapshot after the readlink() fix, too. I'm on win10 however, but not sure it is of importance in this case. Also I don't think the username being not ASCII is relevant. Now see, cmd indeed looks like going into c:\users\default user, doing same as you $ cd "Default User" c:\Users\Default User $ dir Volume in drive C is SYSTEM Volume Serial Number is AE0A-76BD Directory of c:\Users\Default User File Not Found But the dir command shows - it cannot read it. Even the dir were empty, it should have shown periods. IMO, what the cd command does, is weird. It in no case looks, like the access no "c:\users\default user" is granted. The points 3. and 4. are intended to work, otherwise the whole UTF-8 thing were useful. You don't have to explain the russian words to me, anyway ;) The fact is - everything path related is converted to UTF-16 internally, and the corresponding W API is used. Also, PHP doesn't automatically elevate any privileges, it always runs with the default security descriptor, so the one corresponding to the current user. An exception from this the impersonation, in that case the impersonated user identity is used. Thereby, if such an issue is real - it doesn't concern only one particular codepage or language, it is most likely present in any other. For some reason, on machines i've tried this particular case doesn't produce issues you have. Used win 10 and server 2012 both with the cp 437. I'd wonder, whether you maybe have some group policies, that force to create files with particular ACLs, etc. Otherwise, it's an awkward situation, as i can't reproduce the issue from your side :( If you're sure, the non ASCII username could be an issue, please try some user with ASCII symbols only. Otherwise, there has to be another factor that we didn't determine yet, causing different behavior. Thanks.>>please retry putting the code into a file As I wrote I did so. I tested with Windows 7 x64 - the bug is still there for both "c:\Users\All Users" and junctions to cyrillic directories. Note that for me is_dir() gives true if the junction itself is cyrillic but the target is latin. The bug shows only if the target is cyrillic, no matter what charset of the link is. I also tried the old PHP 5.5 and it gave me correct result with jun2ru (target is rus), but not for All Users. I compared phpinfo() output of both versions and found a difference: in 5.5 default_charset="" while in 7.1 it's UTF-8. I tried using different values for default_charset and finally got correct result with Windows-1251 (Cyrillic): ini_set('default_charset', 'Windows-1251'); echo(ini_get('default_charset')."\n"); $p='c:\temp\рус2рус'; var_dump(is_dir($p), readlink($p)); This way PHP 7.1 was correct, at the same time when I used UTF-8 / no value / iso-8859-1 it was not. With PHP 5.5 it seems you can use any default_charset. It may be that is_dir() depends on what language is installed in the system, I have Russian installed and for me Windows-1251 worked with PHP 7. "c:\Users\All Users\" doesn't work at all in both versions.