URL에서 서버로 파일 다운로드


341

글쎄요, 이것은 매우 단순 해 보입니다. 서버로 파일을 다운로드하기 위해해야 ​​할 일은 다음과 같습니다.

file_put_contents("Tmpfile.zip", file_get_contents("http://someurl/file.zip"));

하나의 문제 만 있습니다. 100mb와 같은 큰 파일이 있으면 어떻게해야합니까? 그런 다음 메모리가 부족하여 파일을 다운로드 할 수 없습니다.

내가 원하는 것은 파일을 다운로드 할 때 디스크에 파일을 쓰는 방법입니다. 그렇게하면 메모리 문제가 발생하지 않고 더 큰 파일을 다운로드 할 수 있습니다.


4
서버 구성에 설정되어 있습니다. PHP는 내가 아는 한 직접 접근 할 수 없습니다 (직접 .ini 편집 제외)
Ben

답변:


494

PHP 5.1.0부터는 file_put_contents()스트림 핸들을 $data매개 변수 로 전달하여 조각 단위로 쓰기를 지원합니다 .

file_put_contents("Tmpfile.zip", fopen("http://someurl/file.zip", 'r'));

매뉴얼에서 :

경우 데이터 [즉, 두 번째 인자 인] 스트림 자원, 즉 스트림의 나머지 버퍼 지정된 파일에 복사한다. 이것은을 사용하는 것과 비슷 stream_copy_to_stream()합니다.

( Hakre 감사 합니다 .)


4
그것은 나의 첫번째 선택이 아닐 것입니다. allow_fopen_url Offphp.ini (보안에 대한 좋은 아이디어)에 설정되어 있으면 스크립트가 손상되었을 수 있습니다.
PleaseStand

4
@idealmachine 나는 그것이 사실 file_get_contents()이라면 작동하지 않을 것이라고 생각한다 (OP 참조).
alex

10
@geoff 나는 구체적이었습니다. 원하는 기능을 언급했습니다. 당신이 원했던 것은 누군가 당신을 위해 코드를 작성하는 것이 었습니다. 또한, 우리가 서로의 SO 상호 작용에 대해 언급 할 예정이라면, 답변을 좀 더 받아 주시기 바랍니다. :)
alex

@alex : 편집 내용을 참조하십시오. 자유롭게 통합하십시오. 이 주석을 언제 제거 할 수 있는지 알려주십시오.
hakre

4
'b'플래그는 대부분의 경우 fopen; 이미지 및 기타 일반 텍스트 파일에 대한 부작용을 방지합니다.
Wayne Weibel

132
private function downloadFile($url, $path)
{
    $newfname = $path;
    $file = fopen ($url, 'rb');
    if ($file) {
        $newf = fopen ($newfname, 'wb');
        if ($newf) {
            while(!feof($file)) {
                fwrite($newf, fread($file, 1024 * 8), 1024 * 8);
            }
        }
    }
    if ($file) {
        fclose($file);
    }
    if ($newf) {
        fclose($newf);
    }
}

1
스 니핏 주셔서 감사하지만 코드 @xaav를 설명 할 수 있습니까? 나는 PHP에서 정확하게 훌륭하지 않습니다. 1024 * 8은 무엇입니까? 다시 감사합니다.
vvMINOvv

@wMINOw 줄의 길이입니다.
David Bélanger 2016 년

2
특히 매개 변수가 바이트 단위이므로 한 번에 최대 8KB (KB 당 1,024 바이트 * 8)를 읽는 것을 의미합니다. 줄이 <= 8KB 인 한, 한 번에 전체 줄을 읽습니다.
Doktor J

1
이것이 왜 최선의 대답이 아닌가?
건잭

1
이 방법으로 오류를 어떻게 처리합니까? 404가 반환되거나 연결이 중단되거나 시간 초과되면 어떻게됩니까?
Adam Swinden 2014

67

cURL을 사용해보십시오

