A API Blobstore permite que a sua aplicação publique objetos de dados, denominados blobs,que são muito maiores do que o tamanho permitido para objetos no serviço Datastore. Os BLOBs são úteis para publicar ficheiros grandes, como ficheiros de vídeo ou de imagem, e para permitir que os utilizadores carreguem ficheiros de dados grandes. Os blobs são criados carregando um ficheiro através de um pedido HTTP. Normalmente, as suas aplicações fazem isto apresentando um formulário com um campo de carregamento de ficheiros ao utilizador. Quando o formulário é enviado, o Blobstore cria um blob a partir do conteúdo do ficheiro e devolve uma referência opaca ao blob, denominada chave do blob,que pode usar posteriormente para publicar o blob. A aplicação pode publicar o valor do objeto BLOB completo em resposta a um pedido do utilizador ou pode ler o valor diretamente através de uma interface de streaming semelhante a um ficheiro.
Apresentamos o Blobstore
O App Engine inclui o serviço Blobstore, que permite que as aplicações forneçam objetos de dados limitados apenas pela quantidade de dados que podem ser carregados ou transferidos através de uma única ligação HTTP. Estes objetos são denominados valores do Blobstore ou blobs. Os valores do Blobstore são fornecidos como respostas dos controladores de pedidos e são criados como carregamentos através de formulários Web. As aplicações não criam dados de blobs diretamente. Em alternativa, os blobs são criados indiretamente por um formulário Web enviado ou outro pedido HTTP POST. Os valores do Blobstore podem ser fornecidos ao utilizador ou acedidos pela aplicação num fluxo semelhante a um ficheiro, através da API Blobstore.
Para pedir a um utilizador que carregue um valor do Blobstore, a sua aplicação apresenta um formulário Web com um campo de carregamento de ficheiros. A aplicação gera o URL de ação do formulário chamando a API Blobstore. O navegador do utilizador carrega o ficheiro diretamente para o Blobstore através do URL gerado. Em seguida, o Blobstore armazena o blob, reescreve o pedido para conter a chave do blob e transmite-o a um caminho na sua aplicação. Um controlador de pedidos nesse caminho na sua aplicação pode realizar o processamento adicional de formulários.
Para publicar um objeto BLOB, a sua aplicação define um cabeçalho na resposta de saída e o App Engine substitui a resposta pelo valor do objeto BLOB.
Não é possível modificar os blobs depois de criados, embora seja possível eliminá-los. Cada blob tem um registo de informações do blob correspondente, armazenado no arquivo de dados, que fornece detalhes sobre o blob, como a hora de criação e o tipo de conteúdo. Pode usar a chave do blob para obter registos de informações de blobs e consultar as respetivas propriedades.
Uma aplicação pode ler um valor do Blobstore uma parte de cada vez através de uma chamada da API.
O tamanho da parte pode ser até ao tamanho máximo de um valor de retorno da API.
Este tamanho é um pouco inferior a 32 megabytes, representado
em Python pela
constante google.appengine.ext.blobstore.MAX_BLOB_FETCH_SIZE
.
Uma aplicação não pode criar nem modificar valores do Blobstore, exceto através de ficheiros carregados pelo utilizador.
Usar o Blobstore
As aplicações podem usar o Blobstore para aceitar ficheiros grandes como carregamentos de utilizadores e para disponibilizar esses ficheiros. Os ficheiros são denominados blobs assim que são carregados. As aplicações não acedem diretamente aos blobs.Em alternativa, as aplicações funcionam com blobs através de entidades de informações de blobs (representadas pela classe
BlobInfo
) no Datastore.
O utilizador cria um objeto BLOB enviando um formulário HTML que inclui um ou mais campos de entrada de ficheiros. A sua aplicação
chama
blobstore.create_upload_url()
para obter
o destino (ação) deste formulário, transmitindo à função
um caminho de URL de um controlador na sua aplicação. Quando o utilizador envia o formulário,
o navegador do utilizador carrega os ficheiros especificados diretamente para o Blobstore. O Blobstore reescreve o pedido do utilizador e armazena os dados do ficheiro carregado, substituindo os dados do ficheiro carregado por uma ou mais chaves de blobs correspondentes. Em seguida, transmite o pedido reescrito ao controlador no caminho do URL que forneceu a
blobstore.create_upload_url()
.
Este controlador pode fazer um processamento adicional com base na chave do objeto binário grande.
A aplicação pode ler partes de um valor do Blobstore através de uma interface de streaming semelhante a um ficheiro.
interface de streaming semelhante a ficheiros. Consulte a turma BlobReader.
Carregar um blob
Para criar e carregar um blob, siga este procedimento:
1. Crie um URL de carregamento
Chame
blobstore.create_upload_url()
para criar um URL de carregamento para o formulário que o utilizador vai preencher,
transmitindo o caminho da aplicação a carregar quando o POST do formulário estiver concluído.
Existe uma versão assíncrona, create_upload_url_async(). Permite que o código da sua aplicação continue a ser executado enquanto o Blobstore gera o URL de carregamento.
2. Crie um formulário de carregamento
O formulário tem de incluir um campo de carregamento de ficheiros e o enctype do formulário tem de estar definido como multipart/form-data. Quando o utilizador envia o formulário, o POST é processado pela API Blobstore, que cria o blob. A API também cria um registo de informações para o blob e armazena o registo no Datastore, e transmite o pedido reescrito à sua aplicação no caminho indicado como uma chave de blob.
Tem de publicar a página do formulário com um Content-Type de text/html; charset=utf-8 ou os nomes de ficheiros com carateres não ASCII vão ser mal interpretados.
Uma vez que o Blobstore para Python 3 não usa o webapp, tem de definir o seu próprio Content-Type para impedir que a sua estrutura Web defina um tipo de conteúdo predefinido e o App Engine defina este tipo como um tipo presumido.
Não pode usar um Application Load Balancer externo global com um NEG sem servidor para processar pedidos de carregamento enviados para o URL /_ah/upload/ devolvido pela chamada blobstore.create_upload_url. Em alternativa, tem de encaminhar esses pedidos de carregamento diretamente para o serviço do App Engine. Pode fazê-lo através do domínio appspot.com ou de um domínio personalizado mapeado diretamente para o serviço do App Engine.
3. Implemente o controlador de carregamento
Neste controlador, pode armazenar a chave do objeto BLOB com o resto do modelo de dados da sua aplicação. A própria chave do blob permanece acessível a partir da entidade de informações do blob no Datastore. Tenha em atenção que, depois de o utilizador enviar o formulário e o seu controlador ser chamado, o blob já foi guardado e as informações do blob foram adicionadas ao Datastore. Se a sua aplicação não quiser manter o blob, deve eliminá-lo imediatamente para evitar que fique órfão.
Para todas as apps Flask, todas as chamadas feitas a métodos na classeBlobstoreUploadHandler
requerem o request.environ dictionary (pedido a ser importado do módulo flask). Se a sua app for uma app WSGI sem uma framework Web, usa o parâmetro environ no método get_uploads().
Ao reescrever o pedido do utilizador, o Blobstore esvazia as partes MIME dos ficheiros carregados e adiciona a chave do blob como um cabeçalho de parte MIME. O Blobstore
preserva todos os outros campos e partes do formulário, transmitindo-os ao controlador de carregamento.
Se não especificar um tipo de conteúdo, o Blobstore tenta inferi-lo a partir da extensão do ficheiro. Se não conseguir determinar um tipo de conteúdo, atribui o tipo de conteúdo application/octet-stream ao blob recém-criado.
Publicar um blob
Para publicar blobs, tem de incluir um controlador de transferência de blobs como um caminho na sua aplicação.
A aplicação publica um blob definindo um cabeçalho na resposta de saída. Se usar o Flask, a classe BlobstoreDownloadHandler requer o dicionário request.environ (pedido a ser importado do módulo Flask). Se a sua app for uma app WSGI sem uma framework Web, usa o parâmetro environ nos métodos send_blob().
Os blobs podem ser publicados a partir de qualquer URL de aplicação. Para publicar um blob na sua aplicação, coloca um cabeçalho especial na resposta que contém a chave do blob. O App Engine substitui o corpo da resposta pelo conteúdo do blob.
Intervalos de bytes de BLOB
O Blobstore suporta a publicação de parte de um valor grande em vez do valor completo
em resposta a um pedido. Para publicar um valor parcial, inclua o cabeçalho X-AppEngine-BlobRange
na resposta de saída. O seu valor é um intervalo de bytes HTTP padrão. A numeração de bytes baseia-se em zero. Um X-AppEngine-BlobRange
em branco indica à API para ignorar o cabeçalho de intervalo e disponibilizar o blob completo.
Alguns exemplos de intervalos:
0-499publica os primeiros 500 bytes do valor (bytes 0 a 499, inclusive).500-999serve 500 bytes a partir do 501.º byte.500-publica todos os bytes a partir do 501.º byte até ao final do valor.-500publica os últimos 500 bytes do valor.
Se o intervalo de bytes for válido para o valor do Blobstore, o Blobstore envia um código de estado 206 Partial Content e o intervalo de bytes pedido ao cliente. Se o intervalo não for válido para o valor,
o Blobstore envia 416 Requested Range Not Satisfiable.
O Blobstore não suporta vários intervalos de bytes num único pedido (por exemplo,
100-199,200-299), quer se sobreponham ou não.
Aplicação de amostra completa
Consulte o exemplo da app Flask no guia da API Blobstore para Python 3.
Usar o serviço Images com o Blobstore
O serviço Images pode usar um valor Blobstore como origem de uma transformação. A imagem de origem pode ter o tamanho máximo de um valor do Blobstore. O serviço Images continua a devolver a imagem transformada à aplicação, pelo que a imagem transformada tem de ter menos de 32 megabytes. Isto é útil para criar imagens de miniaturas de fotografias grandes carregadas pelos utilizadores. Para obter informações sobre a utilização do serviço Images com valores Blobstore, consulte a Documentação do serviço Images.
Usar a API Blobstore com o Cloud Storage
Pode usar a Blobstore API para armazenar blobs no Cloud Storage em vez de os armazenar no Blobstore. Tem de configurar um contentor, conforme descrito na documentação do
Cloud Storage
e especificar o contentor e o nome do ficheiro em
blobstore.create_upload_url parâmetro gs_bucket_name.
No controlador de carregamento, tem de processar o mapa de metadados FileInfo e armazenar explicitamente o nome do ficheiro do Cloud Storage necessário para obter o blob mais tarde.
Também pode publicar objetos do Cloud Storage através da API Blobstore.
Se quiser uma solução de armazenamento de objetos mais moderna, considere migrar do Blobstore do App Engine para o Cloud Storage.
Usar o BlobReader
Uma aplicação pode ler dados de valores do Blobstore através de uma interface semelhante a um objeto file do Python. Esta interface pode começar a ler um valor em qualquer posição de bytes e usa
várias chamadas de serviço e armazenamento em buffer, para que uma aplicação possa aceder ao tamanho total
do valor, apesar do limite no tamanho de uma única resposta de chamada de serviço.
A classe BlobReader
pode assumir um de três valores como argumento para o respetivo construtor:
O objeto implementa os métodos de ficheiro familiares para ler o valor. A aplicação não pode modificar o valor do Blobstore; os métodos de ficheiros para escrita não estão implementados.
Fazer pedidos assíncronos
Uma aplicação pode chamar algumas funções do Blobstore que funcionam em segundo plano.
O Blobstore executa o pedido enquanto a aplicação faz outras coisas. Para fazer o pedido, a aplicação chama uma função assíncrona. A função devolve imediatamente um objeto RPC. Este objeto representa o pedido. Quando a aplicação precisa do resultado do pedido, chama o método get_result() do objeto RPC.
Se o serviço não tiver concluído o pedido quando a aplicação chama get_result(), o método aguarda até que o pedido seja concluído (ou tenha atingido o prazo ou ocorra um erro). O método devolve o objeto de resultado ou gera uma exceção se ocorrer um erro ao executar o pedido. Por exemplo, este fragmento do código
upload_url = blobstore.create_upload_url('/upload')
slow_operation()
self.response.out.write("""<form action="%s" method="POST"
enctype="multipart/form-data">""" % upload_url)
passa a
upload_url_rpc = blobstore.create_upload_url_async('/upload')
slow_operation()
upload_url = upload_url_rpc.get_result()
self.response.out.write("""<form action="%s" method="POST"
enctype="multipart/form-data">""" % upload_url)
Neste exemplo, a aplicação executa o código slow_operation() ao mesmo tempo que o Blobstore gera o URL de carregamento.
Quotas e limites
O espaço usado para valores do Blobstore contribui para a quota de dados armazenados (faturáveis). As entidades de informações de blobs no Armazenamento de dados são contabilizadas para os limites relacionados com o Armazenamento de dados. Tenha em atenção que o Cloud Storage é um serviço pago. A faturação é feita de acordo com a folha de preços do Cloud Storage.
Para mais informações sobre as quotas de segurança ao nível do sistema, consulte o artigo Quotas.
Além das quotas de segurança ao nível do sistema, os seguintes limites aplicam-se especificamente à utilização do Blobstore:
- O tamanho máximo dos dados do Blobstore que podem ser lidos pela aplicação com uma chamada API é de 32 megabytes.
- O número máximo de ficheiros que podem ser carregados num único POST de formulário é 500.