95 lines
2.2 KiB
Python
95 lines
2.2 KiB
Python
|
# Copyright (c) 2015-present, Facebook, Inc.
|
||
|
# All rights reserved.
|
||
|
#
|
||
|
# This source code is licensed under the BSD+Patents license found in the
|
||
|
# LICENSE file in the root directory of this source tree.
|
||
|
|
||
|
#! /usr/bin/env python2
|
||
|
|
||
|
"""Binary indexes (de)serialization"""
|
||
|
|
||
|
import numpy as np
|
||
|
import unittest
|
||
|
import faiss
|
||
|
import os
|
||
|
|
||
|
def make_binary_dataset(d, nb, nt, nq):
|
||
|
assert d % 8 == 0
|
||
|
x = np.random.randint(256, size=(nb + nq + nt, int(d / 8))).astype('uint8')
|
||
|
return x[:nt], x[nt:-nq], x[-nq:]
|
||
|
|
||
|
|
||
|
class TestBinaryFlat(unittest.TestCase):
|
||
|
|
||
|
def __init__(self, *args, **kwargs):
|
||
|
unittest.TestCase.__init__(self, *args, **kwargs)
|
||
|
d = 32
|
||
|
nt = 0
|
||
|
nb = 1500
|
||
|
nq = 500
|
||
|
|
||
|
(_, self.xb, self.xq) = make_binary_dataset(d, nb, nt, nq)
|
||
|
|
||
|
def test_flat(self):
|
||
|
d = self.xq.shape[1] * 8
|
||
|
|
||
|
index = faiss.IndexBinaryFlat(d)
|
||
|
index.add(self.xb)
|
||
|
D, I = index.search(self.xq, 3)
|
||
|
|
||
|
tmpnam = os.tempnam()
|
||
|
try:
|
||
|
faiss.write_index_binary(index, tmpnam)
|
||
|
|
||
|
index2 = faiss.read_index_binary(tmpnam)
|
||
|
|
||
|
D2, I2 = index2.search(self.xq, 3)
|
||
|
|
||
|
assert (I2 == I).all()
|
||
|
assert (D2 == D).all()
|
||
|
|
||
|
finally:
|
||
|
os.remove(tmpnam)
|
||
|
|
||
|
|
||
|
class TestBinaryIVF(unittest.TestCase):
|
||
|
|
||
|
def __init__(self, *args, **kwargs):
|
||
|
unittest.TestCase.__init__(self, *args, **kwargs)
|
||
|
d = 32
|
||
|
nt = 200
|
||
|
nb = 1500
|
||
|
nq = 500
|
||
|
|
||
|
(self.xt, self.xb, self.xq) = make_binary_dataset(d, nb, nt, nq)
|
||
|
|
||
|
def test_ivf_flat(self):
|
||
|
d = self.xq.shape[1] * 8
|
||
|
|
||
|
quantizer = faiss.IndexBinaryFlat(d)
|
||
|
index = faiss.IndexBinaryIVF(quantizer, d, 8)
|
||
|
index.cp.min_points_per_centroid = 5 # quiet warning
|
||
|
index.nprobe = 4
|
||
|
index.train(self.xt)
|
||
|
index.add(self.xb)
|
||
|
D, I = index.search(self.xq, 3)
|
||
|
|
||
|
tmpnam = os.tempnam()
|
||
|
|
||
|
try:
|
||
|
faiss.write_index_binary(index, tmpnam)
|
||
|
|
||
|
index2 = faiss.read_index_binary(tmpnam)
|
||
|
|
||
|
D2, I2 = index2.search(self.xq, 3)
|
||
|
|
||
|
assert (I2 == I).all()
|
||
|
assert (D2 == D).all()
|
||
|
|
||
|
finally:
|
||
|
os.remove(tmpnam)
|
||
|
|
||
|
|
||
|
if __name__ == '__main__':
|
||
|
unittest.main()
|