Diferencia entre revisiones de «Plantilla:PHP/Formularios/transmisionFicheros»

De WikiEducator
Saltar a: navegación, buscar
(Tipo de fichero)
 
(17 revisiones intermedias por el mismo usuario no mostrado)
Línea 1: Línea 1:
===Transfirieno ficheros entre clinte y servidor ===
+
===Transfiriendo ficheros entre cliente y servidor ===
 
*Es muy sencillo y frecuente subir ficheros entre cliente y servidor
 
*Es muy sencillo y frecuente subir ficheros entre cliente y servidor
 
[[Imagen:subirFicheros.png]]
 
[[Imagen:subirFicheros.png]]
Línea 5: Línea 5:
 
====Acciones en el Cliente====
 
====Acciones en el Cliente====
 
;input type=file
 
;input type=file
*Debemos especificar un elemento '''''input''''' con  de '''type''' '''''file''''' en un formulario
+
*Debemos especificar un elemento '''''input''''' con  de '''type''' '''''file''''' en un formulario.
*Como todo input debe tener asignado un '''name''' para acceder a él en el servidor  
+
*Como todo input debe tener asignado un '''name''' para acceder a él en el servidor.
 
<source lang=html5>
 
<source lang=html5>
 
<input type=file name=fichero>
 
<input type=file name=fichero>
Línea 12: Línea 12:
 
;form method=POST enctype="multipart/form-data
 
;form method=POST enctype="multipart/form-data
 
*El formulario donde esté el '''input''' ha de tener especificado el atributo '''''enctype''''' establecido con el valor '''''mutipart/form-data'''''.
 
*El formulario donde esté el '''input''' ha de tener especificado el atributo '''''enctype''''' establecido con el valor '''''mutipart/form-data'''''.
*Cuando no especificamos tipo, se asume por defecto el valor  '''application/x-www-form-urlencoded'''. Este valor implica que enviamos texto plano y lo podremos enviar tanto por GET como por POST.
+
*Cuando no especificamos valor a este atributo, se asume por defecto el valor  '''application/x-www-form-urlencoded'''.  
*No obstante si vamos a transferir un fichero no necesariamente de texto '''''debemos''''' especificarlo establemciendo el valor de '''''enctype''''' a '''mutipart/form-data'''. Este valor se emplea para tranferir gran cantidad de texto u otros formatos de fichero entre cliente y servidor
+
*Este valor implica que enviamos texto plano y lo podremos enviar tanto por GET como por POST.
 +
*No obstante si vamos a transferir un fichero no necesariamente de texto '''''debemos''''' especificarlo estableciendo el valor de '''''enctype''''' a '''mutipart/form-data'''.
 +
*Este valor se emplea para transferir gran cantidad de texto u otros formatos de fichero entre cliente y servidor.
 
{{MRM_Recursos de la Web|
 
{{MRM_Recursos de la Web|
 
https://www.w3.org/TR/1999/REC-html401-19991224/interact/forms.html#form-content-type
 
https://www.w3.org/TR/1999/REC-html401-19991224/interact/forms.html#form-content-type
 
}}
 
}}
'''''enctype''''' es un atributo necesario para especificar el tipo de contendio usado para enviar la información del formulario al servidor
+
'''''enctype''''' es un atributo necesario para especificar el tipo de contenio usado para enviar la información del formulario al servidor.
*Necesariamente hemos de usar el método POST para este cometido
+
*Necesariamente hemos de usar el método POST para este cometido.
 
<source lang=html5>
 
<source lang=html5>
<form action='' method=POST entype='multipart/form-data'>
+
<form action="index.php" method="POST" entype='multipart/form-data'>
 
....
 
....
 
</form>
 
</form>
 
</source>
 
</source>
 
;Establecer tamaño en el cliente
 
;Establecer tamaño en el cliente
*El tamaño de bytes que vamos a enviar  también puede quedar establecido en el cliente, de modo que si el fichero tiene un tamaña mayor, no se envía  
+
*El tamaño de bytes que vamos a enviar  también puede quedar establecido en el cliente, de modo que si el fichero tiene un tamaña mayor, no se envía.
*Para esto se establece antes del input file, un input hidden con name = MAX_SIZE_FILE y value el valor del tamaño máximo.
+
*Para esto se establece antes del input file, un input hidden con '''name''' a MAX_SIZE_FILE y '''value''' el valor del tamaño máximo en bytes.
*Este mecanismo no avisa al cliente de nada, simplemente dejará de enviar el fichero al servidor en caso de exceder el tamaño.  
+
<source lang=html5>
 +
