A página mostra como usar o Cloud Firestore para fazer pesquisas de vetor de vizinho mais próximo (KNN, na sigla em inglês) usando estas técnicas:
- Armazenar valores vetoriais
- Criar e gerenciar índices de vetor KNN
- Fazer uma consulta de vizinho mais próximo (KNN) usando uma das medidas de distância de vetor compatíveis
Armazenar embeddings de vetores
Você pode criar valores vetoriais, como embeddings de texto, a partir dos seus dados do Cloud Firestore e armazená-los em documentos do Cloud Firestore.
Operação de gravação com um embedding de vetor
O exemplo a seguir mostra como armazenar um embedding de vetor em um documento do Cloud Firestore:
Python
Node.js
import { Firestore, FieldValue, } from "@google-cloud/firestore"; const db = new Firestore(); const coll = db.collection('coffee-beans'); await coll.add({ name: "Kahawa coffee beans", description: "Information about the Kahawa coffee beans.", embedding_field: FieldValue.vector([1.0 , 2.0, 3.0]) });
Go
Java
import com.google.cloud.firestore.CollectionReference; import com.google.cloud.firestore.DocumentReference; import com.google.cloud.firestore.FieldValue; import com.google.cloud.firestore.VectorQuery; CollectionReference coll = firestore.collection("coffee-beans"); Map<String, Object> docData = new HashMap<>(); docData.put("name", "Kahawa coffee beans"); docData.put("description", "Information about the Kahawa coffee beans."); docData.put("embedding_field", FieldValue.vector(new double[] {1.0, 2.0, 3.0})); ApiFuture<DocumentReference> future = coll.add(docData); DocumentReference documentReference = future.get();
Calcular embeddings de vetores com uma função do Cloud
Para calcular e armazenar embeddings de vetor sempre que um documento for atualizado ou criado, configure uma função do Cloud:
Python
@functions_framework.cloud_event def store_embedding(cloud_event) -> None: """Triggers by a change to a Firestore document. """ firestore_payload = firestore.DocumentEventData() payload = firestore_payload._pb.ParseFromString(cloud_event.data) collection_id, doc_id = from_payload(payload) # Call a function to calculate the embedding embedding = calculate_embedding(payload) # Update the document doc = firestore_client.collection(collection_id).document(doc_id) doc.set({"embedding_field": embedding}, merge=True)
Node.js
/** * A vector embedding will be computed from the * value of the `content` field. The vector value * will be stored in the `embedding` field. The * field names `content` and `embedding` are arbitrary * field names chosen for this example. */ async function storeEmbedding(event: FirestoreEvent<any>): Promise<void> { // Get the previous value of the document's `content` field. const previousDocumentSnapshot = event.data.before as QueryDocumentSnapshot; const previousContent = previousDocumentSnapshot.get("content"); // Get the current value of the document's `content` field. const currentDocumentSnapshot = event.data.after as QueryDocumentSnapshot; const currentContent = currentDocumentSnapshot.get("content"); // Don't update the embedding if the content field did not change if (previousContent === currentContent) { return; } // Call a function to calculate the embedding for the value // of the `content` field. const embeddingVector = calculateEmbedding(currentContent); // Update the `embedding` field on the document. await currentDocumentSnapshot.ref.update({ embedding: embeddingVector, }); }
Go
// Not yet supported in the Go client library
Java
// Not yet supported in the Java client library
Criar e gerenciar índices vetoriais
Antes de realizar a consulta de vizinho mais próximo com seus embeddings vetoriais, crie um índice correspondente. Os exemplos a seguir demonstram como criar e gerenciar índices vetoriais com o Google Cloud CLI. Os índices de vetor também podem ser gerenciados com a CLI do Firebase e o Terraform.
Criar um índice vetorial
Antes de criar um índice de vetor, faça upgrade para a versão mais recente do Google Cloud CLI:
gcloud components update
Para criar um índice de vetores, use gcloud firestore indexes composite create
:
gcloud
gcloud firestore indexes composite create \ --collection-group=collection-group \ --query-scope=COLLECTION \ --field-config field-path=vector-field,vector-config='vector-configuration' \ --database=database-id
em que:
- collection-group é o ID do grupo de coleções.
- vector-field é o nome do campo que contém o embedding de vetor.
- database-id é o ID do banco de dados.
- vector-configuration inclui o vetor
dimension
e o tipo de índice. Odimension
é um número inteiro até 2.048. O tipo de índice precisa serflat
. Formate a configuração do índice da seguinte maneira:{"dimension":"DIMENSION", "flat": "{}"}
.
O exemplo a seguir cria um índice composto, incluindo um índice de vetores para o campo vector-field
e um índice crescente para o campo color
. Você pode usar esse tipo de índice para pré-filtrar
dados antes de uma pesquisa de vizinho mais próximo.
gcloud
gcloud firestore indexes composite create \ --collection-group=collection-group \ --query-scope=COLLECTION \ --field-config=order=ASCENDING,field-path="color" \ --field-config field-path=vector-field,vector-config='{"dimension":"1024", "flat": "{}"}' \ --database=database-id
Listar todos os índices vetoriais
gcloud
gcloud firestore indexes composite list --database=database-id
Substitua database-id pelo ID do banco de dados.
Excluir um índice vetorial
gcloud
gcloud firestore indexes composite delete index-id --database=database-id
em que:
- index-id é o ID do índice a ser excluído.
Use
indexes composite list
para recuperar o ID do índice. - database-id é o ID do banco de dados.
Descrever um índice vetorial
gcloud
gcloud firestore indexes composite describe index-id --database=database-id
em que:
- index-id é o ID do índice a ser descrito. Use ou
indexes composite list
para recuperar o ID do índice. - database-id é o ID do banco de dados.
Fazer uma consulta de vizinho mais próximo
É possível realizar uma pesquisa por similaridade para encontrar os vizinhos mais próximos de um embedding de vetor. As pesquisas por similaridade exigem índices vetoriais. Se um índice não existir, o Cloud Firestore vai sugerir a criação de um usando o gcloud CLI.
O exemplo a seguir encontra dez vizinhos mais próximos do vetor de consulta.
Python
Node.js
import { Firestore, FieldValue, VectorQuery, VectorQuerySnapshot, } from "@google-cloud/firestore"; // Requires a single-field vector index const vectorQuery: VectorQuery = coll.findNearest({ vectorField: 'embedding_field', queryVector: [3.0, 1.0, 2.0], limit: 10, distanceMeasure: 'EUCLIDEAN' }); const vectorQuerySnapshot: VectorQuerySnapshot = await vectorQuery.get();
Go
Java
import com.google.cloud.firestore.VectorQuery; import com.google.cloud.firestore.VectorQuerySnapshot; VectorQuery vectorQuery = coll.findNearest( "embedding_field", new double[] {3.0, 1.0, 2.0}, /* limit */ 10, VectorQuery.DistanceMeasure.EUCLIDEAN); ApiFuture<VectorQuerySnapshot> future = vectorQuery.get(); VectorQuerySnapshot vectorQuerySnapshot = future.get();
Distâncias vetoriais
As consultas de vizinhos mais próximos são compatíveis com as seguintes opções de distância vetorial:
EUCLIDEAN
: mede a distância EUCLIDIANA entre os vetores. Para saber mais, consulte Euclidiano.COSINE
: compara vetores com base no ângulo entre eles, o que permite medir a similaridade que não se baseia na magnitude dos vetores. Recomendamos usarDOT_PRODUCT
com vetores normalizados unitários em vez da distância COSSENO, que é matematicamente equivalente a um melhor desempenho. Para saber mais, consulte Semelhança de cossenos.DOT_PRODUCT
: semelhante aCOSINE
, mas é afetado pela magnitude de vetores. Para saber mais, consulte Produto escalar.
Escolha a medida de distância
Dependendo se todos os embeddings de vetor estiverem normalizadas ou não, você pode determinar qual medida de distância usar para encontrar a medida de distância. Um embedding de vetor normalizado tem uma magnitude (comprimento) de exatamente 1,0.
Além disso, se você souber com qual medida de distância o modelo foi treinado, use essa medida para calcular a distância entre os embeddings de vetor.
Dados normalizados
Se você tiver um conjunto de dados em que todos os embeddings de vetor estiverem normalizadas, as três
medidas de distância vão fornecer os mesmos resultados de pesquisa semântica. Resumindo, embora cada
medida de distância retorne um valor diferente, esses valores são classificados da mesma maneira. Quando
os embeddings estão normalizados, o DOT_PRODUCT
geralmente é o mais eficiente
em termos computacionais, mas a diferença é insignificante na maioria dos casos. No entanto, se o
aplicativo for altamente sensível ao desempenho, DOT_PRODUCT
poderá ajudar no
ajuste de desempenho.
Dados não normalizados
Se você tiver um conjunto de dados em que os embeddings de vetor não estiverem normalizados,
não será matematicamente correto usar DOT_PRODUCT
como uma medida de distância,
porque o produto escalar não mede distância. Dependendo
de como os embeddings foram gerados e de qual tipo de pesquisa é a preferida,
a medida de distância COSINE
ou EUCLIDEAN
produzirá
resultados de pesquisa subjetivamente melhores do que as outras medidas de distância.
A experimentação com COSINE
ou EUCLIDEAN
pode
ser necessária para determinar qual é a melhor para seu caso de uso.
Não tem certeza se os dados estão normalizados ou não
Se você não tiver certeza se os dados estão normalizados e quiser usar
DOT_PRODUCT
, recomendamos usar COSINE
.
COSINE
é como DOT_PRODUCT
com a normalização integrada.
A distância medida usando COSINE
varia de 0
a 2
. Um resultado
próximo a 0
indica que os vetores são muito semelhantes.
Pré-filtrar documentos
Para pré-filtrar documentos antes de encontrar os vizinhos mais próximos, é possível combinar uma
pesquisa de similaridade com outros operadores de consulta. Os filtros compostos and
e
or
são compatíveis. Para mais informações sobre os filtros de campo aceitos, consulte Operadores de consulta.
Python
Node.js
// Similarity search with pre-filter // Requires composite vector index const preFilteredVectorQuery: VectorQuery = coll .where("color", "==", "red") .findNearest({ vectorField: "embedding_field", queryVector: [3.0, 1.0, 2.0], limit: 5, distanceMeasure: "EUCLIDEAN", }); const vectorQueryResults = await preFilteredVectorQuery.get();
Go
Java
import com.google.cloud.firestore.VectorQuery; import com.google.cloud.firestore.VectorQuerySnapshot; VectorQuery preFilteredVectorQuery = coll .whereEqualTo("color", "red") .findNearest( "embedding_field", new double[] {3.0, 1.0, 2.0}, /* limit */ 10, VectorQuery.DistanceMeasure.EUCLIDEAN); ApiFuture<VectorQuerySnapshot> future = preFilteredVectorQuery.get(); VectorQuerySnapshot vectorQuerySnapshot = future.get();
Extrair a distância vetorial calculada
É possível extrair a distância vetorial calculada atribuindo um
nome de propriedade de saída distance_result_field
na consulta FindNearest
, conforme
mostrado no exemplo a seguir:
Python
Node.js
const vectorQuery: VectorQuery = coll.findNearest( { vectorField: 'embedding_field', queryVector: [3.0, 1.0, 2.0], limit: 10, distanceMeasure: 'EUCLIDEAN', distanceResultField: 'vector_distance' }); const snapshot: VectorQuerySnapshot = await vectorQuery.get(); snapshot.forEach((doc) => { console.log(doc.id, ' Distance: ', doc.get('vector_distance')); });
Go
Java
import com.google.cloud.firestore.VectorQuery; import com.google.cloud.firestore.VectorQueryOptions; import com.google.cloud.firestore.VectorQuerySnapshot; VectorQuery vectorQuery = coll.findNearest( "embedding_field", new double[] {3.0, 1.0, 2.0}, /* limit */ 10, VectorQuery.DistanceMeasure.EUCLIDEAN, VectorQueryOptions.newBuilder().setDistanceResultField("vector_distance").build()); ApiFuture<VectorQuerySnapshot> future = vectorQuery.get(); VectorQuerySnapshot vectorQuerySnapshot = future.get(); for (DocumentSnapshot document : vectorQuerySnapshot.getDocuments()) { System.out.println(document.getId() + " Distance: " + document.get("vector_distance")); }
Se você quiser usar uma máscara de campo para retornar um subconjunto de campos de documento com um distanceResultField
, também precisará incluir o valor de distanceResultField
na máscara de campo, conforme mostrado no exemplo a seguir:
Python
Node.js
const vectorQuery: VectorQuery = coll .select('name', 'description', 'vector_distance') .findNearest({ vectorField: 'embedding_field', queryVector: [3.0, 1.0, 2.0], limit: 10, distanceMeasure: 'EUCLIDEAN', distanceResultField: 'vector_distance' });
Go
Java
import com.google.cloud.firestore.VectorQuery; import com.google.cloud.firestore.VectorQueryOptions; import com.google.cloud.firestore.VectorQuerySnapshot; VectorQuery vectorQuery = coll .select("name", "description", "vector_distance") .findNearest( "embedding_field", new double[] {3.0, 1.0, 2.0}, /* limit */ 10, VectorQuery.DistanceMeasure.EUCLIDEAN, VectorQueryOptions.newBuilder() .setDistanceResultField("vector_distance") .build()); ApiFuture<VectorQuerySnapshot> future = vectorQuery.get(); VectorQuerySnapshot vectorQuerySnapshot = future.get(); for (DocumentSnapshot document : vectorQuerySnapshot.getDocuments()) { System.out.println(document.getId() + " Distance: " + document.get("vector_distance")); }
Especificar um limite de distância
É possível especificar um limite de similaridade que retorna apenas documentos dentro do limite. O comportamento do campo de limite depende da medida de distância escolhida:
- As distâncias
EUCLIDEAN
eCOSINE
limitam o limite para documentos em que a distância é menor ou igual ao limite especificado. Essas medidas de distância diminuem à medida que os vetores se tornam mais semelhantes. - A distância
DOT_PRODUCT
limita o limite a documentos em que a distância é maior ou igual ao limite especificado. As distâncias do produto escalar aumentam à medida que os vetores se tornam mais semelhantes.
O exemplo a seguir mostra como especificar um limite de distância para retornar até dez documentos mais próximos que estejam a, no máximo, 4,5 unidades usando a métrica de distância EUCLIDEAN
:
Python
Node.js
const vectorQuery: VectorQuery = coll.findNearest({ vectorField: 'embedding_field', queryVector: [3.0, 1.0, 2.0], limit: 10, distanceMeasure: 'EUCLIDEAN', distanceThreshold: 4.5 }); const snapshot: VectorQuerySnapshot = await vectorQuery.get(); snapshot.forEach((doc) => { console.log(doc.id); });
Go
Java
import com.google.cloud.firestore.VectorQuery; import com.google.cloud.firestore.VectorQueryOptions; import com.google.cloud.firestore.VectorQuerySnapshot; VectorQuery vectorQuery = coll.findNearest( "embedding_field", new double[] {3.0, 1.0, 2.0}, /* limit */ 10, VectorQuery.DistanceMeasure.EUCLIDEAN, VectorQueryOptions.newBuilder() .setDistanceThreshold(4.5) .build()); ApiFuture<VectorQuerySnapshot> future = vectorQuery.get(); VectorQuerySnapshot vectorQuerySnapshot = future.get(); for (DocumentSnapshot document : vectorQuerySnapshot.getDocuments()) { System.out.println(document.getId()); }
Limitações
Ao trabalhar com embeddings de vetor, observe as seguintes limitações:
- A dimensão de incorporação máxima compatível é 2048. Para armazenar índices maiores, use redução de dimensionalidade.
- O número máximo de documentos a serem retornados de uma consulta de vizinho mais próximo é 1.000.
- A consulta de vetor não é compatível com listeners de snapshots em tempo real.
- Somente as bibliotecas de cliente Python, Node.js, Go e Java são compatíveis com a pesquisa de vetor.
A seguir
- Leia sobre as práticas recomendadas para Cloud Firestore.
- Compreensão de leituras e gravações em escala.