Compare commits

..

53 Commits

Author SHA1 Message Date
dd31045338 update
Some checks failed
continuous-integration/drone/push Build is failing
2021-05-24 14:54:51 +02:00
291f011226 adapt stryker
All checks were successful
continuous-integration/drone/push Build is passing
2020-11-05 08:06:14 +01:00
bbcd2050e7 separate filter functions 2020-11-05 07:50:27 +01:00
1162664263 fix
Some checks failed
continuous-integration/drone/push Build is failing
2020-11-03 09:36:31 +01:00
32066ad187 update stryker
All checks were successful
continuous-integration/drone/push Build is passing
2020-10-12 17:28:00 +02:00
1ae21ed665 use jasmine
All checks were successful
continuous-integration/drone/push Build is passing
2020-10-05 20:37:48 +02:00
c7ad5298af update-scanner: automatic update
Some checks failed
continuous-integration/drone/push Build is failing
eslint: 7.7.0 ==> 7.8.1 minor
2020-09-07 13:20:41 +02:00
76cff0e692 update-scanner: automatic update
Some checks failed
continuous-integration/drone/push Build is failing
ava: 3.11.1 ==> 3.12.1 minor
2020-08-30 15:47:45 +02:00
b5aa365324 update-scanner: automatic update
Some checks failed
continuous-integration/drone/push Build is failing
eslint: 7.6.0 ==> 7.7.0 minor
2020-08-19 08:12:39 +02:00
c25ec21d5a remove unnecessary eslint disable
Some checks failed
continuous-integration/drone/push Build is failing
2020-08-10 08:22:11 +02:00
59ed696163 fix drone config
Some checks failed
continuous-integration/drone/push Build is failing
2020-08-07 08:14:36 +02:00
51fe00060b update-scanner: automatic update
Some checks failed
continuous-integration/drone/push Build is failing
ava: 3.10.1 ==> 3.11.1 minor
eslint: 7.5.0 ==> 7.6.0 minor
2020-08-04 12:42:49 +02:00
6d51a10a2a update-scanner: automatic update
All checks were successful
continuous-integration/drone/push Build is passing
@sapphirecode/eslint-config: 2.1.15 ==> 2.1.16 minor
2020-07-19 14:55:27 +02:00
dd94095bcf fix
Some checks failed
continuous-integration/drone/push Build is failing
2020-07-19 14:28:38 +02:00
f1ced3dacf update-scanner: automatic update
Some checks failed
continuous-integration/drone/push Build is failing
eslint: 7.4.0 ==> 7.5.0 minor
2020-07-19 11:59:35 +02:00
cacdcae9e0 update-scanner: automatic update
All checks were successful
continuous-integration/drone/push Build is passing
@sapphirecode/eslint-config: 2.1.13 ==> 2.1.15 minor
@stryker-mutator/core: 3.3.0 ==> 3.3.1 minor
@stryker-mutator/javascript-mutator: 3.3.0 ==> 3.3.1 minor
ava: 3.9.0 ==> 3.10.1 minor
eslint: 7.3.1 ==> 7.4.0 minor
2020-07-10 12:20:12 +02:00
4bdc6354d8 switch to drone
All checks were successful
continuous-integration/drone/push Build is passing
2020-07-10 08:26:23 +02:00
1f63b6ccb6 fix jsdoc 2020-07-03 14:42:37 +02:00
0dab4976ae allow custom functions in filters 2020-07-03 14:36:39 +02:00
dab2b9a852 update-scanner: automatic update
@sapphirecode/eslint-config: 2.1.12 ==> 2.1.13 minor
2020-07-01 09:18:44 +02:00
ed93abeef4 fix 2020-06-30 11:21:30 +02:00
092e84d8bb carry values between filter runs 2020-06-30 11:20:09 +02:00
141c698351 fix type def 2020-06-30 10:13:17 +02:00
9a62bd2905 Revert "jenkins version"
This reverts commit 476aec75b4.
2020-06-30 10:06:53 +02:00
476aec75b4 jenkins version 2020-06-30 10:06:04 +02:00
c9e419d8e9 filter groups 2020-06-30 10:00:36 +02:00
7939aaf3a7 fix 2020-06-29 14:25:03 +02:00
94f7ec85e9 fix 2020-06-29 14:23:24 +02:00
3cca634101 Revert "bump jenkins version"
This reverts commit 82ca2e8a3e.
2020-06-29 14:10:42 +02:00
82ca2e8a3e bump jenkins version 2020-06-29 14:08:53 +02:00
482cebe73a Revert "search index function"
This reverts commit 64f273a6ae.
2020-06-29 13:58:09 +02:00
7c95a0758f Revert "fix indexing"
This reverts commit 01be6cede8.
2020-06-29 13:57:49 +02:00
01be6cede8 fix indexing 2020-06-29 13:41:07 +02:00
64f273a6ae search index function 2020-06-29 13:08:23 +02:00
556645844d filter out undefined fields 2020-06-29 12:37:28 +02:00
8eb807714c optimize recursive filter 2020-06-29 11:33:33 +02:00
d8ee6074fa safer test on copy_object 2020-06-29 11:33:15 +02:00
c5aef0a81e remove auto generated file 2020-06-29 11:07:05 +02:00
af709965c5 linting 2020-06-26 16:12:16 +02:00
9f41d623e4 allow specifying multiple fields 2020-06-26 16:11:30 +02:00
0932891b20 do not modify original object on recursive filter 2020-06-26 15:52:53 +02:00
33597eb4e9 update-scanner: automatic update
@sapphirecode/eslint-config: 2.1.10 ==> 2.1.12 minor
eslint: 7.3.0 ==> 7.3.1 minor
2020-06-24 12:38:35 +02:00
42961836ba update-scanner: automatic update
@sapphirecode/eslint-config: 2.1.9 ==> 2.1.10 minor
eslint: 7.2.0 ==> 7.3.0 minor
2020-06-22 08:08:52 +02:00
dc1e0cf0bd update-scanner: automatic update
@stryker-mutator/core: 3.2.4 ==> 3.3.0 minor
@stryker-mutator/javascript-mutator: 3.2.4 ==> 3.3.0 minor
ava: 3.8.2 ==> 3.9.0 minor
2020-06-19 12:56:13 +02:00
9f10af6d0e update-scanner: automatic update
@sapphirecode/eslint-config: 2.1.8 ==> 2.1.9 minor
eslint: 7.1.0 ==> 7.2.0 minor
2020-06-11 20:15:54 +02:00
84ef0f7bc7 filter recursive function 2020-06-08 14:44:09 +02:00
01d90058ab update-scanner: automatic update
@sapphirecode/eslint-config: 2.1.7 ==> 2.1.8 minor
nyc: 15.0.1 ==> 15.1.0 minor
2020-06-02 08:33:21 +02:00
99fc79d312 update-scanner: automatic update
@sapphirecode/eslint-config: 2.1.6 ==> 2.1.7 minor
@stryker-mutator/core: 3.2.3 ==> 3.2.4 minor
@stryker-mutator/javascript-mutator: 3.2.3 ==> 3.2.4 minor
eslint: 7.0.0 ==> 7.1.0 minor
2020-05-23 18:04:54 +02:00
29a023b32b update-scanner: automatic update
@sapphirecode/eslint-config: 2.1.3 ==> 2.1.6 minor
@stryker-mutator/core: 3.2.1 ==> 3.2.3 minor
@stryker-mutator/javascript-mutator: 3.2.1 ==> 3.2.3 minor
2020-05-17 19:24:21 +02:00
f4f77630d7 update jenkins.js 2020-05-15 13:16:46 +02:00
2d943894c1 update jenkins.js 2020-05-15 12:22:33 +02:00
18f8739fd7 fix 2020-05-15 12:20:10 +02:00
15f41cbb62 documentation, update eslint, copyright 2020-05-15 12:18:37 +02:00
18 changed files with 1618 additions and 2154 deletions