set_time_limit(0); // unlimited max execution time
$options = array(
  CURLOPT_FILE    => '/path/to/download/the/file/to.zip',
  CURLOPT_TIMEOUT =>  28800, // set this to 8 hours so we dont timeout on big files
  CURLOPT_URL     => 'http://remoteserver.com/path/to/big/file.zip',
);

$ch = curl_init();
curl_setopt_array($ch, $options);
curl_exec($ch);
curl_close($ch);

확실하지 않지만 CURLOPT_FILE데이터를 가져올 때 쓰는 옵션, 즉 믿습니다 . 버퍼링되지 않았습니다.


2
일반적으로 이것은 괜찮지 만 웹 응용 프로그램 에이 코드가 있으므로 사용자가 cURL을 설치했는지 확신 할 수 없습니다. 그러나 나는 이것을 투표했다.
xaav

@Geoff는 분산 웹 앱입니까? 호스팅을 제어하면 사용자에 대해서는 중요하지 않습니다 (cURL은 서버의 라이브러리입니다).
alex

아니요. 호스팅을 제어하지 않습니다. 누구나 가질 수있는 분산 웹 앱입니다.
xaav

3
컬이 없을 수 있습니다. 그러나 거의 모든 공유 호스팅 회사에는 기본적으로 CURL이 설치되어 있습니다. 나는 보이지 않는 것을 본 적이 없습니다.
Mangirdas Skripka

19
내 테스트에서와 같이 CURLOPT_FILE에 파일 경로를 직접 할당 할 수 없습니다. 파일 핸들러 여야합니다. 먼저로 파일을 열고 after로 $fh = fopen('/path/to/download/the/file/to.zip', 'w');닫습니다 . 그리고 설정fclose($fh);curl_close($ch);CURLOPT_FILE => $fh
Gustavo

22

prodigitalson의 답변이 효과 가 없었습니다. 더 자세한 정보를 얻었습니다missing fopen in CURLOPT_FILE .

이것은 로컬 URL을 포함하여 저에게 효과적이었습니다.

function downloadUrlToFile($url, $outFileName)
{   
    if(is_file($url)) {
        copy($url, $outFileName); 
    } else {
        $options = array(
          CURLOPT_FILE    => fopen($outFileName, 'w'),
          CURLOPT_TIMEOUT =>  28800, // set this to 8 hours so we dont timeout on big files
          CURLOPT_URL     => $url
        );

        $ch = curl_init();
        curl_setopt_array($ch, $options);
        curl_exec($ch);
        curl_close($ch);
    }
}

19
  1. 대상 서버에 "다운로드"라는 폴더를 만듭니다.
  2. [이 코드]를 .php파일 로 저장하고 대상 서버에서 실행

다운로더 :

<html>
<form method="post">
<input name="url" size="50" />
<input name="submit" type="submit" />
</form>
<?php
    // maximum execution time in seconds
    set_time_limit (24 * 60 * 60);

    if (!isset($_POST['submit'])) die();

    // folder to save downloaded files to. must end with slash
    $destination_folder = 'downloads/';

    $url = $_POST['url'];
    $newfname = $destination_folder . basename($url);

    $file = fopen ($url, "rb");
    if ($file) {
      $newf = fopen ($newfname, "wb");

      if ($newf)
      while(!feof($file)) {
        fwrite($newf, fread($file, 1024 * 8 ), 1024 * 8 );
      }
    }

    if ($file) {
      fclose($file);
    }

    if ($newf) {
      fclose($newf);
    }
?>
</html> 

이것은 사용자가 기존 PHP 응용 프로그램 내에서 작동하는 솔루션이 아닌 독립 실행 형 스크립트를 원한다고 가정하고 후자는 OP 및 대부분의 다른 사람들이 찾고있는 것이라고 생각합니다. 설명은 접근 방식을 이해하려는 사람들에게도 도움이 될 것입니다.
Sean the Bean

1
이 작업을 시도 할 때마다 항상 전송 된 파일 크기는 50816이지만 파일 크기는 이보다 큽니다. 120MB .. 왜 이런 생각입니까?
Riffaz Starr

