codeigniter PHP多部分表单数据PUT请求?

qeeaahzv  于 2022-12-16  发布在  PHP
关注(0)|答案(6)|浏览(182)

我正在写一个API,我在上传图片时遇到了一些麻烦。
考虑:
我有一个对象,可以通过post/put/delete/get请求创建/修改/删除/查看到一个URL。当有一个文件要上传时,请求是多部分的形式,或者当只有文本要处理时,请求是application/xml。
为了处理与对象相关的图像上传,我做了如下操作:

if(isset($_FILES['userfile'])) {
        $data = $this->image_model->upload_image();
        if($data['error']){
            $this->response(array('error' => $error['error']));
        }
        $xml_data = (array)simplexml_load_string( urldecode($_POST['xml']) );           
        $object = (array)$xml_data['object'];
    } else {
        $object = $this->body('object');
    }

这里的主要问题是在尝试处理put请求时,显然$_POST不包含put数据(据我所知!)。
以下是我构建请求的方式,以供参考:

curl -F userfile=@./image.png -F xml="<xml><object>stuff to edit</object></xml>" 
  http://example.com/object -X PUT

有人知道如何访问PUT请求中的xml变量吗?

sg2wtvxw

sg2wtvxw1#

首先,$_FILES在处理PUT请求时没有被填充,只有在处理POST请求时才被PHP填充。
你需要手动解析它,这也适用于“常规”字段:

// Fetch content and determine boundary
$raw_data = file_get_contents('php://input');
$boundary = substr($raw_data, 0, strpos($raw_data, "\r\n"));

// Fetch each part
$parts = array_slice(explode($boundary, $raw_data), 1);
$data = array();

foreach ($parts as $part) {
    // If this is the last part, break
    if ($part == "--\r\n") break; 

    // Separate content from headers
    $part = ltrim($part, "\r\n");
    list($raw_headers, $body) = explode("\r\n\r\n", $part, 2);

    // Parse the headers list
    $raw_headers = explode("\r\n", $raw_headers);
    $headers = array();
    foreach ($raw_headers as $header) {
        list($name, $value) = explode(':', $header);
        $headers[strtolower($name)] = ltrim($value, ' '); 
    } 

    // Parse the Content-Disposition to get the field name, etc.
    if (isset($headers['content-disposition'])) {
        $filename = null;
        preg_match(
            '/^(.+); *name="([^"]+)"(; *filename="([^"]+)")?/', 
            $headers['content-disposition'], 
            $matches
        );
        list(, $type, $name) = $matches;
        isset($matches[4]) and $filename = $matches[4]; 

        // handle your fields here
        switch ($name) {
            // this is a file upload
            case 'userfile':
                 file_put_contents($filename, $body);
                 break;

            // default for all other files is to populate $data
            default: 
                 $data[$name] = substr($body, 0, strlen($body) - 2);
                 break;
        } 
    }

}

每次迭代时,$data数组将填充您的参数,$headers数组将填充每个部件的标题(例如:Content-Type等),并且$filename将包含原始文件名(如果在请求中提供并且适用于该字段)。
请注意,上述内容只适用于multipart内容类型。在使用上述内容解析正文之前,请确保检查请求Content-Type标头。

goucqfw6

goucqfw62#

请不要再次删除此内容,它对大多数访问此网站的人都有帮助!所有以前的答案都是部分答案,没有像大多数问此问题的人希望的那样涵盖解决方案。
这就像上面所说的那样,还可以处理多个文件上传,并将它们放置在$_FILES中。要让它工作,你必须按照Documentation将'Script PUT/put.php'添加到项目的虚拟主机中。我还怀疑我必须设置一个cron来清理任何'.tmp'文件。

