Como abrir o arquivo substituído pelo processo do php?

3

Eis o que tentei fazer sozinho:

$ type 1.sh
#!/bin/bash -eu
php -r 'var_dump(file_get_contents($_SERVER["argv"][1]));' -- <(echo 1)
$ ./1.sh
PHP Warning:  file_get_contents(/dev/fd/63): failed to open stream: No such file or directory in Command line code on line 1

Warning: file_get_contents(/dev/fd/63): failed to open stream: No such file or directory in Command line code on line 1
bool(false)

Eu testei em Debian 6 ( php-5.4.14 , bash-4.1.5 ) e Arch Linux ( php-5.4.12 , bash-4.2.42 ).

UPD

$ strace -f -e trace=file php -r 'var_dump(file_get_contents($_SERVER["argv"][1]));' -- <(echo 1)
...
open("/usr/lib/php5/20100525/mongo.so", O_RDONLY) = 3
lstat("/dev/fd/63", {st_mode=S_IFLNK|0500, st_size=64, ...}) = 0
readlink("/dev/fd/63", "pipe:[405116]"..., 4096) = 13
lstat("/dev/fd/pipe:[405116]", 0x7fff5ea44850) = -1 ENOENT (No such file or directory)
lstat("/dev/fd", {st_mode=S_IFLNK|0777, st_size=13, ...}) = 0
readlink("/dev/fd", "/proc/self/fd"..., 4096) = 13
lstat("/proc/self/fd", {st_mode=S_IFDIR|0500, st_size=0, ...}) = 0
lstat("/proc/self", {st_mode=S_IFLNK|0777, st_size=64, ...}) = 0
readlink("/proc/self", "31536"..., 4096) = 5
lstat("/proc/31536", {st_mode=S_IFDIR|0555, st_size=0, ...}) = 0
lstat("/proc", {st_mode=S_IFDIR|0555, st_size=0, ...}) = 0
open("/proc/31536/fd/pipe:[405116]", O_RDONLY) = -1 ENOENT (No such file or directory)
PHP Warning:  file_get_contents(/dev/fd/63): failed to open stream: No such file or directory in Command line code on line 1

Warning: file_get_contents(/dev/fd/63): failed to open stream: No such file or directory in Command line code on line 1
bool(false)

$ strace -f -e trace=file php <(echo 12)
...
open("/usr/lib/php5/20100525/mongo.so", O_RDONLY) = 3
open("/dev/fd/63", O_RDONLY)            = 3
lstat("/dev/fd/63", {st_mode=S_IFLNK|0500, st_size=64, ...}) = 0
readlink("/dev/fd/63", "pipe:[413359]", 4096) = 13
lstat("/dev/fd/pipe:[413359]", 0x7fffa69c3c00) = -1 ENOENT (No such file or directory)
lstat("/dev/fd/63", {st_mode=S_IFLNK|0500, st_size=64, ...}) = 0
readlink("/dev/fd/63", "pipe:[413359]", 4096) = 13
lstat("/dev/fd/pipe:[413359]", 0x7fffa69c19b0) = -1 ENOENT (No such file or directory)
lstat("/dev/fd", {st_mode=S_IFLNK|0777, st_size=13, ...}) = 0
readlink("/dev/fd", "/proc/self/fd"..., 4096) = 13
lstat("/proc/self/fd", {st_mode=S_IFDIR|0500, st_size=0, ...}) = 0
lstat("/proc/self", {st_mode=S_IFLNK|0777, st_size=64, ...}) = 0
readlink("/proc/self", "32214"..., 4096) = 5
lstat("/proc/32214", {st_mode=S_IFDIR|0555, st_size=0, ...}) = 0
lstat("/proc", {st_mode=S_IFDIR|0555, st_size=0, ...}) = 0
2
    
por x-yuri 03.05.2013 / 20:52

3 respostas

2

O problema é que você quer que o php leia a entrada de um descritor de arquivo, mas você o força a ler como um arquivo normal.

Primeiro, tente isto:

$ echo <(ls)
/dev/fd/63

você pode processar ls output lendo /dev/fd/63 . A substituição do processo retornará o file descriptor , que é usado por outro comando para ler sua saída.

No seu caso, você usa $_SERVER["argv"][1] , significando que o php interpretará assim:

file_get_contents(/dev/fd/63)

A partir do manual do php, você pode ver o protótipo de file_get_contents function:

string file_get_contents ( string $filename [, bool $use_include_path =
false [, resource $context [, int $offset = -1 [, int $maxlen ]]]] )

Ops, php irá considerar /dev/fd/63 como arquivo normal aqui, mas é realmente um file descriptor .

Para acessar o descritor de arquivo, você deve usar php://fd , php://fd/63 acessará o conteúdo do descritor de arquivo 63:

$ php -r 'var_dump(file_get_contents("php://".substr($_SERVER["argv"][1],-5)));' -- <(echo test.txt)
string(9) "test.txt
"

Você pode ver, agora o php pode processar o conteúdo em /dev/fd/63 . Mas o nosso propósito é ler o conteúdo do arquivo, que é fornecido via substituição de processo (no meu exemplo, é test.txt ). Eu não sei muito sobre php, então eu adiciono outro file_get_contents :

$ php -r 'var_dump(file_get_contents(file_get_contents("php://".substr($_SERVER["argv"][1],-5))));' -- <(echo -n test.txt)
string(13) "Hello world!
"

Eu uso echo -n para remover a nova linha da saída de eco, caso contrário, o php verá a saída "test.txt \ n".

Nota

Para mais informações sobre como acessar o descritor de arquivos em php, você pode ver aqui .

    
por 15.04.2014 / 05:55
1

Este é o problema:

readlink("/dev/fd/63", "pipe:[405116]"..., 4096) = 13
lstat("/dev/fd/pipe:[405116]", 0x7fff5ea44850) = -1 ENOENT

Sem um bom motivo (IMHO) php tenta obter o nome real do destino do link. Infelizmente, o destino do link não faz parte do sistema de arquivos, portanto, a tentativa de acessar esse nome falha e causa esse erro. O link simbólico só pode ser aberto como tal. Eu considero isso um bug em php . Você poderia usar um FIFO em vez disso:

mkfifo /my/fifo; output_cmd >/my/fifo & php -r ... /my/fifo
    
por 04.05.2013 / 01:17
0

Veja o que acabei usando ...

php -r "var_dump(file_get_contents('php://stdin'));" < <(echo this is a win)

Se você está tentando fazer outras coisas com stdin, isso obviamente não funcionará para você, e isso não é diferente de

php -r "var_dump(stream_get_contents(STDIN));" < <(echo this is a win)

que me leva a imaginar o que você está realmente tentando fazer e por que você está preso em file_get_contents? :)

Referência - link

    
por 26.05.2016 / 09:58