PHP:获取文件扩展名不能上传到S3

时间:2012-05-13 19:38:29

标签: php file-upload amazon-s3 file-extension

我使用Amazon S3 API上传文件,每次上传时我都会更改文件名。

例如:

Dog.png> 3Sf5f.png

现在我得到了随机部分:

function rand_string( $length ) {
            $chars = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789";  

            $size = strlen( $chars );
            for( $i = 0; $i < $length; $i++ ) {
                $str .= $chars[ rand( 0, $size - 1 ) ];
            }

            return $str;
        }   

所以我将random_string设置为name参数:

$params->key = rand_string(5);

现在我的问题是,这不会显示任何扩展名。因此,该文件将上传为3Sf5f而不是3Sf5f.png

变量$ filename为我提供了文件的全名及其扩展名。

如果我使用$params->key = rand_string(5).'${filename}';,我会:

3Sf5fDog.png

所以我尝试检索$ filename扩展名并应用它。 我尝试了30多种没有任何正面的方法。

例如我尝试了$ path_info(),我尝试了substr(strrchr($ file_name,'。'),1);还有更多。他们都给了我3Sf5fDog.png3Sf5f

我尝试过的一个例子:

// As @jcinacio pointed out. Change this to: 
//
//   $file_name = "${filename}";
//
$file_name = '${filename}'  // Is is wrong, since '..' does not evaluate 

$params->key = rand_string(5).$file_name;
=
3Sf5fDog.png

$file_name = substr(strrchr('${filename}', '.'), 1);

$params->key = rand_string(5).$file_name;
=
3Sf5f

$filename = "example.png"   // If I declare my own the filename it works.
$file_name = substr(strrchr('${filename}', '.'), 1);

$params->key = rand_string(5).$file_name;
=
3Sf5f.png

整个类文件:http://pastebin.com/QAwJphmW(整个脚本没有其他文件)。

我做错了什么?这真令人沮丧。

12 个答案:

答案 0 :(得分:8)

变量$ filename(因此“$ {filename}”)在代码的第1053行不在范围内(基于来自pastebin的原始粘贴的行编号)。

所以,无论你做什么,你都永远找不到不存在的变量的扩展名。


我终于弄清楚你在做什么了。我认为这是PHP: Rename file before upload

的扩展

简单回答:你不能像你想象的那样去做。为什么 - 在创建URL时没有解析'$ filename',但是变量被传递到Amazon S3并在那里处理。

解决方案

因此,我能想到的唯一选择是使用“successRedirect”参数指向另一个URL。该URL将从Amazon(http://doc.s3.amazonaws.com/proposals/post.html#Dealing_with_Success)接收“bucket”和“key”作为查询参数。指向在Amazon S3上重命名文件的PHP脚本(复制+删除),然后将用户重定向到另一个成功屏幕。

所以,

代码,第34行,

  1. 将完全限定的URL添加到您要访问的新php脚本文件中 写。
  2. php脚本将获取桶和密钥传递给它
  3. 从“密钥”
  4. 创建新文件名
  5. 使用“public static function”功能 copyObject($ srcBucket,$ srcUri,$ bucket,$ uri)“复制上传的 文件到新名称
  6. 然后删除原始文件(使用deleteObject($ bucket,$ uri))
  7. 然后将用户重定向到您要发送的位置
  8. 这将完全符合您的要求。


    回应您的评论“这是唯一的方式 - 亚马逊根据请求收取的费用是多少?”

    删除请求是免费的。在同一个桶(甚至是同一个区域)上移动时没有数据传输成本。所以这个解决方案(这是没有你转移到中间服务器,重命名和上传的唯一方法)它将上传a的成本从每1000上传1c上传到每1000上传2c的成本加倍。我花了10分钟@ 200美元/小时来找到它并回复= $ 33 = 1,666,666上传!当你做数学时,成本会略微下降:)

    与其他解决方案比较:向网络服务器发帖,重命名该文件,然后从网络服务器上传:您将所有带宽从clinet直接移至您自己 - 两次。这也会带来风险并增加可能的失败点。


    回应“不起作用。我上传文件然后删除旧文件”

    我认为这不是问题,因为您上传文件然后在一两秒内重命名。但是如果你想要保证每个文件都被上传,那么你需要做的不仅仅是创建一个随机文件名:

    1. 拥有你的“最终”桶
    2. 对于每次上传,创建一个临时存储桶(如果您担心成本,则每1000个存储桶1c)
    3. 上传到临时存储桶
    4. 创建随机名称,检查最终存储桶中是否存在(每1000次检查1c)
    5. 将文件复制到最终存储桶(使用新名称)
    6. 删除上传的文件以及存储桶。
    7. 定期清理文件上传未完成的存储桶。

答案 1 :(得分:4)

$fileSplit = explode('.',$filename);
$extension = '.'.$fileSplit[count($fileSplit) - 1];