set_time_limit (24 * 60 * 60);루프 안에 넣어야합니다. 스크립트 시작 부분에는 영향을 미치지 않습니다.
Viktor Joras

16
set_time_limit(0); 
$file = file_get_contents('path of your file');
file_put_contents('file.ext', $file);

귀하의 답변은 매우 간단하고 잘 작동하며 cURL이 파일을 가져 오지 못한 곳에서 도움이되었습니다. 감사합니다 :)
Tommix

2
이것이 실제로 무엇을하는지 설명하고 싶을 수도 있습니다.
alex

6
이것은 OP의 PHP 메모리 제한을 초과하는 문제를 해결하지 못합니다.
user9645

이것은 매우 간단하고 간단합니다. 파일이 작거나 환경이 로컬 개발 인 간단한 경우에 매우 유용합니다.
Valentine Shi

.xlsx 파일에 대한 아이디어가 있습니까? 0 바이트 메모리로 빈 파일을 저장하고 있습니다.
Dhruv Thakkar

9

세 가지 방법이 있습니다.

  1. file_get_contents 및 file_put_contents
  2. 곱슬 곱슬하다
  3. 열다

여기에서 예제 찾을 수 있습니다 .


8

PHP에서 간단한 방법을 사용하십시오 copy()

copy($source_url, $local_path_with_file_name);

참고 : 대상 파일이 이미 존재하면 덮어 씁니다.

PHP copy () 함수

참고 : 대상 폴더에 대한 권한 777을 설정해야합니다. 로컬 컴퓨터로 다운로드 할 때이 방법을 사용하십시오.

특별 참고 사항 : 777은 소유자, 그룹 및 모든 사람에게 전체 읽기 / 쓰기 / 실행 권한이있는 Unix 기반 시스템의 권한입니다. 일반적으로 우리는 웹 서버에서 공개적으로 숨길 필요가없는 자산에이 권한을 부여합니다. 예 : 이미지 폴더


1
나는 결코 웹 서버에서 절대로 777을 파마로 설정하지 않을 것이며, 그렇게하는 나쁜 아이디어를 가진 웹 개발자를 시작할 것입니다. 언제 어디서나 조심해 ! 당신은 그렇게 할 수 없습니다! 보안에 대해 생각하십시오. OWASP 규칙을 따르는 것만으로는 충분하지 않습니다. 간단한 것에 대해 잘 생각하는 것이 중요합니다.
ThierryB

@ThierryB. 참고 : 로컬 경로를 지정했습니다. & 이것은 내부 응용 프로그램에서 사용할 수 있습니다. 질문과 답변 문제를 잘 읽고 이해합니다. 다른 시나리오를 생각하십시오. 그리고 이것은 받아 들일 수있는 최선의 대답이 아닙니다. 모든 질문에는 장단점이있는 다른 답변이 있습니다. 이해할 수있는 예 : 피보나치조차도 하나만 가장 적합한 여러 가지 고유 한 솔루션이 있습니다. 다른 시나리오는 다른 시나리오에서 사용됩니다.
Pradeep Kumar Prabaharan

그러나 모범 사례에 대해 생각하고 안전한 장소에서 구현하는 데 시간을 내면 구현해야하는 개념을 더 잘 이해할 수 있습니다. 어쩌면 침입자가 ($) 집 안에 있다면, 함정을 짓거나 물건을 짓는 것이 가장 좋은 방법입니다.)
ThierryB

5

이것을 사용하여 파일을 다운로드합니다

function cURLcheckBasicFunctions()
{
  if( !function_exists("curl_init") &&
      !function_exists("curl_setopt") &&
      !function_exists("curl_exec") &&
      !function_exists("curl_close") ) return false;
  else return true;
}

/*
 * Returns string status information.
 * Can be changed to int or bool return types.
 */
