GITHUB#12342 Add new maximum inner product vector similarity method (#12479)

The current dot-product score scaling and similarity implementation assumes normalized vectors. This disregards information that the model may store within the magnitude. 

See: https://github.com/apache/lucene/issues/12342#issuecomment-1658640222 for a good explanation for the need.

To prevent from breaking current scoring assumptions in Lucene, a new `MAXIMUM_INNER_PRODUCT` similarity function is added. 

Because the similarity from a `dotProduct` function call could be negative, this similarity scorer will scale negative dotProducts to between 0-1 and then all positive dotProduct values are from 1-MAX.

One concern with adding this similarity function is that it breaks the triangle inequality. It is assumed that this is needed to build graph structures. But, there is conflicting research here when it comes to real-world data.

See:
 - For: https://github.com/apache/lucene/issues/12342#issuecomment-1618258984
 - Against: https://github.com/apache/lucene/issues/12342#issuecomment-1631577657, https://github.com/apache/lucene/issues/12342#issuecomment-1631808301

To check if any transformation of the input is required to satisfy the triangle inequality, many tests have been ran

See:

 - https://github.com/apache/lucene/issues/12342#issuecomment-1653420640
 - https://github.com/apache/lucene/issues/12342#issuecomment-1656112434
 - https://github.com/apache/lucene/issues/12342#issuecomment-1656718447

If there are any additional tests, or issues with the provided tests & scripts, please let me know. We want to make sure this works well for our users.

closes: https://github.com/apache/lucene/issues/12342
This commit is contained in:
Benjamin Trent 2023-08-16 12:15:25 -04:00 committed by GitHub
parent 71f6f59a75
commit 5a5aa2c8fa
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
6 changed files with 71 additions and 2 deletions

View File

@ -148,6 +148,9 @@ New Features
search results can be provided. The first custom collector provides `ToParentBlockJoin[Float|Byte]KnnVectorQuery` search results can be provided. The first custom collector provides `ToParentBlockJoin[Float|Byte]KnnVectorQuery`
joining child vector documents with their parent documents. (Ben Trent) joining child vector documents with their parent documents. (Ben Trent)
* GITHUB#12479: Add new Maximum Inner Product vector similarity function for non-normalized dot-product
vector search. (Jack Mazanec, Ben Trent)
Improvements Improvements
--------------------- ---------------------
* GITHUB#12374: Add CachingLeafSlicesSupplier to compute the LeafSlices for concurrent segment search (Sorabh Hamirwasia) * GITHUB#12374: Add CachingLeafSlicesSupplier to compute the LeafSlices for concurrent segment search (Sorabh Hamirwasia)

View File