这个explode()将文件名分成一个数组,其中句点为分隔符,然后抓取数组的最后一部分(包含文件名为foo.bar.jpg),并在其前面放一个句点。这应该为您提供所需的扩展名,以将其附加到rand_string(5)。

$params->key = rand_string(5).$extension;

答案 2 :(得分:2)

如果你正在上传图片,试试这个

$dim = getimagesize($file);
$type = $dim[2];
if( $type == IMAGETYPE_JPEG ) $ext=".jpg"; 
if( $type == IMAGETYPE_GIF ) $ext=".gif"; 
if( $type == IMAGETYPE_PNG ) $ext=".png";

$params->key = rand_string(5).$ext;

答案 3 :(得分:2)

我认为下面这么简单的事情应该可以从文件名中提取文件扩展名:

function getFileExtension($fileName)
{
    $ext = '';
    if(strpos($fileName, ".") !== false)
    {
        $ext = end(explode(".", $fileName));
    }
    return $ext;
}

答案 4 :(得分:1)

如果你发生了什么:

$filename = "${filename}";
echo $filename;
die();

你得到像'Dog.png'这样的东西吗?如果不这样,你获取文件名的方式有问题。如果你确实得到像'Dog.png'这样的东西,这就是我用来获取文件扩展名的内容。

$pieces = explode('.',trim($filename));
$extension = end($pieces);

然后你应该能够做到这一点:

$params->key = rand_string(5).'.'.$extension;

答案 5 :(得分:1)

您需要首先找出原始扩展名是什么,而不是重命名整个文件。因此,请保留扩展名并重命名de file name。

假设您在$ image_name中有图像名称:

$image_name = "image.png";
$random_string = "random";

list($filename,$fileext) = explode(".",$image_name);
$new_name = $random_string.'.'.$fileext;
rename($image_name,$new_name);  

答案 6 :(得分:1)

好的,这是我在服务器端遇到扩展时遇到的另一个尝试。我做的是,我使用javascript提取文件扩展名并通过post发送。

<script type="text/javascript" >
function fileinfo() {
var file = document.form.file.value;
document.getElementById("ext").value = file.split('.').pop();
document.myform.submit();   
}
</script>
<form name="myform" enctype="multipart/form-data" onsubmit="fileinfo();">
<input type="file" name="file">
<input type="hidden" name="ext">
//rest of the form
</form>

在下一个php文件中,您可以直接使用$ _POST ['ext']作为扩展名。希望有所帮助。如果您在实施此

时遇到任何问题,请与我们联系

答案 7 :(得分:1)

我在我的网站上使用它(并且可以正常工作多年):

$file = 'yourOriginalfile.png';

//get the file extension
$fileExt = substr(strrchr($file, '.'), 1);

//create a random name and add the original extension
$fileUniqueName = md5(uniqid(mktime())) . '.' . $fileExt;

rename($file,$fileUniqueName); 

您的函数生成的文件名太短(5个字符),这样会创建更长的文件名,避免碰撞文件名。

示例输出:      aff5a25e84311485d4eedea7e5f24a4f.png

答案 8 :(得分:1)

看起来实际发生的是现在而不是完全生成文件名,你实际上是通过接口传递一个非常小的'程序',因此它可以在以后产生文件名(当变量$ filename存在时在范围内)。接口的另一端最终执行您传入的“程序”,从而生成修改后的文件名。 (当然,将'程序'传递给其他以后执行的东西并不会使调试变得非常简单: - )

(当然由你决定“做这项工作”还是“以不同的方式做”。“不同方式”通常涉及在你试图调用上传界面之前自己重命名或复制文件,并在其他答案中描述。)

如果你决定“让它工作”,那么整个文件名参数需要是一个程序,而不仅仅是它的一部分。这种有点不常见的功能通常涉及将整个字符串括在单个引号中。 (你还需要对字符串中的现有单引号做一些事情,这样它们就不会太快终止引用的字符串。一种方法是用反斜杠引用它们。另一种看起来更干净且通常有效的方法是用双引号替换它们。)换句话说,我相信下面的代码对你有用(我没有合适的环境来测试它,所以我不确定)。

$file_extension = 'substr(strrchr(${filename}, "."), 1)';

$params->key = rand_string(5).$file_extension;