function cURLdownload($url, $file)
{
  if( !cURLcheckBasicFunctions() ) return "UNAVAILABLE: cURL Basic Functions";
  $ch = curl_init();
  if($ch)
  {

    $fp = fopen($file, "w");
    if($fp)
    {
      if( !curl_setopt($ch, CURLOPT_URL, $url) )
      {
        fclose($fp); // to match fopen()
        curl_close($ch); // to match curl_init()
        return "FAIL: curl_setopt(CURLOPT_URL)";
      }
      if ((!ini_get('open_basedir') && !ini_get('safe_mode')) || $redirects < 1) {
        curl_setopt($ch, CURLOPT_USERAGENT, '"Mozilla/5.0 (X11; U; Linux i686; en-US; rv:1.8.1.11) Gecko/20071204 Ubuntu/7.10 (gutsy) Firefox/2.0.0.11');
        curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
        //curl_setopt($ch, CURLOPT_REFERER, 'http://domain.com/');
        if( !curl_setopt($ch, CURLOPT_HEADER, $curlopt_header)) return "FAIL: curl_setopt(CURLOPT_HEADER)";
        if( !curl_setopt($ch, CURLOPT_FOLLOWLOCATION, $redirects > 0)) return "FAIL: curl_setopt(CURLOPT_FOLLOWLOCATION)";
        if( !curl_setopt($ch, CURLOPT_FILE, $fp) ) return "FAIL: curl_setopt(CURLOPT_FILE)";
        if( !curl_setopt($ch, CURLOPT_MAXREDIRS, $redirects) ) return "FAIL: curl_setopt(CURLOPT_MAXREDIRS)";

        return curl_exec($ch);
    } else {
        curl_setopt($ch, CURLOPT_USERAGENT, '"Mozilla/5.0 (X11; U; Linux i686; en-US; rv:1.8.1.11) Gecko/20071204 Ubuntu/7.10 (gutsy) Firefox/2.0.0.11');
        curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
        //curl_setopt($ch, CURLOPT_REFERER, 'http://domain.com/');
        if( !curl_setopt($ch, CURLOPT_FOLLOWLOCATION, false)) return "FAIL: curl_setopt(CURLOPT_FOLLOWLOCATION)";
        if( !curl_setopt($ch, CURLOPT_FILE, $fp) ) return "FAIL: curl_setopt(CURLOPT_FILE)";
        if( !curl_setopt($ch, CURLOPT_HEADER, true)) return "FAIL: curl_setopt(CURLOPT_HEADER)";
        if( !curl_setopt($ch, CURLOPT_RETURNTRANSFER, true)) return "FAIL: curl_setopt(CURLOPT_RETURNTRANSFER)";
        if( !curl_setopt($ch, CURLOPT_FORBID_REUSE, false)) return "FAIL: curl_setopt(CURLOPT_FORBID_REUSE)";
        curl_setopt($ch, CURLOPT_USERAGENT, '"Mozilla/5.0 (X11; U; Linux i686; en-US; rv:1.8.1.11) Gecko/20071204 Ubuntu/7.10 (gutsy) Firefox/2.0.0.11');
    }
      // if( !curl_setopt($ch, CURLOPT_FOLLOWLOCATION, true) ) return "FAIL: curl_setopt(CURLOPT_FOLLOWLOCATION)";
      // if( !curl_setopt($ch, CURLOPT_FILE, $fp) ) return "FAIL: curl_setopt(CURLOPT_FILE)";
      // if( !curl_setopt($ch, CURLOPT_HEADER, 0) ) return "FAIL: curl_setopt(CURLOPT_HEADER)";
      if( !curl_exec($ch) ) return "FAIL: curl_exec()";
      curl_close($ch);
      fclose($fp);
      return "SUCCESS: $file [$url]";
    }
    else return "FAIL: fopen()";
  }
  else return "FAIL: curl_init()";
}

4

PHP 4 & 5 솔루션 :

readfile () 은 자체적으로 큰 파일을 보낼 때에도 메모리 문제를 나타내지 않습니다. fopen 래퍼가 활성화 된 경우이 기능을 사용하여 URL을 파일 이름으로 사용할 수 있습니다.

http://php.net/manual/en/function.readfile.php


1
출력 버퍼가 아닌 디스크에 쓰는 것이 문제이기 때문에 질문에 대답하지 않습니다.
Lorenz Meyer
당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.