@ -19,6 +19,7 @@ package org.apache.lucene.index;
import static org.apache.lucene.util.VectorUtil.cosine; import static org.apache.lucene.util.VectorUtil.cosine;
import static org.apache.lucene.util.VectorUtil.dotProduct; import static org.apache.lucene.util.VectorUtil.dotProduct;
import static org.apache.lucene.util.VectorUtil.dotProductScore; import static org.apache.lucene.util.VectorUtil.dotProductScore;
import static org.apache.lucene.util.VectorUtil.scaleMaxInnerProductScore;
import static org.apache.lucene.util.VectorUtil.squareDistance; import static org.apache.lucene.util.VectorUtil.squareDistance;
/** /**
@ -76,6 +77,23 @@ public enum VectorSimilarityFunction {
public float compare(byte[] v1, byte[] v2) { public float compare(byte[] v1, byte[] v2) {
return (1 + cosine(v1, v2)) / 2; return (1 + cosine(v1, v2)) / 2;
} }
},
/**
* Maximum inner product. This is like {@link VectorSimilarityFunction#DOT_PRODUCT}, but does not
* require normalization of the inputs. Should be used when the embedding vectors store useful
* information within the vector magnitude
*/
MAXIMUM_INNER_PRODUCT {
@Override
public float compare(float[] v1, float[] v2) {
return scaleMaxInnerProductScore(dotProduct(v1, v2));
}
@Override
public float compare(byte[] v1, byte[] v2) {
return scaleMaxInnerProductScore(dotProduct(v1, v2));
}
}; };
/** /**

View File

@ -164,6 +164,17 @@ public final class VectorUtil {
return 0.5f + dotProduct(a, b) / denom; return 0.5f + dotProduct(a, b) / denom;
} }
/**
* @param vectorDotProductSimilarity the raw similarity between two vectors
* @return A scaled score preventing negative scores for maximum-inner-product
*/
public static float scaleMaxInnerProductScore(float vectorDotProductSimilarity) {
if (vectorDotProductSimilarity < 0) {
return 1 / (1 + -1 * vectorDotProductSimilarity);
}
return vectorDotProductSimilarity + 1;
}
/** /**
* Checks if a float vector only has finite components. * Checks if a float vector only has finite components.
* *

View File

@ -346,6 +346,29 @@ abstract class BaseKnnVectorQueryTestCase extends LuceneTestCase {
} }
} }
public void testScoreMIP() throws IOException {
try (Directory indexStore =
getIndexStore(
"field",
VectorSimilarityFunction.MAXIMUM_INNER_PRODUCT,
new float[] {0, 1},
new float[] {1, 2},
new float[] {0, 0});
IndexReader reader = DirectoryReader.open(indexStore)) {
IndexSearcher searcher = newSearcher(reader);
AbstractKnnVectorQuery kvq = getKnnVectorQuery("field", new float[] {0, -1}, 10);
assertMatches(searcher, kvq, 3);
ScoreDoc[] scoreDocs = searcher.search(kvq, 3).scoreDocs;
assertIdMatches(reader, "id2", scoreDocs[0]);
assertIdMatches(reader, "id0", scoreDocs[1]);
assertIdMatches(reader, "id1", scoreDocs[2]);
assertEquals(1.0, scoreDocs[0].score, 1e-7);
assertEquals(1 / 2f, scoreDocs[1].score, 1e-7);
assertEquals(1 / 3f, scoreDocs[2].score, 1e-7);
}
}
public void testExplain() throws IOException { public void testExplain() throws IOException {
try (Directory d = newDirectory()) { try (Directory d = newDirectory()) {
try (IndexWriter w = new IndexWriter(d, new IndexWriterConfig())) { try (IndexWriter w = new IndexWriter(d, new IndexWriterConfig())) {
@ -739,11 +762,21 @@ abstract class BaseKnnVectorQueryTestCase extends LuceneTestCase {
/** Creates a new directory and adds documents with the given vectors as kNN vector fields */ /** Creates a new directory and adds documents with the given vectors as kNN vector fields */
Directory getIndexStore(String field, float[]... contents) throws IOException { Directory getIndexStore(String field, float[]... contents) throws IOException {
return getIndexStore(field, VectorSimilarityFunction.EUCLIDEAN, contents);
}
/**
* Creates a new directory and adds documents with the given vectors with similarity as kNN vector
* fields
*/
Directory getIndexStore(
String field, VectorSimilarityFunction vectorSimilarityFunction, float[]... contents)
throws IOException {
Directory indexStore = newDirectory(); Directory indexStore = newDirectory();
RandomIndexWriter writer = new RandomIndexWriter(random(), indexStore); RandomIndexWriter writer = new RandomIndexWriter(random(), indexStore);
for (int i = 0; i < contents.length; ++i) { for (int i = 0; i < contents.length; ++i) {
Document doc = new Document(); Document doc = new Document();
doc.add(getKnnVectorField(field, contents[i])); doc.add(getKnnVectorField(field, contents[i], vectorSimilarityFunction));
doc.add(new StringField("id", "id" + i, Field.Store.YES)); doc.add(new StringField("id", "id" + i, Field.Store.YES));
writer.addDocument(doc); writer.addDocument(doc);
} }

View File

@ -1246,6 +1246,9 @@ public final class DocumentsPanelProvider implements DocumentsTabOperator {
case EUCLIDEAN: case EUCLIDEAN:
sb.append("euc"); sb.append("euc");
break; break;
case MAXIMUM_INNER_PRODUCT:
sb.append("mip");
break;
default: default:
sb.append("???"); sb.append("???");
} }

View File

@ -1278,7 +1278,8 @@ public abstract class BaseKnnVectorsFormatTestCase extends BaseIndexFileFormatTe
assertEquals(0, VectorSimilarityFunction.EUCLIDEAN.ordinal()); assertEquals(0, VectorSimilarityFunction.EUCLIDEAN.ordinal());
assertEquals(1, VectorSimilarityFunction.DOT_PRODUCT.ordinal()); assertEquals(1, VectorSimilarityFunction.DOT_PRODUCT.ordinal());
assertEquals(2, VectorSimilarityFunction.COSINE.ordinal()); assertEquals(2, VectorSimilarityFunction.COSINE.ordinal());
assertEquals(3, VectorSimilarityFunction.values().length); assertEquals(3, VectorSimilarityFunction.MAXIMUM_INNER_PRODUCT.ordinal());
assertEquals(4, VectorSimilarityFunction.values().length);
} }
public void testVectorEncodingOrdinals() { public void testVectorEncodingOrdinals() {