PHP file_get_contents () retorna “falhou ao abrir o stream: solicitação HTTP falhou!”


91

Estou tendo problemas para chamar um url do código PHP. Preciso chamar um serviço usando uma string de consulta do meu código PHP. Se eu digitar o url em um navegador, ele funciona bem, mas se eu usar o arquivo-get-contents () para fazer a chamada, obtenho:

Aviso: file-get-contents (http: // ....) falhou ao abrir o stream: solicitação de HTTP falhou! HTTP / 1.1 202 aceito em ...

O código que estou usando é:

$query=file_get_contents('http://###.##.##.##/mp/get?mpsrc=http://mybucket.s3.amazonaws.com/11111.mpg&mpaction=convert format=flv');
echo($query);

Como eu disse - ligue do navegador e funciona bem. Alguma sugestão?

Também tentei com outro url, como:

$query=file_get_contents('http://www.youtube.com/watch?v=XiFrfeJ8dKM');

Isso funciona bem ... será que o url que eu preciso ligar tem um segundo http://?

Respostas:


110

Tente usar cURL.

<?php

$curl_handle=curl_init();
curl_setopt($curl_handle, CURLOPT_URL,'http://###.##.##.##/mp/get?mpsrc=http://mybucket.s3.amazonaws.com/11111.mpg&mpaction=convert format=flv');
curl_setopt($curl_handle, CURLOPT_CONNECTTIMEOUT, 2);
curl_setopt($curl_handle, CURLOPT_RETURNTRANSFER, 1);
curl_setopt($curl_handle, CURLOPT_USERAGENT, 'Your application name');
$query = curl_exec($curl_handle);
curl_close($curl_handle);

?>

10
Isso é muito complicado quando o problema real está no "e" comercial.
Christian

1
@Christian, você pode explicar?
vonUbisch

4
Nem todo mundo tem (mas deveria) cURL instalado. cURL certeza é muitas vezes mais rápido, mas file_get_contents não é que lento mal, e não requer que você se lembra de todas as opções sempre que você usá-lo.
Christian

1
Isso CURLOPT_USERAGENTfoi muito importante no meu caso, obrigado!
emotality

Eu acredito que meu problema era com o tempo limite e isso resolveu. Obrigado!
evadecaptcha

28

Tive o mesmo erro que o OP teve, e esse foi o meu problema - espaços nos argumentos. urlencode()nos parâmetros GET resolveu o problema.
Walt W

Se você não quiser usar a solução CURL, esta funciona! Use o URLecnode nos parâmetros.
Aaron Gong

Esta é a solução real para este problema.
Henrik Petterson

23
<?php

$lurl=get_fcontent("http://ip2.cc/?api=cname&ip=84.228.229.81");
echo"cid:".$lurl[0]."<BR>";


function get_fcontent( $url,  $javascript_loop = 0, $timeout = 5 ) {
    $url = str_replace( "&amp;", "&", urldecode(trim($url)) );

    $cookie = tempnam ("/tmp", "CURLCOOKIE");
    $ch = curl_init();
    curl_setopt( $ch, CURLOPT_USERAGENT, "Mozilla/5.0 (Windows; U; Windows NT 5.1; rv:1.7.3) Gecko/20041001 Firefox/0.10.1" );
    curl_setopt( $ch, CURLOPT_URL, $url );
    curl_setopt( $ch, CURLOPT_COOKIEJAR, $cookie );
    curl_setopt( $ch, CURLOPT_FOLLOWLOCATION, true );
    curl_setopt( $ch, CURLOPT_ENCODING, "" );
    curl_setopt( $ch, CURLOPT_RETURNTRANSFER, true );
    curl_setopt( $ch, CURLOPT_AUTOREFERER, true );
    curl_setopt( $ch, CURLOPT_SSL_VERIFYPEER, false );    # required for https urls
    curl_setopt( $ch, CURLOPT_CONNECTTIMEOUT, $timeout );
    curl_setopt( $ch, CURLOPT_TIMEOUT, $timeout );
    curl_setopt( $ch, CURLOPT_MAXREDIRS, 10 );
    $content = curl_exec( $ch );
    $response = curl_getinfo( $ch );
    curl_close ( $ch );

    if ($response['http_code'] == 301 || $response['http_code'] == 302) {
        ini_set("user_agent", "Mozilla/5.0 (Windows; U; Windows NT 5.1; rv:1.7.3) Gecko/20041001 Firefox/0.10.1");

        if ( $headers = get_headers($response['url']) ) {
            foreach( $headers as $value ) {
                if ( substr( strtolower($value), 0, 9 ) == "location:" )
                    return get_url( trim( substr( $value, 9, strlen($value) ) ) );
            }
        }
    }

    if (    ( preg_match("/>[[:space:]]+window\.location\.replace\('(.*)'\)/i", $content, $value) || preg_match("/>[[:space:]]+window\.location\=\"(.*)\"/i", $content, $value) ) && $javascript_loop < 5) {
        return get_url( $value[1], $javascript_loop+1 );
    } else {
        return array( $content, $response );
    }
}


?>

5
Obrigado pelo código, parece que você o obteve de: php.net/manual/en/ref.curl.php O principal problema é que a chamada de função get_urldeve realmente ser, get_fcontentjá que você alterou o nome da própria função. Na verdade, esta é uma chamada de função recursiva que tenta novamente obter o conteúdo do URL alterando alguns parâmetros.
SSH Em

você entendeu! estava tentando https e estava sendo rejeitado. você acertou em cheio. ATUALIZADO;)
tony gil