14
.drone.yml Normal file
View File

@ -0,0 +1,14 @@
kind: pipeline
name: default
steps:
- name: setup
image: registry:5000/node-build
commands:
- yarn
- curl https://git.scode.ovh/Timo/standard/raw/branch/master/ci.js > ci.js
- name: build
image: registry:5000/node-build
commands:
- node ci.js

View File

@ -1 +1,2 @@
*.d.ts
jenkins.js

View File

@ -1,24 +1,22 @@
/*
* Copyright (C) Sapphirecode - All Rights Reserved
* This file is part of Utilities which is released under MIT.
* This file is part of utilities which is released under MIT.
* See file 'LICENSE' for full license details.
* Created by Timo Hocker <timo@sapphirecode.ovh>, March 2020
* Created by Timo Hocker <timo@scode.ovh>, May 2020
*/
'use strict';
module.exports = {
env: {
commonjs: true,
es6: true,
node: true
es6: true,
node: true
},
extends: [
'@sapphirecode'
],
extends: [ '@sapphirecode' ],
globals: {
Atomics: 'readonly',
Atomics: 'readonly',
SharedArrayBuffer: 'readonly'
},
parserOptions: {
ecmaVersion: 2018
}
}
parserOptions: { ecmaVersion: 2018 }
};

8
.liconfig.json Normal file
View File

