731 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			731 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| /**!
 | |
|  * FlexSearch.js
 | |
|  * Author and Copyright: Thomas Wilkerling
 | |
|  * Licence: Apache-2.0
 | |
|  * Hosted by Nextapps GmbH
 | |
|  * https://github.com/nextapps-de/flexsearch
 | |
|  */
 | |
| 
 | |
| import Index from "./index.js";
 | |
| import { DocumentInterface } from "./type.js";
 | |
| import Cache, { searchCache } from "./cache.js";
 | |
| import { create_object, is_array, is_string, is_object, parse_option, get_keys } from "./common.js";
 | |
| import apply_async from "./async.js";
 | |
| import { intersect, intersect_union } from "./intersect.js";
 | |
| import { exportDocument, importDocument } from "./serialize.js";
 | |
| import WorkerIndex from "./worker/index.js";
 | |
| 
 | |
| /**
 | |
|  * @constructor
 | |
|  * @implements {DocumentInterface}
 | |
|  * @param {Object=} options
 | |
|  * @return {Document}
 | |
|  */
 | |
| 
 | |
| function Document(options) {
 | |
| 
 | |
|     if (!(this instanceof Document)) {
 | |
| 
 | |
|         return new Document(options);
 | |
|     }
 | |
| 
 | |
|     const document = options.document || options.doc || options;
 | |
|     let opt;
 | |
| 
 | |
|     this.tree = [];
 | |
|     this.field = [];
 | |
|     this.marker = [];
 | |
|     this.register = create_object();
 | |
|     this.key = (opt = document.key || document.id) && parse_tree(opt, this.marker) || "id";
 | |
|     this.fastupdate = parse_option(options.fastupdate, /* append: */ /* skip update: */ /* skip_update: */!0);
 | |
| 
 | |
|     this.storetree = (opt = document.store) && !0 !== opt && [];
 | |
|     this.store = opt && create_object();
 | |
| 
 | |
| 
 | |
|     // TODO case-insensitive tags
 | |
| 
 | |
|     this.tag = (opt = document.tag) && parse_tree(opt, this.marker);
 | |
|     this.tagindex = opt && create_object();
 | |
| 
 | |
| 
 | |
|     this.cache = (opt = options.cache) && new Cache(opt);
 | |
| 
 | |
|     // do not apply cache again for the indexes
 | |
| 
 | |
|     options.cache = !1;
 | |
| 
 | |
| 
 | |
|     this.worker = options.worker;
 | |
| 
 | |
| 
 | |
|     // this switch is used by recall of promise callbacks
 | |
| 
 | |
|     this.async = !1;
 | |
| 
 | |
|     /** @export */
 | |
|     this.index = parse_descriptor.call(this, options, document);
 | |
| }
 | |
| 
 | |
| export default Document;
 | |
| 
 | |
| /**
 | |
|  * @this Document
 | |
|  */
 | |
| 
 | |