private function _parsePut(  )
{
    global $_PUT;

    /* PUT data comes in on the stdin stream */
    $putdata = fopen("php://input", "r");

    /* Open a file for writing */
    // $fp = fopen("myputfile.ext", "w");

    $raw_data = '';

    /* Read the data 1 KB at a time
       and write to the file */
    while ($chunk = fread($putdata, 1024))
        $raw_data .= $chunk;

    /* Close the streams */
    fclose($putdata);

    // Fetch content and determine boundary
    $boundary = substr($raw_data, 0, strpos($raw_data, "\r\n"));

    if(empty($boundary)){
        parse_str($raw_data,$data);
        $GLOBALS[ '_PUT' ] = $data;
        return;
    }

    // Fetch each part
    $parts = array_slice(explode($boundary, $raw_data), 1);
    $data = array();

    foreach ($parts as $part) {
        // If this is the last part, break
        if ($part == "--\r\n") break;

        // Separate content from headers
        $part = ltrim($part, "\r\n");
        list($raw_headers, $body) = explode("\r\n\r\n", $part, 2);

        // Parse the headers list
        $raw_headers = explode("\r\n", $raw_headers);
        $headers = array();
        foreach ($raw_headers as $header) {
            list($name, $value) = explode(':', $header);
            $headers[strtolower($name)] = ltrim($value, ' ');
        }

        // Parse the Content-Disposition to get the field name, etc.
        if (isset($headers['content-disposition'])) {
            $filename = null;
            $tmp_name = null;
            preg_match(
                '/^(.+); *name="([^"]+)"(; *filename="([^"]+)")?/',
                $headers['content-disposition'],
                $matches
            );
            list(, $type, $name) = $matches;

            //Parse File
            if( isset($matches[4]) )
            {
                //if labeled the same as previous, skip
                if( isset( $_FILES[ $matches[ 2 ] ] ) )
                {
                    continue;
                }

                //get filename
                $filename = $matches[4];

                //get tmp name
                $filename_parts = pathinfo( $filename );
                $tmp_name = tempnam( ini_get('upload_tmp_dir'), $filename_parts['filename']);

                //populate $_FILES with information, size may be off in multibyte situation
                $_FILES[ $matches[ 2 ] ] = array(
                    'error'=>0,
                    'name'=>$filename,
                    'tmp_name'=>$tmp_name,
                    'size'=>strlen( $body ),
                    'type'=>$value
                );

                //place in temporary directory
                file_put_contents($tmp_name, $body);
            }
            //Parse Field
            else
            {
                $data[$name] = substr($body, 0, strlen($body) - 2);
            }
        }

    }
    $GLOBALS[ '_PUT' ] = $data;
    return;
}
ax6ht2ek

ax6ht2ek3#

使用Apiato(Laravel)框架的人群:创建一个新的中间件类文件,然后在你的laravel内核文件中的protected $middlewareGroups变量中声明这个文件(在web或API中,无论你想要什么),如下所示:

protected $middlewareGroups = [
    'web' => [],
    'api' => [HandlePutFormData::class],
];
<?php

namespace App\Ship\Middlewares\Http;

use Closure;
use Symfony\Component\HttpFoundation\ParameterBag;

/**
 * @author Quang Pham
 */
class HandlePutFormData
{
    /**
     * Handle an incoming request.
     *
     * @param  \Illuminate\Http\Request $request
     * @param  \Closure                 $next
     *
     * @return mixed
     */
    public function handle($request, Closure $next)
    {
        if ($request->method() == 'POST' or $request->method() == 'GET') {
            return $next($request);
        }
        if (preg_match('/multipart\/form-data/', $request->headers->get('Content-Type')) or
            preg_match('/multipart\/form-data/', $request->headers->get('content-type'))) {
            $parameters = $this->decode();

            $request->merge($parameters['inputs']);
            $request->files->add($parameters['files']);
        }

        return $next($request);
    }