@ -0,0 +1,8 @@
{
"has_license": true,
"license": "MIT",
"author": "Timo Hocker",
"company": "Sapphirecode",
"email": "timo@scode.ovh",
"software": "utilities"
}

27
CHANGELOG.md Normal file
View File

@ -0,0 +1,27 @@
# Changelog
## 1.8.0
custom filter functions
## 1.7.0
carry values between filter runs
## 1.6.0
filter groups
## 1.5.0
filter over multiple fields
## 1.4.0
function to filter arrays of objects recursively
## 1.1.0 - 1.3.0 (lost changelog)
## 1.0.0
initial release

23
Jenkinsfile vendored
View File

@ -1,23 +0,0 @@
pipeline {
agent any
environment {
VERSION = VersionNumber([
versionNumberString:
'${BUILDS_ALL_TIME}',
versionPrefix: '1.0.',
worstResultForIncrement: 'SUCCESS'
])
}
stages {
stage('Building') {
steps {
script {
currentBuild.displayName = env.VERSION
}
sh 'yarn ci ${VERSION}'
}
}
}
}

7
LICENSE Normal file
View File

@ -0,0 +1,7 @@
MIT License Copyright (c) <year> <author>
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice (including the next paragraph) shall be included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.

186
README.md
View File

@ -1,27 +1,193 @@
# Utilities
# @sapphirecode/utilities
General helper functions
version: 1.8.x
small utility functions to make much needed features easier to work with
## Installation
npm:
> npm i --save @sapphirecode/utilities
yarn:
> yarn add @sapphirecode/utilities
## Usage
```js
const util = require('@scode/utilities');
const util = require('@sapphirecode/utilities');
```
// cut off decimal places to a specified point
cut off decimal places to a specified point
```js
util.truncate_decimal(12.345678, 2);
// returns 12.34
```
// will return null instead of throwing on invalid json
will return null instead of throwing on invalid json
```js
util.try_parse_json('{{foo');
```
// copy an object to prevent modification of the original
const obj = {foo:'bar'};
copy an object to prevent modification of the original
```js
const obj = {foo: 'bar'};
const copy = util.copy_object(obj);
copy.foo = 'baz';
console.log(obj.foo); // bar
```
// run a regular expression and get a callback for every result
const data = "foobarfoo";
run a regular expression and get a callback for every result
```js
const data = 'foobarfoo';
const regex = /foo/g;
util.run_regex(regex, data, res => {
util.run_regex(regex, data, (res) => {
console.log(res[0]); // will output 'foo' 2 times
});
```
check if a variable is null, undefined or NaN
```js
console.log(util.is_nil(parseInt('abc'))); // true
console.log(util.is_nil('foo')); // false
console.log(util.is_nil(42)); // false
console.log(util.is_nil(null)); // true
console.log(util.is_nil(undefined)); // true
```
filter an array recursively
```js
const to_filter = [
{name: 'include_foo'},
{
name: 'include_bar',
children: [{name: 'foo'}, {name: 'bar'}],
},
{
name: 'baz',
children: [{name: 'include_foo'}, {name: 'bar'}],
},
{
name: 'barbaz',
children: [{name: 'foo'}, {name: 'bar'}],
},
];
const filter = {
field: 'name', // the field the filter will apply on
filter: /^include_.*/iu, // regex filter
};
const result = util.recursive_filter(
to_filter,
[filter], // you can specify multiple filters, they will be connected using AND
'children' // specify which field an objects children are stored in ('children' is default)
);
console.log(JSON.stringify(result, null, 2));
/* output:
[
{ name: 'include_foo' },
{
name: 'include_bar',
children: [
{ name: 'foo' },
{ name: 'bar' }
]
},
{
name: 'baz',
children: [ { name: 'include_foo' } ]
}
]*/
```
filters can also apply to multiple fields
```js
const result = util.recursive_filter(
[{name: 'foo', name_2: 'bar'}],
[
{
filter: /foo bar/iu,
field: ['name', 'name_2'], // fields will be joined with a space in between
// {name: 'foo', name_2: 'bar'} will become 'foo bar'
},
]
);
```
filter groups can be used to connect filters with OR
```js
const result = util.recursive_filter(
[...],
[
{
filter: /foo bar/iu,
field: 'name'
},
{
or: [
{ filter: /foo/u, field: 'bar' },
{ filter: /bar/u, field: 'bar' }
]
}
]
);
```
field values can be carried to filter runs on children to filter over the entire tree instead of just one element
```js
const to_filter = [
{
name: 'foo',
children: [
{ name: 'bar' },
{ name: 'baz' },
{ foo: 'bar' }
]
}
];
const res = util.recursive_filter (
to_filter,
[ { field: 'name', filter: /foo bar/ui } ], // values will be separaed by a space
'children',
[ 'name' ] // specify which fields should be carried
// the filters will receive the carried values instead of the plain fields
);
/* result:
[
{
name: 'foo',
children: [ { name: 'bar' } ]
}
]
*/
```
custom functions to match items can also be used
```js
const filter = {
function: (element) => {
return element.should_match;
}
}
```
## License
MIT © Timo Hocker <timo@scode.ovh>

