2020-03-04 12:13:28 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) Sapphirecode - All Rights Reserved
|
2020-05-15 12:18:37 +02:00
|
|
|
* This file is part of utilities which is released under MIT.
|
2020-03-25 17:01:22 +01:00
|
|
|
* See file 'LICENSE' for full license details.
|
2020-05-15 12:18:37 +02:00
|
|
|
* Created by Timo Hocker <timo@scode.ovh>, May 2020
|
2020-03-04 12:13:28 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
'use strict';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* truncates a floating point number
|
|
|
|
*
|
|
|
|
* @param {number} num number to truncate
|
|
|
|
* @param {number} len length to truncate to
|
|
|
|
* @returns {number} truncated number
|
|
|
|
*/
|
|
|
|
function truncate_decimal (num, len) {
|
|
|
|
return Math.round (num * (10 ** len)) / (10 ** len);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* parse json and catch invalid strings
|
|
|
|
*
|
|
|
|
* @param {string} text input
|
|
|
|
* @returns {any} parsed
|
|
|
|
*/
|
|
|
|
function try_parse_json (text) {
|
|
|
|
try {
|
|
|
|
return JSON.parse (text);
|
|
|
|
}
|
|
|
|
catch (e) {
|
|
|
|
// noop
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2020-03-10 10:46:12 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* copy an object to prevent modification to the original
|
|
|
|
*
|
|
|
|
* @param {object} obj object to copy
|
|
|
|
* @returns {object} copy
|
|
|
|
*/
|
|
|
|
function copy_object (obj) {
|
|
|
|
return JSON.parse (JSON.stringify (obj));
|
|
|
|
}
|
|
|
|
|
2020-03-30 11:00:52 +02:00
|
|
|
/**
|
|
|
|
* run a regular expression and callback for every result
|
|
|
|
*
|
2020-03-30 11:01:51 +02:00
|
|
|
* @param {any} regex regular expression
|
|
|
|
* @param {any} data data to run on
|
|
|
|
* @param {any} func function to execute
|
2020-03-30 11:00:52 +02:00
|
|
|
*/
|
|
|
|
function run_regex (regex, data, func) {
|
2020-03-30 11:21:05 +02:00
|
|
|
if (!regex.global) {
|
|
|
|
const result = regex.exec (data);
|
|
|
|
if (result)
|
|
|
|
func (result);
|
|
|
|
return;
|
|
|
|
}
|
2020-03-30 11:00:52 +02:00
|
|
|
let res = regex.exec (data);
|
|
|
|
while (res) {
|
|
|
|
func (res);
|
|
|
|
res = regex.exec (data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-27 12:40:54 +02:00
|
|
|
/**
|
|
|
|
* check if an object is either undefined, null or NaN
|
|
|
|
*
|
|
|
|
* @param {any} obj object to check
|
|
|
|
* @returns {boolean} true if nil
|
|
|
|
*/
|
|
|
|
function is_nil (obj) {
|
|
|
|
return typeof obj === 'undefined'
|
|
|
|
|| obj === null
|
|
|
|
|| (typeof obj === 'number' && isNaN (obj));
|
|
|
|
}
|
2020-03-30 11:00:52 +02:00
|
|
|
|
2020-06-29 13:08:23 +02:00
|
|
|
function to_search_string (element, field) {
|
|
|
|
return Array.isArray (field)
|
|
|
|
? field.map ((f) => element[f])
|
|
|
|
.filter ((v) => typeof v !== 'undefined')
|
|
|
|
.join (' ')
|
|
|
|
: element[field];
|
|
|
|
}
|
|
|
|
|
2020-06-08 14:44:09 +02:00
|
|
|
/**
|
|
|
|
* filter nested objects
|
|
|
|
*
|
2020-06-26 15:52:53 +02:00
|
|
|
* @param {Array<object>} input
|
2020-06-29 13:08:23 +02:00
|
|
|
* @param {Array<{field: string|string[], filter: RegExp}>} filters
|
|
|
|
* @param {string} children_key field where children are stored
|
2020-06-08 14:44:09 +02:00
|
|
|
* @returns {Array<object>} filtered data
|
|
|
|
*/
|
2020-06-26 15:52:53 +02:00
|
|
|
function recursive_filter (input, filters, children_key = 'children') {
|
2020-06-29 11:33:33 +02:00
|
|
|
const data = [ ...input ];
|
2020-06-08 14:44:09 +02:00
|
|
|
const filtered = [];
|
2020-06-29 11:33:33 +02:00
|
|
|
for (let i = 0; i < data.length; i++) {
|
|
|
|
const e = { ...data[i] };
|
|
|
|
data[i] = e;
|
2020-06-08 14:44:09 +02:00
|
|
|
let match = true;
|
|
|
|
for (const filter of filters) {
|
2020-06-29 13:08:23 +02:00
|
|
|
const search_str = to_search_string (e, filter.field);
|
2020-06-26 16:11:30 +02:00
|
|
|
|
|
|
|
if (!filter.filter.test (search_str)) {
|
2020-06-08 14:44:09 +02:00
|
|
|
match = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (match) {
|
|
|
|
filtered.push (e);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (typeof e[children_key] === 'undefined')
|
|
|
|
continue;
|
|
|
|
e[children_key] = recursive_filter (
|
|
|
|
e[children_key],
|
|
|
|
filters,
|
|
|
|
children_key
|
|
|
|
);
|
|
|
|
if (e[children_key].length > 0)
|
|
|
|
filtered.push (e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return filtered;
|
|
|
|
}
|
|
|
|
|
2020-06-29 13:08:23 +02:00
|
|
|
/**
|
|
|
|
* create a search index on an object to speed up the recursive_filter function
|
|
|
|
*
|
|
|
|
* @param {Array<object>} input input object
|
|
|
|
* @param {string|string[]} field field or fields to index
|
|
|
|
* @param {string} children_key field where children are stored
|
|
|
|
* @param {string} index_field field to save index in
|
|
|
|
*/
|
|
|
|
function filter_index (
|
|
|
|
input,
|
|
|
|
field,
|
|
|
|
children_key = 'children',
|
|
|
|
index_field = 'search_index'
|
|
|
|
) {
|
|
|
|
for (const e of input) {
|
|
|
|
let search_str = '';
|
|
|
|
if (Array.isArray (e[children_key])) {
|
|
|
|
filter_index (e[children_key], field, children_key, index_field);
|
|
|
|
search_str += e[children_key]
|
|
|
|
.map ((v) => v[index_field])
|
|
|
|
.filter ((v) => typeof v !== 'undefined')
|
|
|
|
.join (' ');
|
|
|
|
}
|
|
|
|
search_str += to_search_string (e, field);
|
|
|
|
e[index_field] = search_str;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-04 12:13:28 +01:00
|
|
|
module.exports = {
|
|
|
|
truncate_decimal,
|
2020-03-10 10:46:12 +01:00
|
|
|
try_parse_json,
|
2020-03-30 11:00:52 +02:00
|
|
|
copy_object,
|
2020-04-27 12:40:54 +02:00
|
|
|
run_regex,
|
2020-06-08 14:44:09 +02:00
|
|
|
is_nil,
|
2020-06-29 13:08:23 +02:00
|
|
|
recursive_filter,
|
|
|
|
filter_index
|
2020-03-04 12:13:28 +01:00
|
|
|
};
|