假设我想知道传递给 fopen 的字符串是否代表文件路径或有效的包装器(例如 "/home/user/example.txt"
vs "php://input"
)。这是为了从内容中创建一个 tmpfilephp://input
以解决fseek
PHP包装器的限制。
如下所示,file_exists
适用于文件,但不适用于包装器 URI:
var_dump(file_exists("php://input"));
var_dump(file_exists("./exists.txt"));
var_dump(file_exists("./non_existent.txt"));
var_dump(file_exists("php://garbage"));
给
bool(false)
bool(true)
bool(false)
bool(false)
唯一返回 true 的是文件。我找到了 stream_get_wrappers() 但我想避免使检查过于复杂(例如使用字符串比较来尝试检测包装器)。
使用 stream_get_meta_data 似乎也有效,但它需要先调用 fopen,这会阻塞错误日志。
var_dump(stream_get_meta_data(fopen("php://input","r+")));
var_dump(stream_get_meta_data(fopen("./exists.txt","r+")));
var_dump(stream_get_meta_data(fopen("./non_existent.txt","r+")));
var_dump(stream_get_meta_data(fopen("php://garbage","r+")));
生产
array(9) {
["timed_out"]=>
bool(false)
["blocked"]=>
bool(true)
["eof"]=>
bool(false)
["wrapper_type"]=>
string(3) "PHP"
["stream_type"]=>
string(5) "Input"
["mode"]=>
string(2) "rb"
["unread_bytes"]=>
int(0)
["seekable"]=>
bool(true)
["uri"]=>
string(11) "php://input"
}
array(9) {
["timed_out"]=>
bool(false)
["blocked"]=>
bool(true)
["eof"]=>
bool(false)
["wrapper_type"]=>
string(9) "plainfile"
["stream_type"]=>
string(5) "STDIO"
["mode"]=>
string(2) "r+"
["unread_bytes"]=>
int(0)
["seekable"]=>
bool(true)
["uri"]=>
string(10) "./exists.txt"
}
NULL
NULL
我可以使用wrapper_type
由返回的数组中的stream_get_meta_data
,但如果文件或包装器 URI 不存在,它仍会将垃圾放入日志中,我想避免这种情况。
检测我的输入字符串(要传递给 fopen)是否包含现有文件的有效文件路径或有效的 PHP 包装器,或者两者都不包含的最佳方法是什么?
更新:我找到了解决问题的解决方法,但代价是额外fopen
调用。我把这个放在下面的答案中。