121
filter.js Normal file
View File

@ -0,0 +1,121 @@
/*
* Copyright (C) Sapphirecode - All Rights Reserved
* This file is part of utilities which is released under MIT.
* See file 'LICENSE' for full license details.
* Created by Timo Hocker <timo@scode.ovh>, November 2020
*/
'use strict';
function get_element_field (element, carried_data, field) {
if (typeof carried_data[field] !== 'undefined')
return carried_data[field];
return element[field];
}
function check_filter (filter, e, carried_data) {
if (typeof filter.function !== 'undefined')
return filter.function ({ ...e, ...carried_data });
const search_str = Array.isArray (filter.field)
? filter.field.map ((f) => get_element_field (e, carried_data, f))
.filter ((v) => typeof v !== 'undefined')
.join (' ')
: get_element_field (e, carried_data, filter.field);
return filter.filter.test (search_str);
}
function check_filters (filters, e, carried_data, or = false) {
for (const filter of filters) {
let res = false;
if (Array.isArray (filter.or))
res = check_filters (filter.or, e, carried_data, true);
else
res = check_filter (filter, e, carried_data);
if (or && res)
return true;
if (!res && !or)
return false;
}
return !or;
}
/**
* @typedef Filter
* @type {object}
* @property {string|string[]} field - fields to apply filter on
* @property {RegExp} filter - filter
*/
/**
* @typedef FilterFunction
* @type {object}
* @property {function} function - function to test element for match
*/
/**
* @typedef FilterOrGroup
* @type {object}
* @property {FilterType[]} or - create an OR group of filters
*/
/**
* @typedef FilterType
* @type {Filter|FilterFunction|FilterOrGroup}
*/
/**
* filter nested objects
*
* @param {Array<object>} input object to filter
* @param {FilterType[]} filters filters
* @param {string[]} carry carry data to children to match
* @param {object} carried_data internal: carried data
* @returns {Array<object>} filtered data
*/
function recursive_filter (
input,
filters,
children_key = 'children',
carry = [],
carried_data = {}
) {
const data = [ ...input ];
const filtered = [];
for (const c of carry) {
if (typeof carried_data[c] !== 'string')
carried_data[c] = '';
}
for (let i = 0; i < data.length; i++) {
const e = { ...data[i] };
data[i] = e;
const sub_carry = { ...carried_data };
for (const c of carry) {
if (typeof e[c] !== 'undefined')
sub_carry[c] += `${sub_carry[c].length > 0 ? ' ' : ''}${e[c]}`;
}
if (check_filters (filters, e, sub_carry)) {
filtered.push (e);
}
else {
if (typeof e[children_key] === 'undefined')
continue;
e[children_key] = recursive_filter (
e[children_key],
filters,
children_key,
carry,
sub_carry
);
if (e[children_key].length > 0)
filtered.push (e);
}
}
return filtered;
}
module.exports = recursive_filter;

