subida de archivos con nombre numerico del array

Iniciado por gAb1, 3 Junio 2016, 16:59 PM

0 Miembros y 1 Visitante están viendo este tema.

gAb1

Estoy teniendo problemas para extender un código existente que guarda imágenes.

Los usuarios pueden subir imagenes y también tienen que poder actualizarlas (eliminar y subir nuevas). Lo primero es lo que tengo terminado, pero lo segundo tengo dudas ya que no quiero guardar las imagenes con su nombre original y en su lugar uso el número del array:

Código (php) [Seleccionar]
$tmp_name   = $_FILES['file']['tmp_name'];

for($i = 0; $i < count($tmp_name); $i++) {

   $img_name   = '/' . ($i + 1) . '.jpg';

}


Por lo tanto si uso este código la segunda vez que suba imágenes, las que ya existen serán reemplazadas. No me importa si el orden se pierde (pero si es posible prefiero algo limpio):

1.jpg, 2.jpg, 3.jpg ---> 1.jpg, 2.jpg, 3.jpg, 4.jpg, 5.jpg

¿Conoceis alguna manera más limpia de hacer esto? No me importa el nombre que tengan, no es de ningún uso.

Podría usar la hora de subida, algo así, pero me gustaría saber vuestras opiniones y como lo haríais vosotros.

Esta es la función completa, ¿podria mejorarse para que fuera más eficiente?

Código (php) [Seleccionar]
public function UploadImages($files, $test = '') {

   $imagesPath     = '../static.website.com/images/property/' . $this->id;
   $thumbnailsPath = '../static.website.com/images/property/' . $this->id . '/thumbnails';

   $log = 'Logging Images Upload:' . "\n\n";
   $upload_err = FALSE;

   if ($test == 'new') {

       // Will be any problem with $thumbnailsPath?
       if (!mkdir($imagesPath) && !mkdir($thumbnailsPath)) {

           return array($upload_err, 'Upload Error: mkdir().');

       }

   }

   $imagesPath     = realpath($imagesPath);
   $thumbnailsPath = realpath($thumbnailsPath);

   // Is it ok to only check for the child directory?
   if ( FALSE !== $thumbnailsPath && is_dir($thumbnailsPath) ) {

       if (isset($files['file'])) {

           $tmp_name   = $files['file']['tmp_name'];
           $img_size   = $files['file']['size'];

           // At least 1 image
           if ($test == 'new' && $tmp_name[0] == '') {

               $upload_err = TRUE;
               return array($upload_err, '¡La primera imagen es necesaria!');

           }

           for($i = 0; $i < count($tmp_name); $i++) {

               if ($tmp_name[$i] != '') {

                   if ($img_size[$i] <= 7340032) {

                       $img_name       = '/' . ($i + 1) . '.jpg';
                       $thumbnail_name = '/thumbnail_' . ($i + 1) . '.jpg';

                       list($width, $height) = getimagesize($tmp_name[$i]);

                       if ($width >= 1100 && $height >= 650) {

                           $img    = imagecreatefromjpeg($tmp_name[$i]);
                           $img_p  = imagecreatetruecolor(1100, 650);
                           imagecopyresampled($img_p, $img, 0, 0, 0, 0, 1100, 650, $width, $height);
                           imagedestroy($img);
                           $img_t  = imagecreatetruecolor(180, 180);
                           imagecopyresampled($img_t, $img_p, 0, 0, 0, 0, 180, 180, 1100, 650);

                           if (!imagejpeg($img_p, $imagesPath . $img_name, 70) || !imagejpeg($img_t, $thumbnailsPath . $thumbnail_name, 70)) {

                               $upload_err = TRUE;
                               return array($upload_err, 'Error al subir la imagen (' . $files['file']['name'][$i] . ')');

                           } else $log .= 'IMG: ' . $img_name . ' (original name: ' . $files['file']['name'][$i] . ') uploaded successfully.' . "\n";

                           imagedestroy($img_p);
                           imagedestroy($img_t);

                       } else {
                           $upload_err = TRUE;
                           return array($upload_err, 'La imagen: ' . $files['file']['name'][$i] . ' debe tener como mínimo 1100px de ancho y 650px de alto.');
                       }
                   } else {
                       $upload_err = TRUE;
                       return array($upload_err, 'La imagen (' . $files['file']['name'][$i] . ') pesa más de 7 MB.');
                   }
               }
           }
       } else return array($upload_err, '$files[\'file\'] is not set:' . "\n" . var_dump($files['file']) . "\n" . 'Property ID: ' . $this->id);
   } else return array($upload_err, '$thumbnailsPath is false and/or not a dir: ' . $thumbnailsPath . '". Property ID: ' . $this->id);

   return $log;

}


