Desenvolvimento - PHP

Upload de Arquivos em PHP

O PHP pode fazer o upload de qualquer tipo de arquivo (texto ou binário). Leia nesse artigo os passos a se seguir e aprenda uma manipulação completa de uploads com o PHP.

por Alfred Reinold Baudisch



O PHP pode fazer o upload de qualquer tipo de arquivo (texto ou binário). Leia nesse artigo os passos a se seguir e aprenda uma manipulação completa de uploads com o PHP.

Esse artigo está dividido nos seguintes tópicos:

  1. Aspectos dos formulários de Upload
  2. Manipulando com o script PHP os arquivos enviados
  3. Verificando e limitando os arquivos enviados
    1. Tamanho dos arquivos
    2. Tipo dos arquivos
    3. Nome dos arquivos

1 - Aspectos dos formulários de Upload

Todo formulário de upload deve:

  1. Ser declarado com o método de envio POST.
  2. Conter a diretiva enctype="multipart/form-data", para "dizer" ao navegador que está se enviando arquivos e não somente os textos dos campos dos formulário
  3. Conter pelo menos 1 campo "file".

Isso em prática fica:

2 - Manipulando com o script PHP os arquivos enviados

Assume-se que o formulário acima foi postado enviando um arquivo. Automaticamente em "script.php" (o action do formulário exemplo), será criado as seguintes variáveis:

Observações:

- Em versões anteriores do PHP a 4.1.0 use a variável $HTTP_POST_FILES e não a variável superglobal $_FILES que foi introduzida na versão 4.1.0;
- No lugar de "arquivo" use o nome do seu campo "file" do formulário;
- Se a opção de configuração do PHP register_globals estiver como "on", você pode usar uma variável com o nome do campo de formulário, exemplo: $arquivo_name, $arquivo_size, etc. Mas, usar dessa maneira não é o recomendável, procure sempre usar a superglobal $_FILES.

O preferível no começo de "script.php" é passar a $_FILES para outra variável, por exemplo:

$arquivo  = isset($_FILES["arquivo"])  ? $_FILES["arquivo"] : FALSE;
$arquivo2 = isset($_FILES["arquivo2"]) ? $_FILES["arquivo2"] : FALSE;
[... etc ...]

Os arquivos enviados pelo formulário serão guardados no diretório temporário do servidor, a menos que outro lugar seja especificado com a opção upload_tmp_dir no php.ini. E o caminho e nome de cada arquivo, são ditos na $_FILES["arquivo"]["tmp_name"].

Para mover o arquivo para um diretório desejado deve-se usar a função move_uploaded_file() (em versões anteriores a 4.0.3 use a função copy).

Vamos então ao que interessa! O código de "script.php":

Esse é o código mínimo que seu script de upload deve conter. Mas, o mesmo ainda não é seguro, pois não faz nenhuma verificação no arquivo, seja de tamanho, nome, tipo, etc. O próximo tópico aborda isso.

3 - Verificando e limitando os arquivos enviados

3.1 - Tamanho dos arquivos

Existem várias formas de verificar o tamanho de arquivo enviado.

- A primeira limitação que você pode impor nos uploads é colocar o seguinte campo em seu formulário:

<input type="hidden" name="MAX_FILE_SIZE" value="30000">

Essa diretiva diz ao navegador o tamanho máximo de arquivo que pode ser enviado, onde o valor fornecido deve ser em bytes. Mas, é fácil contornar este limite, então não conte que o navegador irá obedecer a sua vontade. Mas você deve adicionar MAX_FILE_SIZE em qualquer caso, já que salva os usuarios que escapam do problema de esperar por um grande arquivo ser transferido e somente depois de tudo descobrir que ele é muito grande.

- Verificando com a variável $_FILES["arquivo"]["size"], onde no seu script haverá por exemplo um if:

- Verificando com a variável $_FILES["arquivo"]["error"]:

Ao enviar um formulário de upload, a variável $_FILES["arquivo"]["error"] poderá conter os seguintes valores:

Então, assuma que seu formulário de upload seja:

E o visitante faça um upload de um arquivo maior que o especificado em MAX_FILE_SIZE e maior que o da configuração upload_max_filesize. Seu script PHP (no caso script2.php), conterá as seguintes verificações:

3.2 - Tipo dos arquivos

Assuma que você possui um site com a área de usuários, onde os mesmos possam enviar suas fotos. O tipo de imagem aceita seria apenas PNG. Mas, como sempre tem os "engraçadinhos" que possam acabar a enviando arquivos não-imagem, você então deve verificar se o arquivo é uma imagem. Para isso, você pode verificar a extensão do nome de arquivo ou o tipo de arquivo.

Mas, conforme expliquei em meu artigo, "Upload de Imagens com Seguranga", verificar a extensão do nome de arquivo não é seguro (leia o artigo citado para mais informações), então o melhor é sempre verificar o mime tipo do arquivo. Para isso, usa-se então a variável $_FILES["arquivo"]["type"] criada no upload de um arquivo.

No nosso exemplo, onde aceitamos apenas imagens PNG o mime tipo é: image/png.
O script que verificaria isso:

Obs: Caso aceite mais tipos de arquivos, use expressões regulares na verificação. Há um exemplo disso no meu artigo "Upload de Imagens com Segurança".

3.3 - Nome dos arquivos

Vamos continuar com o nosso exemplo onde os usuários enviam suas fotos. Usuários podem acabar enviando arquivos com nomes com espaços, várias letras maiúsculas e minúsculas, etc... É recomendável você definir um padrão para os nomes de arquivos e o melhor é sempre colocar underscores ( _ ) no lugar de espaços e que todas as letras sejam minúsculas.

Assuma também que todas as fotos são salvas no mesmo diretório. Isso tem a falha de fotos de usuários acabarem ficando com o mesmo nome e se sobrescreverem, então 2 usuários diferentes tendo a mesma foto! Você pode verificar se o arquivo com mesmo nome já existe no diretório ou usar uma função que gera nomes únicos para a imagem. Aqui irei mostrar apenas a verificação do diretório, o nome único há no meu artigo "Upload de Imagens com Segurança".

Então é isso! Foi dada uma ampla explicação sobre os uploads de arquivos com o PHP. Futuramente estarei publicando um artigo explicando o upload de múltiplos arquivos direto do mesmo formulário.

Obs: Alguns textos e códigos foram pegos do Manual Oficial do PHP:
http://www.php.net/manual/pt_BR/features.file-upload.php

Obs2: Há uma lista ampla de Mime-Types aqui:
http://www.auriumsoft.com.br/mime_types.html

Alfred Reinold Baudisch
Desenvolvedor de Sistemas Web
alfred@auriumsoft.com.br
www.estacaonet.com

Alfred Reinold Baudisch

Alfred Reinold Baudisch - Desenvolvedor web freelance, com atuação na área há 7 anos. Experiência avançada em PHP, SQL e modelagem de sistemas multi-camadas. Atualmente dedicado ao aprendizado em desenvolvimento mobile, especificamente mobile games, com J2ME. Apaixonado e conhecedor do mercado financeiro, gestão e estratégias de novos negócios, visão constantemente empreendedora. Editor dos blogs Jornada Imperial e O Desenvolvedor PHP.