37
index.d.ts vendored
View File

@ -1,37 +0,0 @@
/**
* truncates a floating point number
*
* @param {number} num number to truncate
* @param {number} len length to truncate to
* @returns {number} truncated number
*/
export function truncate_decimal(num: number, len: number): number;
/**
* parse json and catch invalid strings
*
* @param {string} text input
* @returns {any} parsed
*/
export function try_parse_json(text: string): any;
/**
* copy an object to prevent modification to the original
*
* @param {object} obj object to copy
* @returns {object} copy
*/
export function copy_object(obj: any): any;
/**
* run a regular expression and callback for every result
*
* @param {any} regex regular expression
* @param {any} data data to run on
* @param {any} func function to execute
*/
export function run_regex(regex: any, data: any, func: any): void;
/**
* check if an object is either undefined, null or NaN
*
* @param {any} obj object to check
* @returns {boolean} true if nil
*/
export function is_nil(obj: any): boolean;

View File

@ -1,12 +1,14 @@
/*
* Copyright (C) Sapphirecode - All Rights Reserved
* This file is part of Utilities which is released under MIT.
* This file is part of utilities which is released under MIT.
* See file 'LICENSE' for full license details.
* Created by Timo Hocker <timo@sapphirecode.ovh>, March 2020
* Created by Timo Hocker <timo@scode.ovh>, May 2020
*/
'use strict';
const recursive_filter = require ('./filter');
/**
* truncates a floating point number
*
@ -83,5 +85,6 @@ module.exports = {
try_parse_json,
copy_object,
run_regex,
is_nil
is_nil,
recursive_filter
};

14
jasmine.json Normal file
View File

@ -0,0 +1,14 @@
{
"spec_dir": "test",
"spec_files": [
"spec/*.js",
"spec/*.ts"
],
"helpers": [
"helpers/*.js",
"helpers/*.ts"
],
"stopSpecOnExpectationFailure": false,
"random": false
}

View File

@ -1,47 +0,0 @@
/* eslint-disable */
'use strict';
const fs = require ('fs');
const child_process = require ('child_process');
const pkg = JSON.parse (fs.readFileSync ('package.json', 'utf-8'));
[
,, pkg.version
] = process.argv;
fs.writeFileSync ('package.json', JSON.stringify (pkg, null, 2));
child_process.execSync ('yarn lint', { stdio: 'inherit' });
child_process.execSync ('yarn test', { stdio: 'inherit' });
child_process.execSync ('yarn compile', { stdio: 'inherit' });
if (typeof pkg.description === 'undefined' || pkg.description === '') {
console.log ('description undefined');
process.exit (1);
}
if (typeof pkg.repository === 'undefined') {
console.log ('repository undefined');
process.exit (1);
}
function major (version) {
return version.replace (/\.[0-9]+$/ui, '');
}
if (fs.existsSync ('README.md')) {
const readme = fs.readFileSync ('README.md', 'utf-8');
const version = (/version: ([0-9.]+)/ui).exec (readme);
if (
version === null
|| major (version[1]) !== major (pkg.version)
) {
console.log ('readme version does not match package version');
process.exit (1);
}
else { child_process.execSync ('yarn publish --access public'); }
}
else {
console.log ('readme does not exist');
process.exit (1);
}