Cualquier surgerencia es bienvenida. Gracias!

[u]nsigned

Yo te recomendaría que renombres las imagenes usando md5_file(). Entonces la única forma de que se te duplique un nombre es que sea una imagen ya existente. Y creo que tenes un error, porque $_FILES['file']['tmp_name'] devuelve una cadena de texto con la ubicación temporal del archivo, no un array. Deberias iterar sobre $_FILES, que es el array de archivos recibidos por el script.

Tu código recibiría varios archivos a la vez o solo se sube de a uno?

No hay atajo ante la duda, el misterio se hace aquí...
Se hace carne en cada uno, el misterio es existir!

gAb1

Si, ya habia leido en alguna parte lo de usar md5, mientras sea una opción segura. La única pega es que el output es un poco largo para mi gusto, pero bueno supongo que no es relevante (yo y mis manías  :P)

El array $_FILES con más de un archivo tiene la siguiente estructura:

Código (php) [Seleccionar]
array(1) {
  ["file"]=>
  array(5) {
    ["name"]=>
    array(3) {
      [0]=>
      string(6) "01.jpg"
      [1]=>
      string(6) "01.jpg"
      [2]=>
      string(6) "01.jpg"
    }
    ["type"]=>
    array(3) {
      [0]=>
      string(10) "image/jpeg"
      [1]=>
      string(10) "image/jpeg"
      [2]=>
      string(10) "image/jpeg"
    }
    ["tmp_name"]=>
    array(3) {
      [0]=>
      string(14) "/tmp/phpRBV68j"
      [1]=>
      string(14) "/tmp/phpjuvvG9"
      [2]=>
      string(14) "/tmp/phpJpxmoZ"
    }
    ["error"]=>
    array(3) {
      [0]=>
      int(0)
      [1]=>
      int(0)
      [2]=>
      int(0)
    }
    ["size"]=>
    array(3) {
      [0]=>
      int(74357)
      [1]=>
      int(74357)
      [2]=>
      int(74357)
    }
  }
}


Si, recibe varios a la vez.Estoy usando el plugin dropzone con la opción de multiples archivos.

[u]nsigned

Para mi es la mejor opción, y no es tan larga, siempre serán 32 caracteres. Lo importantes es usar md5_file y no md5 solo, asi el hash se hace al contenido binario del archivo y no al nombre. Asi, la unica forma de que se te reemplace un archivo es que sea exactamente la misma foto.

Por cuestiones de seguridad siempre debes renombrar los archivos que subes y nunca suar los nombres con los que el usuario te los envia, porque podrias dejar una puerta abierta a vulnerabilidades.

Saludos!

No hay atajo ante la duda, el misterio se hace aquí...
Se hace carne en cada uno, el misterio es existir!

gAb1

Vale pero md5_file se lo hago al tmp_name y al nombre? Son dos cosas distintas. No sería md5 a name y md5_file a tmp_name?

[u]nsigned

Solo tenes que usar md5_file con tmp_name, ya que esta función recibe la ruta (path) del archivo y no el nombre. Una vez que ejecutes md5_file sobre tmp_name obtendrás un string de 32 caracteres, este sera el nombre final con el que copiaras la imagen a su destino definitivo. Y no olvides agregarle la extension (.jpg|.png).

No hay atajo ante la duda, el misterio se hace aquí...
Se hace carne en cada uno, el misterio es existir!

gAb1

Ah claro ya entiendo, si tiene sentido.

Gracias!