Fast Full Text Search based on BM25
Add fast in-memory semantic search to your application using wink-bm25-text-search
. It is a part of wink — a growing family of high quality packages for Statistical Analysis, Natural Language Processing and Machine Learning in NodeJS.
It is based on state-of-the-art text search algorithm — BM25 — a Probabilistic Relevance Framework for document retrieval. It's API offers a rich set of features:
-
Scalable Design allows easy addition/customization of features like geolocation and more.
-
Search on exact values of pre-defined fields, makes search results more relevant.
-
Index optimized for size and speed can be exported (and imported) from the added documents in a JSON format.
-
Full control over BM25 configuration — while default values work well for most situations, there is an option to control them.
-
Add semantic flavor to the search by:
- Assigning different numerical weights to the fields. A negative field weight will pull down the document's score whenever a match with that field occurs.
- Using
amplifyNegation()
andpropagateNegations()
from wink-nlp-utils will ensure different search results for query texts containing phrases like "good" and "not good". - Defining different text preparation tasks separately for the fields and query text.
-
Complete flexibility in text preparation — perform tasks such as tokenization and stemming using wink-nlp-utils or any other package of your choice.
Use npm to install:
npm install wink-bm25-text-search --save
// Load wink-bm25-text-search
var bm25 = require( 'wink-bm25-text-search' );
// Create search engine's instance
var engine = bm25();
// Load NLP utilities
var nlp = require( 'wink-nlp-utils' );
// Load sample data (load any other JSON data instead of sample)
var docs = require( 'wink-bm25-text-search/sample-data/data-for-wink-bm25.json' );
// Step I: Define config
// Only field weights are required in this example.
engine.defineConfig( { fldWeights: { title: 4, body: 1, tags: 2 } } );
// Step II: Define PrepTasks
// Set up preparatory tasks for 'body' field
engine.definePrepTasks( [
nlp.string.lowerCase,
nlp.string.removeExtraSpaces,
nlp.string.tokenize0,
nlp.tokens.propagateNegations,
nlp.tokens.removeWords,
nlp.tokens.stem
], 'body' );
// Set up 'default' preparatory tasks i.e. for everything else
engine.definePrepTasks( [
nlp.string.lowerCase,
nlp.string.removeExtraSpaces,
nlp.string.tokenize0,
nlp.tokens.propagateNegations,
nlp.tokens.stem
] );
// Step III: Add Docs
// Add documents now...
docs.forEach( function ( doc, i ) {
// Note, 'i' becomes the unique id for 'doc'
engine.addDoc( doc, i );
} );
// Step IV: Consolidate
// Consolidate before searching
engine.consolidate();
// All set, start searching!
var results = engine.search( 'who is married to barack' );
// results is an array of [ doc-id, score ], sorted by score
// results[ 0 ][ 0 ] i.e. the top result is:
console.log( docs[ results[ 0 ][ 0 ] ].body );
// -> Michelle LaVaughn Robinson Obama (born January 17, 1964) is...
Defines the configuration from the config
object. This object defines following 3 properties:
-
The
fldWeights
(mandatory) is an object where each key is the document's field name and the value is the numerical weight i.e. the importance of that field. -
The
bm25Params
(optional) is also an object that defines upto 3 keys viz.k1
,b
, andk
. Their default values are respectively1.2
,0.75
, and1
. Note:k1
controls TF saturation;b
controls degree of normalization, andk
manages IDF. -
The
ovFieldNames
(optional) is an array containing the names of the fields, whose original value must be retained. This should be defined, if search on exact field values is required.
Defines the text preparation tasks
to transform raw incoming text into an array of tokens required during addDoc()
, and search()
operations. It returns the count of tasks
.
The tasks
should be an array of functions. The first function in this array must accept a string as input; and the last function must return an array of tokens as JavaScript Strings. Each function must accept one input argument and return a single value.
The second argument — field
is optional. It defines the field
of the document for which the tasks
will be defined; in absence of this argument, the tasks
become the default for everything else. The configuration must be defined via defineConfig()
prior to this call.
As illustrated in the example above, wink-nlp-utils offers a rich set of such functions.
Adds the doc
with the uniqueId
to the BM25 model. Prior to adding docs, defineConfig()
and definePrepTasks()
must be called. It accepts structured JSON documents as input for creating the model. Following is an example document structure of the sample data JSON contained in this package:
{
title: 'Barack Obama',
body: 'Barack Hussein Obama II born August 4, 1961 is an American politician...'
tags: 'democratic nobel peace prize columbia michelle...'
}
The sample data is created using excerpts from Wikipedia articles such as one on Barack Obama.
It has an alias learn( doc, uniqueId )
to maintain API level uniformity across various wink packages such as wink-naive-bayes-text-classifier.
Consolidates the BM25 model for all the added documents. The fp
defines the precision at
which term frequency values are stored. The default value is 4 and is good enough for most situations. It is a prerequisite for search()
and documents cannot be added post consolidation.
Searches for the text
and returns upto the limit
number of results. The filter
should be a function that must return true or false based on params
. Think of it as Javascript Array's filter function. Both filter
and params
are optional.
The result is an array of
[ uniqueId, relevanceScore ]
, sorted on the relevanceScore
. The default value of limit
is 10.
Like addDoc()
, it also has an alias predict( doc, uniqueId )
to maintain API level uniformity across various wink packages such as wink-naive-bayes-text-classifier.
The BM25 model can be exported as JSON text that may be saved in a file. It is a good idea to export JSON prior to consolidation and use the same whenever more documents need to be added; whereas JSON exported after consolidation is only good for search operation.
An existing JSON BM25 model can be imported for search. It is essential to call definePrepTasks()
before attempting to search.
It completely resets the BM25 model by re-initializing all the variables, except the preparatory tasks.
If you spot a bug and the same has not yet been reported, raise a new issue or consider fixing it and sending a pull request.
wink-bm25-text-search is copyright 2017 GRAYPE Systems Private Limited.
It is licensed under the under the terms of the GNU Affero General Public License as published by the Free Software Foundation, version 3 of the License.