View File

@ -1,27 +1,45 @@
{
"name": "@sapphirecode/utilities",
"version": "1.0.0",
"version": "1.8.8",
"main": "index.js",
"author": "Timo Hocker <t-hocker@web.de>",
"author": {
"name": "Timo Hocker",
"email": "timo@scode.ovh"
},
"bugs": "https://redmine.scode.ovh/projects/utilities",
"license": "MIT",
"keywords": [
"utilities",
"regex",
"json",
"filters"
],
"repository": {
"type": "git",
"url": "https://git.scode.ovh:timo/utilities.git"
},
"description": "small utility functions to make much needed features easier to work with",
"devDependencies": {
"@sapphirecode/eslint-config": "^2.0.2",
"@stryker-mutator/core": "^3.0.2",
"@stryker-mutator/javascript-mutator": "^3.0.2",
"ava": "^3.5.0",
"eslint": "^6.8.0",
"nyc": "^15.0.0"
"@sapphirecode/eslint-config": "^2.1.3",
"@stryker-mutator/core": "^5.0.0",
"@stryker-mutator/jasmine-runner": "^5.0.0",
"@types/jasmine": "^3.5.14",
"eslint": "^7.0.0",
"jasmine": "^3.6.1",
"nyc": "^15.0.1"
},
"scripts": {
"lint": "eslint . --ext .js,.jsx,.ts,.tsx,.vue,.mjs",
"test": "nyc ava",
"ci": "yarn && node jenkins.js",
"test": "nyc jasmine --config=\"jasmine.json\"",
"mutate": "stryker run",
"compile": "tsc --allowJs --declaration --emitDeclarationOnly index.js"
},
"files": [
"LICENSE",
"index.js",
"index.d.ts"
]
"*.js",
"*.d.ts"
],
"engines": {
"node": ">=10"
}
}

View File

@ -1,8 +1,8 @@
/*
* Copyright (C) Sapphirecode - All Rights Reserved
* This file is part of Utilities which is released under MIT.
* This file is part of utilities which is released under MIT.
* See file 'LICENSE' for full license details.
* Created by Timo Hocker <timo@sapphirecode.ovh>, March 2020
* Created by Timo Hocker <timo@scode.ovh>, May 2020
*/
'use strict';
@ -11,14 +11,19 @@
* @type {import('@stryker-mutator/api/core').StrykerOptions}
*/
module.exports = {
mutator: 'javascript',
packageManager: 'yarn',
reporters: [
'clear-text',
'progress'
],
testRunner: 'command',
transpilers: [],
coverageAnalysis: 'all',
mutate: [ 'index.js' ]
testRunner: 'jasmine',
jasmineConfigFile: 'jasmine.json',
coverageAnalysis: 'perTest',
mutate: [
'**/*.js',
'**/*.ts',
'!**/test/**/*',
'!**/spec/**/*',
'!stryker.conf.js'
]
};

View File

