Menu Docs
Página inicial do Docs
/ / /
Driver C++
/

Armazenar arquivos grandes

Nesta página

  • Visão geral
  • Como funciona o GridFS
  • Crie um intervalo GridFS
  • Personalizar o bucket
  • Fazer upload de arquivos
  • Escrever em um fluxo de upload
  • Carregar um stream existente
  • Recuperar informações do arquivo
  • Exemplo
  • Baixar arquivos
  • Ler de um fluxo de download
  • Baixar em um stream existente
  • Excluir arquivos
  • Documentação da API

Neste guia, você pode aprender como armazenar e recuperar arquivos grandes no MongoDB usando GridFS. O GridFS é uma especificação implementada pelo driver C++ que descreve como divisão os arquivos em blocos ao armazená-los e remontá-los ao recuperá-los. A implementação do driver do GridFS é uma abstração que gerencia as operações e a organização do armazenamento de arquivos.

Use o GridFS se o tamanho dos seus arquivos exceder o limite de tamanho de documento BSON de 16MB. Para obter informações mais detalhadas sobre se o GridFS é adequado para seu caso de uso, consulte GridFS no manual do MongoDB Server .

O GridFS organiza os arquivos em um bucket, um grupo de coleções do MongoDB que contém os blocos de arquivos e as informações que os descrevem. O bloco contém as seguintes coleções, nomeadas usando a convenção definida na especificação do GridFS:

  • chunks collection, que armazena os chunks de arquivos binários

  • files coleção, que armazena os metadados do arquivo

O driver cria o bucket GridFS , se ele não existir, quando você grava dados pela primeira vez. O bucket contém as collections anteriores prefixadas com o nome de bucket padrão fs, a menos que você especifique um nome diferente. Para garantir a recuperação eficiente dos arquivos e de seus respectivos metadados, o driver cria um índice em cada collection. O driver garante que esses índices existam antes de realizar operações de leitura e gravação no bucket GridFS .

Para obter mais informações sobre os índices do GridFS , consulte Índices do GridFS no manual do MongoDB Server .

Ao usar o GridFS para armazenar arquivos, o driver divide os arquivos em partes menores, cada um representado por um documento separado na coleção chunks. Ele também cria um documento na coleção files que contém um ID de arquivo, nome de arquivo e outros metadados de arquivo. Você pode carregar o arquivo passando um fluxo para o driver C++ para consumir ou criando um novo fluxo e gravando diretamente nele.

O diagrama a seguir mostra como o GridFS divide os arquivos quando eles são carregados em um bucket:

Um diagrama que mostra como o GridFS carrega um arquivo em um bloco

Ao recuperar arquivos, o GridFS obtém os metadados da coleção files no bloco especificado e utiliza as informações para reconstruir o arquivo a partir de documentos na coleção chunks. Você pode ler o arquivo gravando seu conteúdo em um fluxo existente ou criando um novo fluxo que aponte para o arquivo.

Para começar a armazenar ou recuperar arquivos do GridFS, chame o método gridfs_bucket() no seu banco de dados de dados. Este método acessa um bucket existente ou cria um novo bucket se não existir.

O exemplo seguinte chama o método gridfs_bucket() no banco de dados de dados do db :

auto bucket = db.gridfs_bucket();

Você pode personalizar a configuração do bucket GridFS passando uma instância da classe mongocxx::options::gridfs::bucket como um argumento opcional para o método gridfs_bucket(). A tabela seguinte descreve os campos que você pode definir em uma instância do mongocxx::options::gridfs::bucket :

Campo
Descrição

bucket_name

Specifies the bucket name to use as a prefix for the files and chunks collections. The default value is "fs".
Type: std::string

chunk_size_bytes

Specifies the chunk size that GridFS splits files into. The default value is 261120.
Type: std::int32_t

read_concern

Specifies the read concern to use for bucket operations. The default value is the database's read concern.
Type: mongocxx::read_concern

read_preference

Specifies the read preference to use for bucket operations. The default value is the database's read preference.
Type: mongocxx::read_preference

write_concern

Specifies the write concern to use for bucket operations. The default value is the database's write concern.
Type: mongocxx::write_concern

O exemplo seguinte cria um bucket denominado "myCustomBucket" configurando o campo bucket_name de uma instância mongocxx::options::gridfs::bucket:

mongocxx::options::gridfs::bucket opts;
opts.bucket_name("myCustomBucket");
auto bucket = db.gridfs_bucket(opts);

Você pode fazer upload de arquivos para um bucket GridFS usando os seguintes métodos:

Utilize o método open_upload_stream() para criar um fluxo de upload para um determinado nome de arquivo. O método open_upload_stream() permite a você especificar informações de configuração em uma instância do options::gridfs::upload, que você pode passar como um parâmetro.

Este exemplo usa um fluxo de upload para executar as seguintes ações:

  • Define o campo chunk_size_bytes de uma instância de opções

  • Abre um fluxo gravável para um novo arquivo GridFS chamado "my_file" e aplica a opção chunk_size_bytes

  • Chama o método write() para gravar dados em my_file, para o qual o stream aponta

  • Chama o método close() para fechar o fluxo apontando para my_file