    public function decode()
    {
        $files = [];
        $data  = [];
        // Fetch content and determine boundary
        $rawData  = file_get_contents('php://input');
        $boundary = substr($rawData, 0, strpos($rawData, "\r\n"));
        // Fetch and process each part
        $parts = $rawData ? array_slice(explode($boundary, $rawData), 1) : [];
        foreach ($parts as $part) {
            // If this is the last part, break
            if ($part == "--\r\n") {
                break;
            }
            // Separate content from headers
            $part = ltrim($part, "\r\n");
            list($rawHeaders, $content) = explode("\r\n\r\n", $part, 2);
            $content = substr($content, 0, strlen($content) - 2);
            // Parse the headers list
            $rawHeaders = explode("\r\n", $rawHeaders);
            $headers    = array();
            foreach ($rawHeaders as $header) {
                list($name, $value) = explode(':', $header);
                $headers[strtolower($name)] = ltrim($value, ' ');
            }
            // Parse the Content-Disposition to get the field name, etc.
            if (isset($headers['content-disposition'])) {
                $filename = null;
                preg_match(
                    '/^form-data; *name="([^"]+)"(; *filename="([^"]+)")?/',
                    $headers['content-disposition'],
                    $matches
                );
                $fieldName = $matches[1];
                $fileName  = (isset($matches[3]) ? $matches[3] : null);
                // If we have a file, save it. Otherwise, save the data.
                if ($fileName !== null) {
                    $localFileName = tempnam(sys_get_temp_dir(), 'sfy');
                    file_put_contents($localFileName, $content);
                    $files = $this->transformData($files, $fieldName, [
                        'name'     => $fileName,
                        'type'     => $headers['content-type'],
                        'tmp_name' => $localFileName,
                        'error'    => 0,
                        'size'     => filesize($localFileName)
                    ]);
                    // register a shutdown function to cleanup the temporary file
                    register_shutdown_function(function () use ($localFileName) {
                        unlink($localFileName);
                    });
                } else {
                    $data = $this->transformData($data, $fieldName, $content);
                }
            }
        }
        $fields = new ParameterBag($data);

        return ["inputs" => $fields->all(), "files" => $files];
    }

    private function transformData($data, $name, $value)
    {
        $isArray = strpos($name, '[]');
        if ($isArray && (($isArray + 2) == strlen($name))) {
            $name = str_replace('[]', '', $name);
            $data[$name][]= $value;
        } else {
            $data[$name] = $value;
        }
        return $data;
    }
}

请注意:上面的代码不全是我的,有些是上面的评论,有些是我修改的。

gcuhipw9

gcuhipw94#

引用netcoder回复:“请注意,上述操作仅适用于多部分内容类型”
为了处理任何内容类型,我在netcoder先生的解决方案中添加了以下代码行:

// Fetch content and determine boundary
   $raw_data = file_get_contents('php://input');
   $boundary = substr($raw_data, 0, strpos($raw_data, "\r\n"));

   /*...... My edit --------- */
    if(empty($boundary)){
        parse_str($raw_data,$data);
        return $data;
    }
   /* ........... My edit ends ......... */
    // Fetch each part
    $parts = array_slice(explode($boundary, $raw_data), 1);
    $data = array();
    ............
    ...............
rdlzhqv9

rdlzhqv95#

我一直在试图找出如何处理这个问题,而不必打破REST的惯例和男孩howdie,什么兔子洞,让我告诉你。
我把这个添加到任何我能找到的地方,希望它能在未来帮助一些人。
我刚刚失去了一天的发展首先找出这是一个问题,然后找出问题所在。
如前所述,这不是symfony(或laravel,或任何其他框架)的问题,而是PHP的限制。
在浏览了不少php core的RFC之后,核心开发团队似乎有点抗拒实现任何与HTTP请求处理现代化有关的东西。这个问题最早在2011年被报道,看起来并没有更接近于本地解决方案。
也就是说,我设法找到了this PECL extension,名为“始终填充表单数据”。我对pecl不是很熟悉,而且似乎无法使用pear使其工作。但我使用的是CentOS和Remi PHP,其中有一个yum包。
我运行了X1 M0 N1 X,它立即修复了这个问题(我不得不重新启动我的Docker容器,但这是给定的)。
我相信还有其他不同风格的linux包,我相信任何对pear/pecl/general php扩展有更多了解的人都可以让它在windows或mac上运行,没有任何问题。

n3ipq98p

n3ipq98p6#

我知道这篇文章很老了。
但遗憾的是,PHP仍然不关注Post方法以外的表单数据。
感谢朋友们(@netcoder,@greendot,@pham-quang),他们提出了以上解决方案。
使用这些解决方案,我写了一个库的目的:

composer require alireaza/php-form-data

您也可以在Laravel中使用composer require alireaza/laravel-form-data

相关问题