@ -1,102 +0,0 @@
/*
* Copyright (C) Sapphirecode - All Rights Reserved
* This file is part of Utilities which is released under MIT.
* See file 'LICENSE' for full license details.
* Created by Timo Hocker <timo@scode.ovh>, March 2020
*/
/* eslint-disable no-magic-numbers */
// @ts-nocheck
'use strict';
const test = require ('ava');
const util = require ('../index');
test ('truncate_decimal', (t) => {
const trunc = util.truncate_decimal (1.23456, 2);
t.is (trunc, 1.23);
});
test ('try_parse_json should parse', (t) => {
const str = '{"test":"foo"}';
t.notThrows (() => {
const json = util.try_parse_json (str);
t.deepEqual (json, { test: 'foo' });
});
});
test ('try_parse_json should fail', (t) => {
const str = '{"test":foo"}';
t.notThrows (() => {
const json = util.try_parse_json (str);
t.is (json, null);
});
});
test ('copy object', (t) => {
const obj = { foo: 'bar' };
const copy = util.copy_object (obj);
copy.foo = 'baz';
t.is (copy.foo, 'baz');
t.is (obj.foo, 'bar');
});
test ('run regex', (t) => {
const data = 'foobarfoo';
const regex = /foo/gu;
let count = 0;
util.run_regex (regex, data, (res) => {
t.is (res[0], 'foo');
count++;
});
t.is (count, 2);
});
test ('run non-global regex', (t) => {
const data = 'foobarfoo';
const regex = /foo/u;
let count = 0;
util.run_regex (regex, data, (res) => {
t.is (res[0], 'foo');
count++;
});
t.is (count, 1);
});
test ('run non-global regex without result', (t) => {
const data = 'foobarfoo';
const regex = /baz/u;
let count = 0;
util.run_regex (regex, data, () => {
count++;
});
t.is (count, 0);
});
test ('check isnil with undefined', (t) => {
t.is (util.is_nil (), true);
});
test ('check isnil with null', (t) => {
t.is (util.is_nil (null), true);
});
test ('check isnil with empty string', (t) => {
t.is (util.is_nil (''), false);
});
test ('check isnil with string', (t) => {
t.is (util.is_nil ('foo'), false);
});
test ('check isnil with 0', (t) => {
t.is (util.is_nil (0), false);
});
test ('check isnil with nan', (t) => {
t.is (util.is_nil (parseInt ('foo')), true);
});
test ('check isnil with int', (t) => {
t.is (util.is_nil (parseInt ('42')), false);
});

309
test/spec/index.js Normal file
View File