mongocxx::options::gridfs::upload opts;
opts.chunk_size_bytes(1048576);
auto uploader = bucket.open_upload_stream("my_file", opts);
// ASCII for "HelloWorld"
std::uint8_t bytes[10] = {72, 101, 108, 108, 111, 87, 111, 114, 108, 100};
for (auto i = 0; i < 5; ++i) {
uploader.write(bytes, 10);
}
uploader.close();

Utilize o método upload_from_stream() para carregar o conteúdo de um stream para um novo arquivo GridFS . O método upload_from_stream() permite a você especificar informações de configuração em uma instância do options::gridfs::upload, que você pode passar como um parâmetro.

Este exemplo executa as seguintes ações:

  • Abre um arquivo localizado em /path/to/input_file como um stream no modo de leitura binária

  • Chama o método upload_from_stream() para carregar o conteúdo do stream para um arquivo GridFS chamado "new_file"

std::ifstream file("/path/to/input_file", std::ios::binary);
bucket.upload_from_stream("new_file", &file);

Nesta seção, você pode aprender como recuperar metadados de arquivo armazenados na coleção files do contêiner GridFS. Os metadados contêm informações sobre o arquivo a que se refere, incluindo:

  • O _id do arquivo

  • O nome do arquivo

  • O tamanho/comprimento do arquivo

  • A data e a hora do carregamento

  • Um documento metadata no qual você pode armazenar outras informações

Para recuperar arquivos de um GridFS , chame o método mongocxx::gridfs::bucket::find() no seu contêiner. O método retorna uma instância do mongocxx::cursor da qual você pode acessar os resultados. Para saber mais sobre cursores, consulte o guiaAcessar dados de um cursor.

O seguinte exemplo de código mostra como recuperar e imprimir metadados de arquivo de arquivos em um bucket GridFS . Ele usa um for loop para iterar pelo cursor retornado e exibir o conteúdo dos arquivos carregados nos exemplos de Arquivos de upload:

auto cursor = bucket.find({});
for (auto&& doc : cursor) {
std::cout << bsoncxx::to_json(doc) << std::endl;
}
{ "_id" : { "$oid" : "..." }, "length" : 13, "chunkSize" : 261120, "uploadDate" :
{ "$date" : ... }, "filename" : "new_file" }
{ "_id" : { "$oid" : "..." }, "length" : 50, "chunkSize" : 1048576, "uploadDate" :
{ "$date" : ... }, "filename" : "my_file" }

O método find() aceita várias especificações de query. Você pode usar seu parâmetro mongocxx::options::find para especificar a ordem de classificação, o número máximo de documentos a serem retornados e o número de documentos a serem ignorados antes de retornar. Para visualizar uma lista de opções disponíveis, consulte a documentação da API.

Você pode baixar arquivos de um bucket GridFS usando os seguintes métodos:

Você pode baixar arquivos do seu banco de banco de dados MongoDB utilizando o método open_download_stream() para criar um stream de download.

Este exemplo usa um fluxo de download para executar as seguintes ações:

  • Recupera o valor _id do arquivo GridFS denominado "new_file"

  • Passa o valor _id para o método open_download_stream() para abrir o arquivo como um fluxo legível

  • Cria um vetor buffer para armazenar o conteúdo do arquivo

  • Chama o método read() para ler o conteúdo do arquivo do fluxo downloader para o vetor

auto doc = db["fs.files"].find_one(make_document(kvp("filename", "new_file")));
auto id = doc->view()["_id"].get_value();
auto downloader = bucket.open_download_stream(id);
std::vector<uint8_t> buffer(downloader.file_length());
downloader.read(buffer.data(), buffer.size());

Você pode baixar o conteúdo de um arquivo GridFS para um fluxo existente ligando para o método download_to_stream() em seu bloco.

Este exemplo executa as seguintes ações:

  • Abre um arquivo localizado em /path/to/output_file como um stream no modo de escrita binária

  • Recupera o valor _id do arquivo GridFS denominado "new_file"

  • Passa o valor de _id para download_to_stream() baixar o arquivo para o stream

std::ofstream output_file("/path/to/output_file", std::ios::binary);
auto doc = db["fs.files"].find_one(make_document(kvp("filename", "new_file")));
auto id = doc->view()["_id"].get_value();
bucket.download_to_stream(id, &output_file);

Use o método delete_file() para remover o documento de collection de um arquivo e os chunks associados do seu bucket. Isso exclui efetivamente o arquivo. Você deve especificar o arquivo pelo campo _id em vez do nome do arquivo.

O exemplo a seguir mostra como excluir um arquivo chamado "my_file" passando seu valor _id para delete_file():

auto doc = db["fs.files"].find_one(make_document(kvp("filename", "my_file")));
auto id = doc->view()["_id"].get_value();
bucket.delete_file(id);

Observação

Revisões de arquivos

O método delete_file() suporta a exclusão de somente um arquivo de cada vez. Se você quiser excluir cada revisão de arquivo ou arquivos com tempos de carregamento diferentes que compartilham o mesmo nome de arquivo, colete os valores _id de cada revisão. Em seguida, passe cada valor _id em chamadas separadas para o método delete_file().

Para saber mais sobre como usar o driver C++ para armazenar e recuperar arquivos grandes, consulte a seguinte documentação da API:

Voltar

Gravação em massa