....
 +
<form action="index.php" method="POST" entype='multipart/form-data'>
 +
 
 +
<input type=hidden name ="MAX_SIZE_FILE" value=10000000>
 +
<input type=file name=fichero>
 +
....
 +
</form>
 +
 
 +
</source>
 +
*Este mecanismo no envía nada al servidor, dejará de enviar el fichero al servidor .
 
*En el servidor se recibirá un error de valor '''''2''''' o constante  UPLOAD_ERR_FORM_SIZE, (Ver código de errores más abajo o en '''''http://php.net/manual/es/features.file-upload.errors.php '''''
 
*En el servidor se recibirá un error de valor '''''2''''' o constante  UPLOAD_ERR_FORM_SIZE, (Ver código de errores más abajo o en '''''http://php.net/manual/es/features.file-upload.errors.php '''''
 
*Con todo lo dicho, la especificación en el cliente quedaría
 
*Con todo lo dicho, la especificación en el cliente quedaría
Línea 40: Línea 52:
 
</source>
 
</source>
 
====Acciones en el Servidor: $_FILES====
 
====Acciones en el Servidor: $_FILES====
Con el fichero que viene del cliente, en el servidor podemos hacer:
+
*La forma de acceder al input del tipo file que viene del cliente en la solicitud al servidor es a través de la superglobal '''''$_FILES'''''.
#Capturar el fichero y dejarlo en un directorio concreto
+
*Lo primero que deberemos hacer es acceder a este elemento con el nombre del input.
#Ver si se ha producido algún error especificando el código de error mediante una constante numérica
+
<source lang=html5>
#Ver el tamaño del fichero
+
$fichero = $_FILES['nombre_input_file']
#Analizar el tipo de fichero para poder por ejemplo aceptarlo o descartarlo o decidir en qué carpeta dejarlo en función del tipo
+
</source>
;Copiando el fichero a una carpeta
+
*$_FILES es un array asociativo con tantos elementos con input de tipo file como vengan del formulario cuyo submit ha generado una solicitud al servidor.
*Lógicamente primero deberemos crear esa carpeta y asegurarnos que tenga permisos de escritura en ella el usuario apache (normalmente www-data)
+
*Cada posición a su vez contiene un array asociativo con información de ese fichero  almacenada en 5 componentes:
*Cuando viene la solicitud del cliente con un fichero como contenido de un formulario, se accede a ese input a través de la superglobal '''''$_FILES'''''.
+
*$_FILES va a ser un array asociativo con un elemento por cada input de tipo file que venga del servidor, o sea por cada fichero subido
+
*Cada posición a su vez contiene un array asociativo con información de ese fichero donde entre otros tenemos los siguientes campos que nos interesan
+
 
# '''''name''''' Nombre del fichero en el cliente
 
# '''''name''''' Nombre del fichero en el cliente
 
# '''''type''''' Tupo de fichero subido
 
# '''''type''''' Tupo de fichero subido
 
# '''''size''''' Tamaño en bytes del fichero  
 
# '''''size''''' Tamaño en bytes del fichero  
 
# '''''tpm_name''''' Nombre asignado de forma temporal en el servidor  
 
# '''''tpm_name''''' Nombre asignado de forma temporal en el servidor  
# '''''error''''' Error que se haya podido producir o 0 si no ha habido ninguno
+
# '''''error''''' Error que se haya podido producir o 0 si no ha habido ninguno (Ver tabla más abajo)
  
{|style="width:65%" height:200px" border="1" cellpadding="5" cellspacing="0" align="center" border = 1
+
 
|+ <FONT COLOR="Blue"><span style="font-size:11pt;">CÓDIGOS DE ERROR SUBIENDO FICHEROS</span></font><br/><br/>
+
Con el fichero que viene del cliente, en el servidor podemos hacer una serie de acciones:
!Valor enterio !! Constante !! Descripción !
+
#Capturar el fichero y dejarlo en un directorio concreto.
 +
#Ver si se ha producido algún error especificando el código de error mediante una constante numérica.
 +
#Ver el tamaño del fichero.
 +
#Analizar el tipo de fichero para poder, por ejemplo, aceptarlo o descartarlo, o decidir en qué carpeta dejarlo en función del tipo. (Tener en cuenta que haya permisos de escritura para el usuario de apache (www-data) en la carpeta donde queramos dejar el fichero)
 +
=====Copiando el fichero a una carpeta=====
 +
*La primera acción será copiarnos el fichero en una ubicación concreta dentro de nuestro servidor
 +
*Lógicamente primero deberemos crear esa carpeta y asegurarnos que tenga permisos de escritura en ella el usuario apache (normalmente www-data)
 +
{{MRM_Actividad|title=Dando permisos de escritura|
 +
<source lang=bash>chmod +w ....
 +
</source>}}
 +
*En el servidor tenemos el fichero disponible de forma temporal en la capeta /tmp. Podemos acceder a esta información en el elemento '''''$_FILES['tmp_name']'''''
 +
*Para copiarlo usaremos la función '''''move_uploaded_file($origen, $destino);''''', dónde '''''$origen''''' es el fichero que queremos copiar con ubicación y $destino es la ubicación y nombre de fichero donde queremos dejarlo.
 +
*Lo más habitual es dejar el fichero con el mismo nombre que tenía en el cliente, esta información la tenemos disponible en el atributo '''''$_FILES['name']'''''
 +
*La función '''''move_upload(..)''' retorna un booleano que indica el éxito o fracaso de la acción. http://php.net/manual/es/function.move-uploaded-file.php.
 +
*A continuación un resumen de estas acciones
 +
<source lang=php>
 +
//Suponemos en el cliente
 +
//... <input type=file name= fichero>
 +
//
 +
//Accedemos al fichero que está de forma temporal en el servidor
 +
$origen = $_FILES['fichero']['tmp_name'];
 +
/Accedemos al nombre del fichero con el que el cliente lo subió
 +
$nombreFichero = $_FILES['fichero']['name'];
 +
//Establecemos la ruta donde queremos dejar el fichero
 +
//En este caso en la carpeta del proyecto tenemos una carpeta llamada descargas con permiso de escritura para www-data
 +
$destino ="./descargas/".$nombreFichero
 +
//Ahora procedemos a copiar y ver el éxito o fracaso
 +
if (move_uploaded_file($origen, $destino))
 +
  echo ("El fichero $nombreFichero se ha subido correctamente");
 +
else
 +
  echo ("Error subiendo el fichero $nombreFichero");
 +
</source>
 +
{{MRM_Actividad|Titel=Probando las acciones|
 +
*Vamos a realizar una práctica dónde verifiquemos el funcionamiento de todo lo visto anteriormente.
 +
*Sé curiosa y comprueba los atributos comentados.
 +
}}
 +
 
 +
=====Comprobando errores=====
 +
* '''''$_FILES[error]''''' contiene información del error que se ha podido producir al subir el fichero
 +
*La siguiente tabla es la lista de los posibles valores que va a haber en este elemento del array superglobal $_FILES
 +
[[Imagen:inputFilePhp.png]]
 +
{|style="width:65%" height:200px" border="1" cellpadding="5" cellspacing="0" align="center"
 +
|+ <FONT COLOR="Blue"><span style="font-size:11pt;">CÓDIGOS DE ERROR SUBIENDO FICHEROS</span><br/><br/>
 +
!Valor entero !! Constante !! Descripción </font>
 
|-
 
|-
 
|align="center"|0
 
|align="center"|0
|align="center"|UPLOAD_ERR_OK
+
|align="center"|'''''UPLOAD_ERR_OK'''''
|align="center"|Fichero subido exitonsamente
+
|align="center"|Fichero subido exitosamente
 
|-
 
|-
 
|align="center"|1
 
|align="center"|1
|align="center"|UPLOAD_ERR_INI_SIZE
+
|align="center"|'''''UPLOAD_ERR_INI_SIZE'''''
|align="center"|tamaño excedido según directiva '''''upload_max_filesize''''' de php.ini.
+
|align="center"|''Tamaño excedido según directiva'' '''''upload_max_filesize''''' ''de php.ini.''
 
|-
 
|-
 
|align="center"| 2
 
|align="center"| 2
|align="center"|UPLOAD_ERR_FORM_SIZE
+
|align="center"|'''''UPLOAD_ERR_FORM_SIZE'''''
|align="center"|El fichero subido excede la directiva MAX_FILE_SIZE especificada en el formulario HTML.
+
|align="center"|''El fichero subido excede la directiva MAX_FILE_SIZE especificada en el formulario HTML.''
 
|-
 
|-
 
|align="center"| 3
 
|align="center"| 3
|align="center"| UPLOAD_ERR_PARTIAL
+
|align="center"| '''''UPLOAD_ERR_PARTIAL'''''
|align="center"| El fichero fue sólo parcialmente subido.
+
|align="center"| ''El fichero fue sólo parcialmente subido.''
 
|-
 
|-
 
|align="center"| 4
 
|align="center"| 4
|align="center"| UPLOAD_ERR_NO_FILE
+
|align="center"| '''''UPLOAD_ERR_NO_FILE'''''
|align="center"| No se subió ningún fichero.
+
|align="center"| ''No se subió ningún fichero.''
|-
+
|align="center"| 5
+
|align="center"| UPLOAD_ERR_NO_TMP_DIR
+
|align="center"|
+
 
|-
 
|-
 
|align="center"| 6
 
|align="center"| 6
|align="center"| UPLOAD_ERR_CANT_WRITE
+
|align="center"| '''''UPLOAD_ERR_NO_TMP_DIR'''''
|align="center"| Falta la carpeta temporal. Introducido en PHP 5.0.3.
+
|align="center"| ''Falta la carpeta temporal. ''
 
|-
 
|-
 
|align="center"| 7
 
|align="center"| 7
|align="center"|  
+
|align="center"| '''''UPLOAD_ERR_CANT_WRITE'''''
|align="center"| No se pudo escribir el fichero en el disco. Introducido en PHP 5.1.0.
+
|align="center"| ''No se pudo escribir el fichero en el disco. ''
 
|-
 
|-
 
|align="center"| 8
 
|align="center"| 8
|align="center"| UPLOAD_ERR_EXTENSION
+
|align="center"| '''''UPLOAD_ERR_EXTENSION'''''
|align="center"| Una extensión de PHP detuvo la subida de ficheros
+
|align="center"| ''Una extensión de PHP detuvo la subida de ficheros.''
 
|}
 
|}
 
+
{{MRM_Recursos de la Web|
*Una vez presionado el submit en el cliente el fichero se copia en un fichero en una carpeta temporal
+
http://php.net/manual/es/features.file-upload.errors.php
;Almacenando el fichero en el servidor
+
}}
*Una vez que está subido en la carpeta temporal y después de hacer las comprobaciones que consideremos, debemos guardarlo en el servidor
+
*Un posible código para obtener esta información
*Para ello usaremos la función
+
 
<source lang=php>
 
<source lang=php>
  $rtdo= move_uploaded_file($origen, $destino);
+
//Suponemos en el cliente
</source>
+
//... <input type=file name= fichero>
;Cómo leer ficheros de un directorio  en el servidor
+
 
*Vamos a usar la clase Directory
+
$fichero = $_FILES['fichero'];
*El tema de programación orientado a objetos lo veremos mas tarde, pero sabemos que para invocar a un método de un objeto se usa el operador de indierección '''''->'''''
+
.....
*Los métodos los vemos como funciones que son de una clase y los puede invocar un objeto. (Esta idea imprecisa de momento es suficiente.
+
$error = $fichero['error']
*Es una clase que me permite trabajar con ficheros (los directorios son ficheros igualmente)
+
//Esto es igual que hacer $error = $_FILES['fichero']['error']
*Para inicializarlo podemos usar un alias de '''''new Directory()''''' llamada '''''dir(...)'''''
+
 
*En su invocación pasamos el directorio que queremos ver
+
$error = $_FILES['error'];
<source lang=php>
+
switch ($error){
  $directorio = dir("/var/www/musica/subidas/");
+
    case 0:
 +
        echo "ERROR. Fichero subido de forma correcta. <br />";
 +
        break;
 +
    case 1:
 +
        echo "ERROR. Tamaño de fichero superior al establecido en el servidor <br />";
 +
       
 +
        break;
 +
    case 2:
 +
        echo "ERROR. Tamaño de fichero superior al establecido en cliente<br />";
 +
        echo "El tamaño se estableció en el input MAX_FILE_SIZE<br/>";
 +
        echo "Tamaño establecido ".$_POST['MAX_FILE_SIZE']."<br/>";
 +
        break;
 +
    case 3:
 +
        echo "ERROR. EL fichero sólo se subió parcialmente <br/>";
 +
        break;
 +
    case 4:
 +
        echo "ERROR. No se subió ningún fichero <br/>";
 +
        break;
 +
    case 6:
 +
        echo "ERROR. No se encuentra la carpeta temporal <br/>";
 +
        break;
 +
    case 7:
 +
        echo "ERROR. No se pudo escribir en disco. revisa permisos <br/>";       
 +
        break;
 +
    case 8:
 +
        echo "ERROR. Una extensión de php detuvo la subida del fichero <br/>";       
 +
        break;
 +
    default:
 +
        echo "Valor de error desconocido";
 +
}
 +
 
 
</source>
 
</source>
  
*Para leer el contenido de un fichero o un directorio (los archivos que contiene), se usa el método '''''read()'''''
+
=====Ver tamaño del fichero y otras directivas en php.ini=====
*Una forma de hacerlo es
+
*El tamaño de fichero queda definido en el servidor por la directiva de '''''php.ini''''
 +
upload_max_filesize=
 +
*Otras directivas relacionadas con la descargas de ficheros están establecidas en php.ini
 +
*A continuación se detallan con sus valores por defecto. (Ver el fichero php.ini)
 +
<source lang=bash>
 +
;;;;;;;;;;;;;;;;
 +
; File Uploads ;
 +
;;;;;;;;;;;;;;;;
 +
 
 +
; Whether to allow HTTP file uploads.
 +
; http://php.net/file-uploads
 +
;Comentario: Permite la descarga de ficheros
 +
file_uploads = On
 +
 
 +
; Temporary directory for HTTP uploaded files (will use system default if not
 +
; specified).
 +
; http://php.net/upload-tmp-dir
 +
;;Comentario : Establece el directorio temporal en el servidor donde se deja temporalmente el fichero subido
 +
;;Si no se especifica se tomará /tmp en linux, o el directorio por defecto que use el SO de forma temporal
 +
;upload_tmp_dir =
 +
 
 +
; Maximum allowed size for uploaded files.
 +
; http://php.net/upload-max-filesize
 +
;;Comentario : Tamaño máximo del fichero permitido en el servidor
 +
;;Se puede usar los múltiplos K M G T
 +
upload_max_filesize = 20M
 +
 
 +
; Maximum number of files that can be uploaded via a single request
 +
;;Número máximo de ficheros que se pueden descargar en una sola solicitud httlp
 +
max_file_uploads = 20
 +
</source>
 +
 
 +
*Podemos ver el tamaño exacto del fichero subido mediante el elemento size del array
 
<source lang=php>
 
<source lang=php>
while ($archivo = $directorio -> read()){
+
//Suponemos en el cliente
        .................        }
+
//... <input type=file name= fichero>
}
+
....
</source>
+
$size = $_FILES['fichero']['size'];
*No olvidemos cerrar el fichero con el método close
+
....
<source lang=php>
+
    $directorio->close();
+
 
</source>
 
</source>
 +
*Entre otras cosas puede servir para descartar un fichero de menos de un tamaño concreto.
  
 +
=====Tipo de fichero=====
 +
*Este es un atributo importante
  
[[Actividad
+
#Analizar el tipo de fichero para poder por ejemplo aceptarlo o descartarlo o decidir en qué carpeta dejarlo en función del tipo
Se pide que hagas un programa para subir ficheros de música a un servidor y qué este me visualice todos los ficheros subidos]]
+
*Para ver el tipo podemos observar la extensión del fichero.
 +
*O bien analizar el tipo MIME que nos viene en '''''$_FILES['type']
 +
*Por ejemplo suponemos que queremos distribuir los ficheros en tres carpetas
 +
#Los ficheros que contengan imágenes a la carpeta '''''./descargas/imagenes/'''''
 +
#Los ficheros que contengan música a la carpeta '''''./descargas/musica/'''''
 +
#El resto de ficheros a la carpeta '''''./descargas/otros/'''''
 +
*En el tipo mime separa el tipo general del fichero con una barra.
 +
*Así los de tipo música o audio sería '''''audio/....'
 +
*Así los de tipo imagen  '''''image/....'
 +
.....
 +
 
 +
*Un posible código sería
 +
<source lang=php>
 +
//Suponemos en el cliente
 +
//... <input type=file name= fichero>
 +
 
 +
.....
 +
$origen = $_FILES['fichero']['tmp_name'];
 +
$nombreFichero = $_FILES['fichero']['name'];
 +
$tipo = $_FILES['fichero']['type'];
 +
$tipo_fichero = explode('/', $tipo);
 +
switch ($tipo_fichero[0]) {
 +
    case 'audio':
 +
        $dir_destino = "/var/www/descargas/uploads/musica";
 +
        break;
 +
    case 'image':
 +
        $dir_destino = "/var/www/descargas/uploads/imagenes";
 +
        break;
 +
    default:
 +
        $dir_destino = "/var/www/descargas/uploads/otros";
 +
}
 +
$destino = $dir_destino . '/' . basename($nombreFichero);
 +
move_uploaded_file($origen, $destino);
 +
</source>
 +
{{Tip| la función '''''explode'''''
 +
Esta función rompe una cadena de caracteres en diferentes campos de un array indexado cada vez que encuentre un determinado  carácter.  Tango el carácter, como la cadena son argumentos pasados a la función. http://php.net/manual/es/function.explode.php}}

Última revisión de 21:53 20 nov 2020

Transfiriendo ficheros entre cliente y servidor

  • Es muy sencillo y frecuente subir ficheros entre cliente y servidor

SubirFicheros.png

  • Cuando vamos a subir ficheros hay que conocer acciones a indicar tanto en la parte de cliente como en la de servidor.

Acciones en el Cliente

input type=file
  • Debemos especificar un elemento input con de type file en un formulario.
  • Como todo input debe tener asignado un name para acceder a él en el servidor.
<input type=file name=fichero>
form method=POST enctype="multipart/form-data
  • El formulario donde esté el input ha de tener especificado el atributo enctype establecido con el valor mutipart/form-data.
  • Cuando no especificamos valor a este atributo, se asume por defecto el valor application/x-www-form-urlencoded.
  • Este valor implica que enviamos texto plano y lo podremos enviar tanto por GET como por POST.
  • No obstante si vamos a transferir un fichero no necesariamente de texto debemos especificarlo estableciendo el valor de enctype a mutipart/form-data.
  • Este valor se emplea para transferir gran cantidad de texto u otros formatos de fichero entre cliente y servidor.



enctype es un atributo necesario para especificar el tipo de contenio usado para enviar la información del formulario al servidor.

  • Necesariamente hemos de usar el método POST para este cometido.
<form action="index.php" method="POST" entype='multipart/form-data'>
....
</form>
Establecer tamaño en el cliente
  • El tamaño de bytes que vamos a enviar también puede quedar establecido en el cliente, de modo que si el fichero tiene un tamaña mayor, no se envía.
  • Para esto se establece antes del input file, un input hidden con name a MAX_SIZE_FILE y value el valor del tamaño máximo en bytes.
....
<form action="index.php" method="POST" entype='multipart/form-data'>
 
 <input type=hidden name ="MAX_SIZE_FILE" value=10000000>
 <input type=file name=fichero>
....
</form>
  • Este mecanismo no envía nada al servidor, dejará de enviar el fichero al servidor .
  • En el servidor se recibirá un error de valor 2 o constante UPLOAD_ERR_FORM_SIZE, (Ver código de errores más abajo o en http://php.net/manual/es/features.file-upload.errors.php
  • Con todo lo dicho, la especificación en el cliente quedaría
  <form action="descarga.php" method="POST" enctype="multipart/form-data">
            <input type="hidden" name="MAX_FILE_SIZE" value="10000000">
            <h3>Selecciona fichero </h3>
            <input type="file" name="fichero" id="" >
            <br />
            <input type="submit" value="Acceder" name="descarga">
        </form>

Acciones en el Servidor: $_FILES

  • La forma de acceder al input del tipo file que viene del cliente en la solicitud al servidor es a través de la superglobal $_FILES.
  • Lo primero que deberemos hacer es acceder a este elemento con el nombre del input.
$fichero = $_FILES['nombre_input_file']
  • $_FILES es un array asociativo con tantos elementos con input de tipo file como vengan del formulario cuyo submit ha generado una solicitud al servidor.
  • Cada posición a su vez contiene un array asociativo con información de ese fichero almacenada en 5 componentes:
  1. name Nombre del fichero en el cliente
  2. type Tupo de fichero subido
  3. size Tamaño en bytes del fichero
  4. tpm_name Nombre asignado de forma temporal en el servidor
  5. error Error que se haya podido producir o 0 si no ha habido ninguno (Ver tabla más abajo)


Con el fichero que viene del cliente, en el servidor podemos hacer una serie de acciones:

  1. Capturar el fichero y dejarlo en un directorio concreto.
  2. Ver si se ha producido algún error especificando el código de error mediante una constante numérica.
  3. Ver el tamaño del fichero.
  4. Analizar el tipo de fichero para poder, por ejemplo, aceptarlo o descartarlo, o decidir en qué carpeta dejarlo en función del tipo. (Tener en cuenta que haya permisos de escritura para el usuario de apache (www-data) en la carpeta donde queramos dejar el fichero)
Copiando el fichero a una carpeta
  • La primera acción será copiarnos el fichero en una ubicación concreta dentro de nuestro servidor
  • Lógicamente primero deberemos crear esa carpeta y asegurarnos que tenga permisos de escritura en ella el usuario apache (normalmente www-data)


Icon activity.jpg
Actividad
chmod +w ....


  • En el servidor tenemos el fichero disponible de forma temporal en la capeta /tmp. Podemos acceder a esta información en el elemento $_FILES['tmp_name']
  • Para copiarlo usaremos la función move_uploaded_file($origen, $destino);, dónde $origen es el fichero que queremos copiar con ubicación y $destino es la ubicación y nombre de fichero donde queremos dejarlo.
  • Lo más habitual es dejar el fichero con el mismo nombre que tenía en el cliente, esta información la tenemos disponible en el atributo $_FILES['name']
  • La función move_upload(..) retorna un booleano que indica el éxito o fracaso de la acción. http://php.net/manual/es/function.move-uploaded-file.php.
  • A continuación un resumen de estas acciones
//Suponemos en el cliente
//... <input type=file name= fichero>
//
//Accedemos al fichero que está de forma temporal en el servidor
$origen = $_FILES['fichero']['tmp_name'];
/Accedemos al nombre del fichero con el que el cliente lo subió
$nombreFichero = $_FILES['fichero']['name'];
//Establecemos la ruta donde queremos dejar el fichero
//En este caso en la carpeta del proyecto tenemos una carpeta llamada descargas con permiso de escritura para www-data
$destino ="./descargas/".$nombreFichero
//Ahora procedemos a copiar y ver el éxito o fracaso
if (move_uploaded_file($origen, $destino))
   echo ("El fichero $nombreFichero se ha subido correctamente");
else
   echo ("Error subiendo el fichero $nombreFichero");


Icon activity.jpg
Actividad
  • Vamos a realizar una práctica dónde verifiquemos el funcionamiento de todo lo visto anteriormente.
  • Sé curiosa y comprueba los atributos comentados.




Comprobando errores
  • $_FILES[error] contiene información del error que se ha podido producir al subir el fichero
  • La siguiente tabla es la lista de los posibles valores que va a haber en este elemento del array superglobal $_FILES

InputFilePhp.png

CÓDIGOS DE ERROR SUBIENDO FICHEROS

Valor entero Constante Descripción
0 UPLOAD_ERR_OK Fichero subido exitosamente
1 UPLOAD_ERR_INI_SIZE Tamaño excedido según directiva upload_max_filesize de php.ini.
2 UPLOAD_ERR_FORM_SIZE El fichero subido excede la directiva MAX_FILE_SIZE especificada en el formulario HTML.
3 UPLOAD_ERR_PARTIAL El fichero fue sólo parcialmente subido.
4 UPLOAD_ERR_NO_FILE No se subió ningún fichero.
6 UPLOAD_ERR_NO_TMP_DIR Falta la carpeta temporal.
7 UPLOAD_ERR_CANT_WRITE No se pudo escribir el fichero en el disco.
8 UPLOAD_ERR_EXTENSION Una extensión de PHP detuvo la subida de ficheros.



  • Un posible código para obtener esta información
//Suponemos en el cliente
//... <input type=file name= fichero>
 
$fichero = $_FILES['fichero'];
.....
$error = $fichero['error']
//Esto es igual que hacer $error = $_FILES['fichero']['error']
 
$error = $_FILES['error'];
switch ($error){
    case 0:
        echo "ERROR. Fichero subido de forma correcta. <br />";
        break;
    case 1:
        echo "ERROR. Tamaño de fichero superior al establecido en el servidor <br />"; 
 
        break;
    case 2:
        echo "ERROR. Tamaño de fichero superior al establecido en cliente<br />"; 
        echo "El tamaño se estableció en el input MAX_FILE_SIZE<br/>"; 
        echo "Tamaño establecido ".$_POST['MAX_FILE_SIZE']."<br/>";
        break;
    case 3:
        echo "ERROR. EL fichero sólo se subió parcialmente <br/>";
        break;
    case 4:
        echo "ERROR. No se subió ningún fichero <br/>";
        break;
    case 6:
        echo "ERROR. No se encuentra la carpeta temporal <br/>";
        break;
    case 7:
        echo "ERROR. No se pudo escribir en disco. revisa permisos <br/>";        
        break;
    case 8:
        echo "ERROR. Una extensión de php detuvo la subida del fichero <br/>";        
        break;
    default:
        echo "Valor de error desconocido";
}
Ver tamaño del fichero y otras directivas en php.ini
  • El tamaño de fichero queda definido en el servidor por la directiva de php.ini'
upload_max_filesize=
  • Otras directivas relacionadas con la descargas de ficheros están establecidas en php.ini
  • A continuación se detallan con sus valores por defecto. (Ver el fichero php.ini)
;;;;;;;;;;;;;;;;
; File Uploads ;
;;;;;;;;;;;;;;;;
 
; Whether to allow HTTP file uploads.
; http://php.net/file-uploads
;Comentario: Permite la descarga de ficheros 
file_uploads = On
 
; Temporary directory for HTTP uploaded files (will use system default if not
; specified).
; http://php.net/upload-tmp-dir
;;Comentario : Establece el directorio temporal en el servidor donde se deja temporalmente el fichero subido
;;Si no se especifica se tomará /tmp en linux, o el directorio por defecto que use el SO de forma temporal
;upload_tmp_dir =
 
; Maximum allowed size for uploaded files.
; http://php.net/upload-max-filesize
;;Comentario : Tamaño máximo del fichero permitido en el servidor 
;;Se puede usar los múltiplos K M G T
upload_max_filesize = 20M
 
; Maximum number of files that can be uploaded via a single request
;;Número máximo de ficheros que se pueden descargar en una sola solicitud httlp
max_file_uploads = 20
  • Podemos ver el tamaño exacto del fichero subido mediante el elemento size del array
//Suponemos en el cliente
//... <input type=file name= fichero>
....
$size = $_FILES['fichero']['size'];
....
  • Entre otras cosas puede servir para descartar un fichero de menos de un tamaño concreto.
Tipo de fichero
  • Este es un atributo importante
  1. Analizar el tipo de fichero para poder por ejemplo aceptarlo o descartarlo o decidir en qué carpeta dejarlo en función del tipo
  • Para ver el tipo podemos observar la extensión del fichero.
  • O bien analizar el tipo MIME que nos viene en $_FILES['type']
  • Por ejemplo suponemos que queremos distribuir los ficheros en tres carpetas
  1. Los ficheros que contengan imágenes a la carpeta ./descargas/imagenes/
  2. Los ficheros que contengan música a la carpeta ./descargas/musica/
  3. El resto de ficheros a la carpeta ./descargas/otros/
  • En el tipo mime separa el tipo general del fichero con una barra.
  • Así los de tipo música o audio sería audio/....'
  • Así los de tipo imagen image/....'

.....

  • Un posible código sería
//Suponemos en el cliente
//... <input type=file name= fichero>
 
.....
$origen = $_FILES['fichero']['tmp_name'];
$nombreFichero = $_FILES['fichero']['name'];
$tipo = $_FILES['fichero']['type'];
$tipo_fichero = explode('/', $tipo);
switch ($tipo_fichero[0]) {
    case 'audio':
        $dir_destino = "/var/www/descargas/uploads/musica";
        break;
    case 'image':
        $dir_destino = "/var/www/descargas/uploads/imagenes";
        break;
    default:
        $dir_destino = "/var/www/descargas/uploads/otros";
}
$destino = $dir_destino . '/' . basename($nombreFichero);
move_uploaded_file($origen, $destino);
Icon present.gif
Tip: la función explode

Esta función rompe una cadena de caracteres en diferentes campos de un array indexado cada vez que encuentre un determinado carácter. Tango el carácter, como la cadena son argumentos pasados a la función. http://php.net/manual/es/function.explode.php