21

file_get_contents()utiliza os fopen()invólucros, portanto, não pode acessar URLs por meio da allow_url_fopenopção dentro do php.ini.

Você precisará alterar seu php.ini para ativar esta opção ou usar um método alternativo, ou seja, cURL - de longe o mais popular e, para ser honesto, a maneira padrão de realizar o que você está tentando fazer.


Esqueça, acabei de notar que ele disse que file_get_contents()trabalhava em uma URL diferente. No entanto, esta ainda é uma boa dica para outras pessoas com esse problema.
Michael Wales

sim, eu olhei para outras opções e vi que cURL é o caminho padrão a seguir, mas tentei fazer isso porque era mais fácil e funcionava com outros urls, acho que tenho que reiniciar o apache para instalar o cURL? e não consigo descobrir como fazer isso (outra pergunta) ... obrigado por sua resposta rápida
indefinido

11

Basicamente, você deve enviar algumas informações com a solicitação.

Tente isto,

$opts = array('http'=>array('header' => "User-Agent:MyAgent/1.0\r\n")); 
//Basically adding headers to the request
$context = stream_context_create($opts);
$html = file_get_contents($url,false,$context);
$html = htmlspecialchars($html);

Isso funcionou para mim


2
Isso funcionou para mim também! Parece que precisava de um agente de usuário
bryan

Se o url parecer codificado corretamente, esse pode muito bem ser o problema. Alguns sites irão restringir em 'Agente do Usuário', talvez 'Aceitar' também.
Jahmic

1
super resposta. vote para promover este ao topo. tudo em php e funcionou para solicitação https também. apenas duas linhas extras de código. e se o servidor compartilhado fornecer curl? isso é muito útil.
ndasusers

9

Percebi que seu URL contém espaços. Acho que geralmente é uma coisa ruim. Tente codificar o URL com

$my_url = urlencode("my url");

e então ligando

file_get_contents($my_url);

e veja se você tem melhor sorte.


4

Eu tenho um problema semelhante, analisei o url do youtube. O código é;

$json_is = "http://gdata.youtube.com/feeds/api/videos?q=".$this->video_url."&max-results=1&alt=json";
$video_info = json_decode ( file_get_contents ( $json_is ), true );     
$video_title = is_array ( $video_info ) ? $video_info ['feed'] ['entry'] [0] ['title'] ['$t'] : '';

Então eu percebo que $this->video_urlincluem os espaços em branco. Eu resolvi isso usando trim($this->video_url).

Talvez isso te ajude . Boa sorte


3

Não tenho certeza sobre os parâmetros (mpaction, format), se eles são especificados para a página amazonaws ou ##. ##.

Tente urlencode () o url.


obrigado - esses são parâmetros para a instância mediaplug. Se eu urlencode o url ainda não funciona - recebo um url muito truncado no erro ... ??
indefinido

Você deve codificar apenas a string de parâmetro: "convert format" deve ser "convert% 20format" (ou, alternativamente, "convert + format").
bobince

2
$query=file_get_contents('http://###.##.##.##/mp/get?' . http_build_query(array('mpsrc' => 'http://mybucket.s3.amazonaws.com/11111.mpg&mpaction=convert format=flv')));

2

Eu tenho um problema semelhante.

Devido ao tempo limite!

O tempo limite pode ser indicado assim:

$options = array(
    'http' => array(
        'header'  => "Content-type: application/x-www-form-urlencoded\r\n",
        'method'  => "POST",
        'content' => http_build_query($data2),
        'timeout' => 30,
    ),
);
$context = stream_context_create($options); $retour =
$retour = @file_get_contents("http://xxxxx.xxx/xxxx", false, $context);

rs isso resolveu meu problema, muito obrigado!
Niggo

Ao utilizar nosso site, você reconhece que leu e compreendeu nossa Política de Cookies e nossa Política de Privacidade.
Licensed under cc by-sa 3.0 with attribution required.