-
Notifications
You must be signed in to change notification settings - Fork 48
/
Copy pathretrieve_nq.py
89 lines (65 loc) · 2.85 KB
/
retrieve_nq.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
"""
# Example: Retrieve from pre-built index of Natural Questions
This shows how to load an index built with BM25.index and saved with BM25.save, and retrieve
the top-k results for a set of queries from the Natural Questions dataset, via BEIR library.
To run this example, you need to install the following dependencies:
```bash
pip install bm25s[core]
```
To build an index, please refer to the `examples/index_nq.py` script. You
can run this script with:
```bash
python examples/index_nq.py
```
Then, run this script with:
```bash
python examples/retrieve_nq.py
```
"""
from pathlib import Path
import numpy as np
import bm25s
import Stemmer
from tqdm import tqdm
def main(index_dir="bm25s_indices", data_dir="datasets", dataset="nq", split="test", mmap=True):
index_dir = Path(index_dir) / dataset
if mmap:
print("Using memory-mapped index (mmap) to reduce memory usage.")
timer = bm25s.utils.benchmark.Timer("[BM25S]")
queries = bm25s.utils.beir.load_queries(dataset, save_dir=data_dir)
qrels = bm25s.utils.beir.load_qrels(dataset, split=split, save_dir=data_dir)
queries_lst = [v["text"] for k, v in queries.items() if k in qrels]
print(f"Loaded {len(queries_lst)} queries.")
stemmer = Stemmer.Stemmer("english")
# Tokenize the queries
queries_tokenized = bm25s.tokenize(queries_lst, stemmer=stemmer, return_ids=False)
# # Alternatively, you can use the following code to tokenize the queries
# # using the saved tokenizer from the index directory
# tokenizer = bm25s.tokenization.Tokenizer(stemmer=stemmer)
# tokenizer.load_stopwords(index_dir)
# tokenizer.load_vocab(index_dir)
# queries_tokenized = tokenizer.tokenize(queries_lst, update_vocab=False)
mem_use = bm25s.utils.benchmark.get_max_memory_usage()
print(f"Initial memory usage: {mem_use:.2f} GB")
# Load the BM25 index and retrieve the top-k results
print(f"Loading the BM25 index for: {dataset}")
t = timer.start("Loading index")
retriever = bm25s.BM25.load(index_dir, mmap=mmap, load_corpus=True)
retriever.backend = "numba"
num_docs = retriever.scores['num_docs']
timer.stop(t, show=True, n_total=num_docs)
mem_use = bm25s.utils.benchmark.get_max_memory_usage()
print(f"Memory usage after loading the index: {mem_use:.2f} GB")
print("Retrieving the top-k results...")
t = timer.start("Retrieving")
results = retriever.retrieve(queries_tokenized, k=10)
timer.stop(t, show=True, n_total=len(queries_lst))
# get memory usage
mem_use = bm25s.utils.benchmark.get_max_memory_usage()
print(f"Final (peak) memory usage: {mem_use:.2f} GB")
print("-" * 50)
first_result = results.documents[0]
print(f"First score (# 1 result): {results.scores[0, 0]:.4f}")
print(f"First result (# 1 result):\n{first_result[0]}")
if __name__ == "__main__":
main(mmap=True)