CURLアップロードファイルのバージョン違い問題の解析

小云云
リリース: 2023-03-20 18:16:01
オリジナル
1517 人が閲覧しました

=フロントエンド投稿フォームがファイルをアップロードした後、バックエンドはファイルを受信した後、投稿を画像サーバーに転送します。そこで、curlを使用して「@file path」を使用してアップロードしました

コードは次のとおりです

<?php
    if($_FILES[&#39;video&#39;][&#39;size&#39;]>0){        $data = array('video'=>$_FILES['video']['tmp_name']);         $ch = curl_init();         $url="test.php";
        // 设置URL和相应的选项
        curl_setopt($ch, CURLOPT_URL, $url);
        //启用时会将头文件的信息作为数据流输出。 
        curl_setopt($ch, CURLOPT_HEADER, 0);
        //将curl_exec()获取的信息以字符串返回,而不是直接输出
        curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
        curl_setopt($ch, CURLOPT_POST, 1);
        curl_setopt($ch, CURLOPT_POSTFIELDS, $data);         $res = curl_exec($ch);
        // 抓取URL并把它传递给浏览器
        curl_exec($ch);
        // 关闭cURL资源,并且释放系统资源
        curl_close($ch);

   }
ログイン後にコピー
ログイン後にコピー

テストマシンのPHPバージョンは5.2です。コードが壊れても問題ありませんが、オンラインにすると。 、$_FILES 変数は値を受け取ることができません。次に、$_POST で $_POST['video']='@/tmp/phps12d63'; を受信したことをよく観察してください。これは、Content-Type が正常に送信されていないことです。アップロードされたファイルの名前は「multipart/form-data」である必要があります。この場合は「application/x-www-form-urlencoded」である必要があります。次に、画像サーバー上の $_SERVER[‘HTTP_CONTENT_TYPE’] 変数を出力し、それが「multipart/form-data」であることを確認します。
マニュアルを確認すると、CURLOPT_POSTFIELDS の渡されたパラメータが配列の場合、Content-Type ヘッダーは multipart/form-data に設定されることがわかりました。
次に、php5.5.0 で追加された魔法の CURLOPT_SAFE_UPLOAD パラメータがあります。デフォルト値は false、5.6.0 のデフォルトは true、

CURLOPT_SAFE_UPLOAD
TRUE を指定すると、CURLOPT_POSTFIELDS で @ プレフィックスが付いたファイルの送信が無効になります。 これは、@ をフィールドで安全に使用できることを意味します。 利用可能です
アップロードの代わりに CURLFile を使用します。

および CURLOPT_POSTFIELDS パラメーターの説明

CURLOPT_POSTFIELDS
すべてのデータは、HTTP プロトコルの「POST」操作を使用して送信されます。 ファイルを送信するには、ファイル名の前に @ を付け、フルパスを使用します。 ファイルの種類は、ファイル名の後に「;type=mimetype」の形式で指定できます。 このパラメータは、「para1=val1¶2=val2&...」のような URL コード化された文字列にすることも、フィールド名をキーとして、フィールド データを値として持つ配列を使用することもできます。 value が配列の場合、Content-Type ヘッダーは multipart/form-data に設定されます。 PHP 5.2.0 以降、@ プレフィックスを使用してファイルを渡す場合、値は配列である必要があります。 PHP 5.5.0 以降、@ プレフィックスは非推奨になり、ファイルは CURLFile 経由で送信できるようになりました。 CURLOPT_SAFE_UPLOAD を TRUE に設定すると、セキュリティを強化するために @ プレフィックスが付いたファイルの送信が無効になります。

バージョンを確認すると、テスト環境はphp5.3、オンラインテスト環境は5.6であることがわかります。つまり、CURLOPT_SAFE_UPLOADのデフォルトはtrue、@アップロードは無効、@は通常の文字列です。 5.5 以降、ファイルをアップロードするための @ プレフィックスは廃止されました。 5.5 以降のバージョンは、CURLFile を使用してアップロードする必要があります。最終的な互換性のあるソリューションでは、5.5 未満の場合は @ プレフィックスを使用し、5.5 よりも前のバージョンの場合は @ を使用してアップロードし、追加のパラメーターを追加できます。

<?php
//curl_file_create是函数的别名CURLFile::__construct() if (!function_exists('curl_file_create')) {
    function curl_file_create($filename, $mimetype = '', $postname = '') {
        return "@$filename;filename="
            . ($postname ?: basename($filename))
            . ($mimetype ? ";type=$mimetype" : '');
    }
}
ログイン後にコピー
ログイン後にコピー

最終的なコードは