| function parse_descriptor(options, document) {
 | |
| 
 | |
|     const index = create_object();
 | |
|     let field = document.index || document.field || document;
 | |
| 
 | |
|     if (is_string(field)) {
 | |
| 
 | |
|         field = [field];
 | |
|     }
 | |
| 
 | |
|     for (let i = 0, key, opt; i < field.length; i++) {
 | |
| 
 | |
|         key = field[i];
 | |
| 
 | |
|         if (!is_string(key)) {
 | |
| 
 | |
|             opt = key;
 | |
|             key = key.field;
 | |
|         }
 | |
| 
 | |
|         opt = is_object(opt) ? Object.assign({}, options, opt) : options;
 | |
| 
 | |
|         if (this.worker) {
 | |
| 
 | |
|             index[key] = new WorkerIndex(opt);
 | |
| 
 | |
|             if (!index[key].worker) {
 | |
| 
 | |
|                 this.worker = !1;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         if (!this.worker) {
 | |
| 
 | |
|             index[key] = new Index(opt, this.register);
 | |
|         }
 | |
| 
 | |
|         this.tree[i] = parse_tree(key, this.marker);
 | |
|         this.field[i] = key;
 | |
|     }
 | |
| 
 | |
|     if (this.storetree) {
 | |
| 
 | |
|         let store = document.store;
 | |
| 
 | |
|         if (is_string(store)) {
 | |
| 
 | |
|             store = [store];
 | |
|         }
 | |
| 
 | |
|         for (let i = 0; i < store.length; i++) {
 | |
| 
 | |
|             this.storetree[i] = parse_tree(store[i], this.marker);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return index;
 | |
| }
 | |
| 
 | |
| function parse_tree(key, marker) {
 | |
| 
 | |
|     const tree = key.split(":");
 | |
|     let count = 0;
 | |
| 
 | |
|     for (let i = 0; i < tree.length; i++) {
 | |
| 
 | |
|         key = tree[i];
 | |
| 
 | |
|         if (0 <= key.indexOf("[]")) {
 | |
| 
 | |
|             key = key.substring(0, key.length - 2);
 | |
| 
 | |
|             if (key) {
 | |
| 
 | |
|                 marker[count] = !0;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         if (key) {
 | |
| 
 | |
|             tree[count++] = key;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (count < tree.length) {
 | |
| 
 | |
|         tree.length = count;
 | |
|     }
 | |
| 
 | |
|     return 1 < count ? tree : tree[0];
 | |
| }
 | |
| 
 | |
| // TODO support generic function created from string when tree depth > 1
 | |
| 
 | |
| function parse_simple(obj, tree) {
 | |
| 
 | |
|     if (is_string(tree)) {
 | |
| 
 | |
|         obj = obj[tree];
 | |
|     } else {
 | |
| 
 | |
|         for (let i = 0; obj && i < tree.length; i++) {
 | |
| 
 | |
|             obj = obj[tree[i]];
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return obj;
 | |
| }
 | |
| 
 | |
| // TODO support generic function created from string when tree depth > 1
 | |
| 
 | |
| function store_value(obj, store, tree, pos, key) {
 | |
| 
 | |
|     obj = obj[key];
 | |
| 
 | |
|     // reached target field
 | |
| 
 | |
|     if (pos === tree.length - 1) {
 | |
| 
 | |
|         // store target value
 | |
| 
 | |
|         store[key] = obj;
 | |
|     } else if (obj) {
 | |
| 
 | |
|         if (is_array(obj)) {
 | |
| 
 | |
|             store = store[key] = Array(obj.length);
 | |
| 
 | |
|             for (let i = 0; i < obj.length; i++) {
 | |
| 
 | |
|                 // do not increase pos (an array is not a field)
 | |
|                 store_value(obj, store, tree, pos, i);
 | |
|             }
 | |
|         } else {
 | |
| 
 | |
|             store = store[key] || (store[key] = create_object());
 | |
|             key = tree[++pos];
 | |
| 
 | |
|             store_value(obj, store, tree, pos, key);
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| function add_index(obj, tree, marker, pos, index, id, key, _append) {
 | |
| 
 | |
|     obj = obj[key];
 | |
| 
 | |
|     if (obj) {
 | |
| 
 | |
|         // reached target field
 | |
| 
 | |
|         if (pos === tree.length - 1) {
 | |
| 
 | |
|             // handle target value
 | |
| 
 | |
|             if (is_array(obj)) {
 | |
| 
 | |
|                 // append array contents so each entry gets a new scoring context
 | |
| 
 | |
|                 if (marker[pos]) {
 | |
| 
 | |
|                     for (let i = 0; i < obj.length; i++) {
 | |
| 
 | |
|                         index.add(id, obj[i], !0, !0);
 | |
|                     }
 | |
| 
 | |
|                     return;
 | |
|                 }
 | |
| 
 | |
|                 // or join array contents and use one scoring context
 | |
| 
 | |
|                 obj = obj.join(" ");
 | |
|             }
 | |
| 
 | |
|             index.add(id, obj, _append, !0);
 | |
|         } else {
 | |
| 
 | |
|             if (is_array(obj)) {
 | |
| 
 | |
|                 for (let i = 0; i < obj.length; i++) {
 | |
| 
 | |
|                     // do not increase index, an array is not a field
 | |
| 
 | |
|                     add_index(obj, tree, marker, pos, index, id, i, _append);
 | |
|                 }
 | |
|             } else {
 | |
| 
 | |
|                 key = tree[++pos];
 | |
| 
 | |
|                 add_index(obj, tree, marker, pos, index, id, key, _append);
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| /**
 | |
|  *
 | |
|  * @param id
 | |
|  * @param content
 | |
|  * @param {boolean=} _append
 | |
|  * @returns {Document|Promise}
 | |
|  */
 | |
| 
 | |
| Document.prototype.add = function (id, content, _append) {
 | |
| 
 | |
|     if (is_object(id)) {
 | |
| 
 | |
|         content = id;
 | |
|         id = parse_simple(content, this.key);
 | |
|     }
 | |
| 
 | |
|     if (content && (id || 0 === id)) {
 | |
| 
 | |
|         if (!_append && this.register[id]) {
 | |
| 
 | |
|             return this.update(id, content);
 | |
|         }
 | |
| 
 | |
|         for (let i = 0, tree, field; i < this.field.length; i++) {
 | |
| 
 | |
|             field = this.field[i];
 | |
|             tree = this.tree[i];
 | |
| 
 | |
|             if (is_string(tree)) {
 | |
| 
 | |
|                 tree = [tree];
 | |
|             }
 | |
| 
 | |
|             add_index(content, tree, this.marker, 0, this.index[field], id, tree[0], _append);
 | |
|         }
 | |
| 
 | |
|         if (this.tag) {
 | |
|             let tag = parse_simple(content, this.tag),
 | |
|                 dupes = create_object();
 | |
| 
 | |
| 
 | |
|             if (is_string(tag)) {
 | |
| 
 | |
|                 tag = [tag];
 | |
|             }
 | |
| 
 | |
|             for (let i = 0, key, arr; i < tag.length; i++) {
 | |
| 
 | |
|                 key = tag[i];
 | |
| 
 | |
|                 if (!dupes[key]) {
 | |
| 
 | |
|                     dupes[key] = 1;
 | |
|                     arr = this.tagindex[key] || (this.tagindex[key] = []);
 | |
| 
 | |
|                     if (!_append || !arr.includes(id)) {
 | |
| 
 | |
|                         arr[arr.length] = id;
 | |
| 
 | |
|                         // add a reference to the register for fast updates
 | |
| 
 | |
|                         if (this.fastupdate) {
 | |
| 
 | |
|                             const tmp = this.register[id] || (this.register[id] = []);
 | |
|                             tmp[tmp.length] = arr;
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         // TODO: how to handle store when appending contents?
 | |
| 
 | |
|         if (this.store && (!_append || !this.store[id])) {
 | |
| 
 | |
|             let store;
 | |
| 
 | |
|             if (this.storetree) {
 | |
| 
 | |
|                 store = create_object();
 | |
| 
 | |
|                 for (let i = 0, tree; i < this.storetree.length; i++) {
 | |
| 
 | |
|                     tree = this.storetree[i];
 | |
| 
 | |
|                     if (is_string(tree)) {
 | |
| 
 | |
|                         store[tree] = content[tree];
 | |
|                     } else {
 | |
| 
 | |
|                         store_value(content, store, tree, 0, tree[0]);
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             this.store[id] = store || content;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return this;
 | |
| };
 | |
| 
 | |
| Document.prototype.append = function (id, content) {
 | |
| 
 | |
|     return this.add(id, content, !0);
 | |
| };
 | |
| 
 | |
| Document.prototype.update = function (id, content) {
 | |
| 
 | |
|     return this.remove(id).add(id, content);
 | |
| };
 | |
| 
 | |
| Document.prototype.remove = function (id) {
 | |
| 
 | |
|     if (is_object(id)) {
 | |
| 
 | |
|         id = parse_simple(id, this.key);
 | |
|     }
 | |
| 
 | |
|     if (this.register[id]) {
 | |
| 
 | |
|         for (let i = 0; i < this.field.length; i++) {
 | |
| 
 | |
|             // workers does not share the register
 | |
| 
 | |
|             this.index[this.field[i]].remove(id, !this.worker);
 | |
| 
 | |
|             if (this.fastupdate) {
 | |
| 
 | |
|                 // when fastupdate was enabled all ids are removed
 | |
| 
 | |
|                 break;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         if (this.tag) {
 | |
| 
 | |
|             // when fastupdate was enabled all ids are already removed
 | |
| 
 | |
|             if (!this.fastupdate) {
 | |
| 
 | |
|                 for (let key in this.tagindex) {
 | |
|                     const tag = this.tagindex[key],
 | |
|                           pos = tag.indexOf(id);
 | |
| 
 | |
| 
 | |
|                     if (-1 !== pos) {
 | |
| 
 | |
|                         if (1 < tag.length) {
 | |
| 
 | |
|                             tag.splice(pos, 1);
 | |
|                         } else {
 | |
| 
 | |
|                             delete this.tagindex[key];
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         if (this.store) {
 | |
| 
 | |
|             delete this.store[id];
 | |
|         }
 | |
| 
 | |
|         delete this.register[id];
 | |
|     }
 | |
| 
 | |
|     return this;
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * @param {!string|Object} query
 | |
|  * @param {number|Object=} limit
 | |
|  * @param {Object=} options
 | |
|  * @param {Array<Array>=} _resolve For internal use only.
 | |
|  * @returns {Promise|Array}
 | |
|  */
 | |
| 
 | |
| Document.prototype.search = function (query, limit, options, _resolve) {
 | |
| 
 | |
|     if (!options) {
 | |
| 
 | |
|         if (!limit && is_object(query)) {
 | |
| 
 | |
|             options = /** @type {Object} */query;
 | |
|             query = "";
 | |
|         } else if (is_object(limit)) {
 | |
| 
 | |
|             options = /** @type {Object} */limit;
 | |
|             limit = 0;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     let result = [],
 | |
|         result_field = [],
 | |
|         pluck,
 | |
|         enrich,
 | |
|         field,
 | |
|         tag,
 | |
|         bool,
 | |
|         offset,
 | |
|         count = 0;
 | |
| 
 | |
| 
 | |
|     if (options) {
 | |
| 
 | |
|         if (is_array(options)) {
 | |
| 
 | |
|             field = options;
 | |
|             options = null;
 | |
|         } else {
 | |
| 
 | |
|             query = options.query || query;
 | |
|             pluck = options.pluck;
 | |
|             field = pluck || options.index || options.field /*|| (is_string(options) && [options])*/;
 | |
|             tag = options.tag;
 | |
|             enrich = this.store && options.enrich;
 | |
|             bool = "and" === options.bool;
 | |
|             limit = options.limit || limit || 100;
 | |
|             offset = options.offset || 0;
 | |
| 
 | |
|             if (tag) {
 | |
| 
 | |
|                 if (is_string(tag)) {
 | |
| 
 | |
|                     tag = [tag];
 | |
|                 }
 | |
| 
 | |
|                 // when tags is used and no query was set,
 | |
|                 // then just return the tag indexes
 | |
| 
 | |
|                 if (!query) {
 | |
| 
 | |
|                     for (let i = 0, res; i < tag.length; i++) {
 | |
| 
 | |
|                         res = get_tag.call(this, tag[i], limit, offset, enrich);
 | |
| 
 | |
|                         if (res) {
 | |
| 
 | |
|                             result[result.length] = res;
 | |
|                             count++;
 | |
|                         }
 | |
|                     }
 | |
| 
 | |
|                     return count ? result : [];
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             if (is_string(field)) {
 | |
| 
 | |
|                 field = [field];
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     field || (field = this.field);
 | |
|     bool = bool && (1 < field.length || tag && 1 < tag.length);
 | |
| 
 | |
|     const promises = !_resolve && (this.worker || this.async) && [];
 | |
| 
 | |
|     // TODO solve this in one loop below
 | |
| 
 | |
|     for (let i = 0, res, key, len; i < field.length; i++) {
 | |
| 
 | |
|         let field_options;
 | |
| 
 | |
|         key = field[i];
 | |
| 
 | |
|         if (!is_string(key)) {
 | |
| 
 | |
|             field_options = key;
 | |
|             key = field_options.field;
 | |
|             query = field_options.query || query;
 | |
|             limit = field_options.limit || limit;
 | |
|             enrich = field_options.enrich || enrich;
 | |
|         }
 | |
| 
 | |
|         if (promises) {
 | |
| 
 | |
|             promises[i] = this.index[key].searchAsync(query, limit, field_options || options);
 | |
| 
 | |
|             // just collect and continue
 | |
| 
 | |
|             continue;
 | |
|         } else if (_resolve) {
 | |
| 
 | |
|             res = _resolve[i];
 | |
|         } else {
 | |
| 
 | |
|             // inherit options also when search? it is just for laziness, Object.assign() has a cost
 | |
| 
 | |
|             res = this.index[key].search(query, limit, field_options || options);
 | |
|         }
 | |
| 
 | |
|         len = res && res.length;
 | |
| 
 | |
|         if (tag && len) {
 | |
| 
 | |
|             const arr = [];
 | |
|             let count = 0;
 | |
| 
 | |
|             if (bool) {
 | |
| 
 | |
|                 // prepare for intersection
 | |
| 
 | |
|                 arr[0] = [res];
 | |
|             }
 | |
| 
 | |
|             for (let y = 0, key, res; y < tag.length; y++) {
 | |
| 
 | |
|                 key = tag[y];
 | |
|                 res = this.tagindex[key];
 | |
|                 len = res && res.length;
 | |
| 
 | |
|                 if (len) {
 | |
| 
 | |
|                     count++;
 | |
|                     arr[arr.length] = bool ? [res] : res;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             if (count) {
 | |
| 
 | |
|                 if (bool) {
 | |
| 
 | |
|                     res = intersect(arr, limit || 100, offset || 0);
 | |
|                 } else {
 | |
| 
 | |
|                     res = intersect_union(res, arr);
 | |
|                 }
 | |
| 
 | |
|                 len = res.length;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         if (len) {
 | |
| 
 | |
|             result_field[count] = key;
 | |
|             result[count++] = res;
 | |
|         } else if (bool) {
 | |
| 
 | |
|             return [];
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (promises) {
 | |
| 
 | |
|         const self = this;
 | |
| 
 | |
|         // anyone knows a better workaround of optionally having async promises?
 | |
|         // the promise.all() needs to be wrapped into additional promise,
 | |
|         // otherwise the recursive callback wouldn't run before return
 | |
| 
 | |
|         return new Promise(function (resolve) {
 | |
| 
 | |
|             Promise.all(promises).then(function (result) {
 | |
| 
 | |
|                 resolve(self.search(query, limit, options, result));
 | |
|             });
 | |
|         });
 | |
|     }
 | |
| 
 | |
|     if (!count) {
 | |
| 
 | |
|         // fast path "not found"
 | |
| 
 | |
|         return [];
 | |
|     }
 | |
| 
 | |
|     if (pluck && (!enrich || !this.store)) {
 | |
| 
 | |
|         // fast path optimization
 | |
| 
 | |
|         return result[0];
 | |
|     }
 | |
| 
 | |
|     for (let i = 0, res; i < result_field.length; i++) {
 | |
| 
 | |
|         res = result[i];
 | |
| 
 | |
|         if (res.length) {
 | |
| 
 | |
|             if (enrich) {
 | |
| 
 | |
|                 res = apply_enrich.call(this, res);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         if (pluck) {
 | |
| 
 | |
|             return res;
 | |
|         }
 | |
| 
 | |
|         result[i] = {
 | |
| 
 | |
|             field: result_field[i],
 | |
|             result: res
 | |
|         };
 | |
|     }
 | |
| 
 | |
|     return result;
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * @this Document
 | |
|  */
 | |
| 
 | |
| function get_tag(key, limit, offset) {
 | |
|     let res = this.tagindex[key],
 | |
|         len = res && res.length - offset;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @this Document
 | |
|  */
 | |
| 
 | |
| function apply_enrich(res) {
 | |
| 
 | |
|     const arr = Array(res.length);
 | |
| 
 | |
|     for (let x = 0, id; x < res.length; x++) {
 | |
| 
 | |
|         id = res[x];
 | |
| 
 | |
|         arr[x] = {
 | |
| 
 | |
|             id: id,
 | |
|             doc: this.store[id]
 | |
|         };
 | |
|     }
 | |
| 
 | |
|     return arr;
 | |
| }
 | |
| 
 | |
| Document.prototype.contain = function (id) {
 | |
| 
 | |
|     return !!this.register[id];
 | |
| };
 | |
| 
 | |
| Document.prototype.get = function (id) {
 | |
| 
 | |
|     return this.store[id];
 | |
| };
 | |
| 
 | |
| Document.prototype.set = function (id, data) {
 | |
| 
 | |
|     this.store[id] = data;
 | |
|     return this;
 | |
| };
 | |
| 
 | |
| 
 | |
| Document.prototype.searchCache = searchCache;
 | |
| 
 | |
| 
 | |
| Document.prototype.export = exportDocument;
 | |
| Document.prototype.import = importDocument;
 | |
| 
 | |
| 
 | |
| apply_async(Document.prototype); |