(一旦你开始工作,你可能想重新审视你的命名方案。也许名字需要更长一点。或者它可能包括一些可识别的信息{如今天的日期,或文件的原始名称}你可能会遇到像$ file_base.rand_string(7)这样的东西。$ file_extension。

答案 9 :(得分:0)

重新命名文件并计算扩展程序的简单解决方案

$fileName = 'myRandomFile.jpg';

// separate the '.'-separated parts of the file name
$parts = explode( '.', $fileName );

// Solution will not work, if no extension is present
assert( 1 < count( $parts ) );

// keep the extension and drop the last part
$extension = $parts[ count( $parts ) - 1 ];
unset( $parts[ count( $parts ) - 1 ] );

// finally, form the new file name
$newFileName = md5( 'someSeedHere' + implode( '.', $parts )) . '.' . $extension;

echo $extension             // outputs jpg
   . ' - ' 
   . $newFileName           // outputs cfcd208495d565ef66e7dff9f98764da.jpg
   ;

注意,md5()总是32字节长,非唯一关于计算值。对于许多实际情况,它足够独特

<强>附录

此外,您可以使用此解决方案跟踪变量

abstract class CSTReportDelegate {

    abstract public function emitVariableChange( $variableName, $oldValue, $newValue );
    abstract public function emitVariableSetNew( $variableName, $newValue );

}

class CSTSimpleReportDelegate extends CSTReportDelegate {

    public function emitVariableChange( $variableName, $oldValue, $newValue ) {
        echo '<br />[global/change] '. $variableName . ' : ' . print_r( $oldValue, true ) . ' &rarr; ' . print_r( $newValue, true );
    }

    public function emitVariableSetNew( $variableName, $newValue ) {
        echo '<br />[global/init] '. $variableName . '   &rarr; ' . print_r( $newValue, TRUE );
    }

}


class CSysTracer {

    static protected 
        $reportDelegate;

    static private 
        $globalState = array();

    static private  
        $traceableGlobals = array();

    static private 
        $globalTraceEnabled = FALSE;

    const 
        DEFAULT_TICK_AMOUNT = 1;

    static public 
    function setReportDelegate( CSTReportDelegate $aDelegate ) {
        self::$reportDelegate = $aDelegate;
    }


    static public 
    function start( $tickAmount = self::DEFAULT_TICK_AMOUNT ) {

        register_tick_function ( array( 'CSysTracer', 'handleTick' ) );

    }


    static public 
    function stop() {

        unregister_tick_function( array( 'CSysTracer', 'handleTick' ) );

    }

    static public 
    function evalAndTrace( $someStatement ) {

        declare( ticks = 1 ); {
            self::start();
            eval( $someStatement );
            self::stop();
        }
    }

    static public 
    function addTraceableGlobal( $varName ) {

        if ( is_array( $varName )) {
            foreach( $varName as $singleName ) {
                self::addTraceableGlobal( $singleName ); 
            }
            return;
        }

        self::$traceableGlobals[ $varName ] = $varName;

    }

    static public 
    function removeTraceableGlobal( $varName ) {
        unset( self::$traceableGlobals[ $varName ] );   
    }

    /**
     * Main function called at each tick. Calls those functions, which
     * really perform the checks.
     * 
     */
    static public 
    function handleTick( ) {

        if ( TRUE === self::$globalTraceEnabled ) { 
            self::traceGlobalVariable();
        }

    }

    static public 
    function enableGlobalsTrace() {
        self::$globalTraceEnabled = TRUE;   
    }


    static public 
    function disableGlobalsTrace() {
        self::$globalTraceEnabled = FALSE;  
    }

    static public 
    function traceGlobalVariable( ) {

        foreach( self::$traceableGlobals as $aVarname ) {

            if ( ! isset( $GLOBALS[ $aVarname ] )) {
                continue;
            }

            if ( ! isset( self::$globalState[ $aVarname ] ) ) {

                self::$reportDelegate->emitVariableSetNew( $aVarname, $GLOBALS[ $aVarname ] );
                self::$globalState[ $aVarname ] = $GLOBALS[ $aVarname ];
                continue;
            }

           if ( self::$globalState[ $aVarname ] !== $GLOBALS[ $aVarname ]) {

             self::$reportDelegate->emitVariableChange( $aVarname, self::$globalState[ $aVarname ], $GLOBALS[ $aVarname ] );

           }

           self::$globalState[ $aVarname ] = $GLOBALS[ $aVarname ];

        }

    }

}

示例用例:

ini_set("display_errors", TRUE);
error_reporting(E_ALL);

require_once( dirname( __FILE__ ) . '/CStatementTracer.inc.php' );

/* Ticks make it easy to have a function called for every line of PHP
 *  code. We can use this to track the state of a variable throughout
 * the execution of a script.
 */



CSysTracer::addTraceableGlobal( array( 'foo', 'bar' ));

CSysTracer::setReportDelegate( new CSTSimpleReportDelegate() ); 
CSysTracer::enableGlobalsTrace();

CSysTracer::start(); 
declare( ticks = 1 );

   //
   // At this point, tracing is enabled. 
   // Add your code or call your functions/methods here
   //

CSysTracer::stop();

答案 10 :(得分:0)

未经测试,但足够简单:

$ext = pathinfo($filename, PATHINFO_EXTENSION);

将返回扩展部分(不带'。')

答案 11 :(得分:0)

这个怎么样?

$temp = rand_string(5).'${filename}';         //should be 3Sf5fDog.png
$ext = pathinfo($temp, PATHINFO_EXTENSION); //should be .png
$temp2 = rand_string(5) . $ext;             //should be 4D47a.png