<?php
    if($_FILES[&#39;video&#39;][&#39;size&#39;]>0){        $data = array('video'=>$_FILES['video']['tmp_name']);         $ch = curl_init();         $url="test.php";
        // 设置URL和相应的选项
        curl_setopt($ch, CURLOPT_URL, $url);
        //启用时会将头文件的信息作为数据流输出。 
        curl_setopt($ch, CURLOPT_HEADER, 0);
        //将curl_exec()获取的信息以字符串返回,而不是直接输出
        curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
        curl_setopt($ch, CURLOPT_POST, 1);        $data['video']=curl_file_create($_FILES['video']['tmp_name'],'video/mp4',$_FILES['video']['name']);
        curl_setopt($ch, CURLOPT_POSTFIELDS, $data);         $res = curl_exec($ch);
        // 抓取URL并把它传递给浏览器
        curl_exec($ch);
        // 关闭cURL资源,并且释放系统资源
        curl_close($ch);

   }
ログイン後にコピー
ログイン後にコピー

です。 まとめ, 問題に遭遇したら、まず冷静に分析してからマニュアルをよく読みましょう。

フロントエンドの投稿フォームがファイルをアップロードした後、バックエンドがファイルを受信した後、投稿を画像サーバーに転送しました。金曜日に問題が発生しました。そこで、curlを使ってアップロードし、「@file path」を使用してアップロードしました
コードは次のとおりです

<?php
    if($_FILES[&#39;video&#39;][&#39;size&#39;]>0){        $data = array('video'=>$_FILES['video']['tmp_name']);         $ch = curl_init();         $url="test.php";
        // 设置URL和相应的选项
        curl_setopt($ch, CURLOPT_URL, $url);
        //启用时会将头文件的信息作为数据流输出。 
        curl_setopt($ch, CURLOPT_HEADER, 0);
        //将curl_exec()获取的信息以字符串返回,而不是直接输出
        curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
        curl_setopt($ch, CURLOPT_POST, 1);
        curl_setopt($ch, CURLOPT_POSTFIELDS, $data);         $res = curl_exec($ch);
        // 抓取URL并把它传递给浏览器
        curl_exec($ch);
        // 关闭cURL资源,并且释放系统资源
        curl_close($ch);

   }
ログイン後にコピー
ログイン後にコピー

テストマシンのPHPバージョンは5.2です。コードが壊れても問題ありませんが、オンラインにすると、 $_FILES 変数は値を受け取ることができません。次に、$_POST で $_POST['video']='@/tmp/phps12d63'; を受信したことをよく観察してください。これは、Content-Type が正常に送信されていないことです。アップロードされたファイルの名前は「multipart/form-data」である必要があります。この場合は「application/x-www-form-urlencoded」である必要があります。次に、画像サーバー上の $_SERVER[‘HTTP_CONTENT_TYPE’] 変数を出力し、それが「multipart/form-data」であることを確認します。
マニュアルを確認すると、CURLOPT_POSTFIELDS の渡されたパラメータが配列の場合、Content-Type ヘッダーは multipart/form-data に設定されることがわかりました。
次に、php5.5.0 で追加された魔法の CURLOPT_SAFE_UPLOAD パラメータがあります。デフォルト値は false、5.6.0 のデフォルトは true、

CURLOPT_SAFE_UPLOAD
TRUE を指定すると、CURLOPT_POSTFIELDS で @ プレフィックスが付いたファイルの送信が無効になります。 これは、@ をフィールドで安全に使用できることを意味します。 利用可能です
アップロードの代わりに CURLFile を使用します。

および CURLOPT_POSTFIELDS パラメーターの説明

CURLOPT_POSTFIELDS
すべてのデータは、HTTP プロトコルの「POST」操作を使用して送信されます。 ファイルを送信するには、ファイル名の前に @ を付け、フルパスを使用します。 ファイルの種類は、ファイル名の後に「;type=mimetype」の形式で指定できます。 このパラメータは、「para1=val1¶2=val2&...」のような URL コード化された文字列にすることも、フィールド名をキーとして、フィールド データを値として持つ配列を使用することもできます。 value が配列の場合、Content-Type ヘッダーは multipart/form-data に設定されます。 PHP 5.2.0 以降、@ プレフィックスを使用してファイルを渡す場合、値は配列である必要があります。 PHP 5.5.0 以降、@ プレフィックスは非推奨になり、ファイルは CURLFile 経由で送信できるようになりました。 CURLOPT_SAFE_UPLOAD を TRUE に設定すると、セキュリティを強化するために @ プレフィックスが付いたファイルの送信が無効になります。

查看版本,果然测试环境是php5.3,而线上测试环境是5.6,也是就是说CURLOPT_SAFE_UPLOAD默认为true,禁用了@ 上传,@就是普通字符串了。而从5.5开始@前缀上传文件已经被废弃。大于5.5版本需使用CURLFile 上传。最后兼容方案小于5.5使用@前缀,大于5.5使用CURLFile,在小于5.5的版本是用@上传还可以添加额外的参数;filename=文件名;type=mime类型

<?php
//curl_file_create是函数的别名CURLFile::__construct() if (!function_exists('curl_file_create')) {
    function curl_file_create($filename, $mimetype = '', $postname = '') {
        return "@$filename;filename="
            . ($postname ?: basename($filename))
            . ($mimetype ? ";type=$mimetype" : '');
    }
}
ログイン後にコピー
ログイン後にコピー

最后代码为

<?php
    if($_FILES[&#39;video&#39;][&#39;size&#39;]>0){        $data = array('video'=>$_FILES['video']['tmp_name']);         $ch = curl_init();         $url="test.php";
        // 设置URL和相应的选项
        curl_setopt($ch, CURLOPT_URL, $url);
        //启用时会将头文件的信息作为数据流输出。 
        curl_setopt($ch, CURLOPT_HEADER, 0);
        //将curl_exec()获取的信息以字符串返回,而不是直接输出
        curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
        curl_setopt($ch, CURLOPT_POST, 1);        $data['video']=curl_file_create($_FILES['video']['tmp_name'],'video/mp4',$_FILES['video']['name']);
        curl_setopt($ch, CURLOPT_POSTFIELDS, $data);         $res = curl_exec($ch);
        // 抓取URL并把它传递给浏览器
        curl_exec($ch);
        // 关闭cURL资源,并且释放系统资源
        curl_close($ch);

   }
ログイン後にコピー
ログイン後にコピー

相关推荐:

php通过CURL上传文件

php curl上传文件的简单例子

post - php curl上传文件如何像表单一样指定其name值

以上がCURLアップロードファイルのバージョン違い問題の解析の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

関連ラベル:
ソース:php.cn
このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。
最新の問題
人気のチュートリアル
詳細>
最新のダウンロード
詳細>
ウェブエフェクト
公式サイト
サイト素材
フロントエンドテンプレート