@ -0,0 +1,309 @@
/*
* Copyright (C) Sapphirecode - All Rights Reserved
* This file is part of utilities which is released under MIT.
* See file 'LICENSE' for full license details.
* Created by Timo Hocker <timo@scode.ovh>, May 2020
*/
// @ts-nocheck
'use strict';
const util = require ('../../index');
// eslint-disable-next-line max-lines-per-function
describe ('utilities', () => {
it ('should truncate_decimal', () => {
const trunc = util.truncate_decimal (1.23456, 2);
expect (trunc)
.toEqual (1.23);
});
it ('try_parse_json should parse', () => {
const str = '{"test":"foo"}';
expect (() => {
const json = util.try_parse_json (str);
expect (json)
.toEqual ({ test: 'foo' });
}).not.toThrow ();
});
it ('try_parse_json should fail', () => {
const str = '{"test":foo"}';
expect (() => {
const json = util.try_parse_json (str);
expect (json)
.toBeNull ();
}).not.toThrow ();
});
it ('should copy object', () => {
const obj = { foo: 'bar', bar: { foo: 'baz' } };
const copy = util.copy_object (obj);
expect (obj).not.toBe (copy);
expect (obj)
.toEqual (copy);
copy.foo = 'baz';
copy.bar.foo = 'bar';
expect (copy.foo)
.toEqual ('baz');
expect (copy.bar.foo)
.toEqual ('bar');
expect (obj.foo)
.toEqual ('bar');
expect (obj.bar.foo)
.toEqual ('baz');
});
it ('should run regex', () => {
const data = 'foobarfoo';
const regex = /foo/gu;
let count = 0;
util.run_regex (regex, data, (res) => {
expect (res[0])
.toEqual ('foo');
count++;
});
expect (count)
.toEqual (2);
});
it ('should run non-global regex', () => {
const data = 'foobarfoo';
const regex = /foo/u;
let count = 0;
util.run_regex (regex, data, (res) => {
expect (res[0])
.toEqual ('foo');
count++;
});
expect (count)
.toEqual (1);
});
it ('should run non-global regex without result', () => {
const data = 'foobarfoo';
const regex = /baz/u;
let count = 0;
util.run_regex (regex, data, () => {
count++;
});
expect (count)
.toEqual (0);
});
it ('should check isnil with undefined', () => {
expect (util.is_nil ())
.toEqual (true);
});
it ('should check isnil with null', () => {
expect (util.is_nil (null))
.toEqual (true);
});
it ('should check isnil with empty string', () => {
expect (util.is_nil (''))
.toEqual (false);
});
it ('should check isnil with string', () => {
expect (util.is_nil ('foo'))
.toEqual (false);
});
it ('should check isnil with 0', () => {
expect (util.is_nil (0))
.toEqual (false);
});
it ('should check isnil with nan', () => {
expect (util.is_nil (parseInt ('foo')))
.toEqual (true);
});
it ('should check isnil with int', () => {
expect (util.is_nil (parseInt ('42')))
.toEqual (false);
});
it ('recursive filter', () => {
const raw = [
{ name: 'include_foo' },
{
name: 'include_bar',
children: [
{ name: 'foo' },
{ name: 'bar' }
]
},
{
name: 'baz',
children: [
{ name: 'include_foo' },
{ name: 'bar' }
]
},
{
name: 'barbaz',
children: [
{ name: 'foo' },
{ name: 'bar' }
]
}
];
const filtered = [
{ name: 'include_foo' },
{
name: 'include_bar',
children: [
{ name: 'foo' },
{ name: 'bar' }
]
},
{
name: 'baz',
children: [ { name: 'include_foo' } ]
}
];
const filter = {
field: 'name',
filter: /^include_.*/ui
};
const result = util.recursive_filter (raw, [ filter ]);
expect (filtered)
.toEqual (result);
});
it ('recursive filter multifield', () => {
const raw = [
{ name: 'foo', f: 'include' },
{
name: 'include_bar',
children: [
{ name: 'foo' },
{ name: 'bar' }
]
},
{
name: 'baz',
children: [
{ name: 'include_foo' },
{ name: 'bar' }
]
},
{
name: 'barbaz',
children: [
{ name: 'foo' },
{ name: 'bar' }
]
}
];
const filtered = [ { name: 'foo', f: 'include' } ];
const filter = {
field: [
'name',
'f'
],
filter: /foo include/ui
};
const result = util.recursive_filter (raw, [ filter ]);
expect (filtered)
.toEqual (result);
});
it ('recursive filter undefined multifield', () => {
const res = util.recursive_filter (
[ { foo: 'bar' } ],
[
{
field: [
'foo',
'bar'
],
filter: /\s/u
}
]
);
expect (res)
.toEqual ([]);
});
it ('recursive filter with or group', () => {
const to_filter = [
{ name: 'foo' },
{ name: 'bar' },
{ name: 'baz' }
];
const filter = [
{
or: [
{ field: 'name', filter: /foo/u },
{ field: 'name', filter: /bar/u }
]
}
];
const res = util.recursive_filter (to_filter, filter);
expect (res)
.toEqual (to_filter.slice (0, 2));
});
it ('recursive filter carry field', () => {
const to_filter = [
{
name: 'foo',
children: [
{ name: 'bar' },
{ name: 'baz' },
{ foo: 'bar' }
]
}
];
const res = util.recursive_filter (
to_filter,
[ { field: 'name', filter: /foo bar/ui } ],
'children',
[ 'name' ]
);
expect (res)
.toEqual ([
{
name: 'foo',
children: [ { name: 'bar' } ]
}
]);
});
it ('recursive filter custom function', () => {
const to_filter = [
{
name: 'foo',
children: [
{ name: 'bar' },
{ name: 'baz' },
{ foo: 'bar' }
]
}
];
const res = util.recursive_filter (
to_filter,
[ { function: (elem) => elem.name === 'foo bar' } ],
'children',
[ 'name' ]
);
expect (res)
.toEqual ([
{
name: 'foo',
children: [ { name: 'bar' } ]
}
]);
});
});

2782
yarn.lock

File diff suppressed because it is too large Load Diff