tnp-helpers 13.1.1 → 13.1.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/app.d.ts +1 -1
- package/app.js +6 -6
- package/browser/README.md +24 -24
- package/browser/esm2020/lib/base-component.mjs +47 -47
- package/browser/esm2020/lib/base-formly-component.mjs +125 -125
- package/browser/esm2020/lib/condition-wait.mjs +53 -53
- package/browser/esm2020/lib/constants.mjs +6 -6
- package/browser/esm2020/lib/dual-component-ctrl.mjs +120 -120
- package/browser/esm2020/lib/helpers-array-obj.mjs +80 -80
- package/browser/esm2020/lib/helpers-environment.mjs +30 -30
- package/browser/esm2020/lib/helpers-numbers.mjs +5 -5
- package/browser/esm2020/lib/helpers-strings-regexes.mjs +48 -48
- package/browser/esm2020/lib/helpers-strings.mjs +257 -257
- package/browser/esm2020/lib/helpers.mjs +276 -276
- package/browser/esm2020/lib/index.mjs +14 -14
- package/browser/esm2020/lib/long-press.directive.mjs +111 -111
- package/browser/esm2020/lib/project.mjs +413 -413
- package/browser/esm2020/lib/resize-service.mjs +19 -19
- package/browser/esm2020/public-api.mjs +1 -1
- package/browser/esm2020/tnp-helpers.mjs +4 -4
- package/browser/fesm2015/tnp-helpers.mjs +1535 -1535
- package/browser/fesm2020/tnp-helpers.mjs +1527 -1527
- package/browser/lib/base-component.d.ts +17 -17
- package/browser/lib/base-formly-component.d.ts +31 -31
- package/browser/lib/condition-wait.d.ts +8 -8
- package/browser/lib/constants.d.ts +6 -6
- package/browser/lib/dual-component-ctrl.d.ts +25 -25
- package/browser/lib/helpers-array-obj.d.ts +15 -15
- package/browser/lib/helpers-environment.d.ts +5 -5
- package/browser/lib/helpers-numbers.d.ts +3 -3
- package/browser/lib/helpers-strings-regexes.d.ts +28 -28
- package/browser/lib/helpers-strings.d.ts +50 -50
- package/browser/lib/helpers.d.ts +28 -28
- package/browser/lib/index.d.ts +9 -9
- package/browser/lib/long-press.directive.d.ts +25 -25
- package/browser/lib/project.d.ts +91 -91
- package/browser/lib/resize-service.d.ts +9 -9
- package/browser/public-api.d.ts +1 -1
- package/browser/tnp-helpers.d.ts +5 -5
- package/client/README.md +24 -24
- package/client/esm2020/lib/base-component.mjs +47 -47
- package/client/esm2020/lib/base-formly-component.mjs +125 -125
- package/client/esm2020/lib/condition-wait.mjs +53 -53
- package/client/esm2020/lib/constants.mjs +6 -6
- package/client/esm2020/lib/dual-component-ctrl.mjs +120 -120
- package/client/esm2020/lib/helpers-array-obj.mjs +80 -80
- package/client/esm2020/lib/helpers-environment.mjs +30 -30
- package/client/esm2020/lib/helpers-numbers.mjs +5 -5
- package/client/esm2020/lib/helpers-strings-regexes.mjs +48 -48
- package/client/esm2020/lib/helpers-strings.mjs +257 -257
- package/client/esm2020/lib/helpers.mjs +276 -276
- package/client/esm2020/lib/index.mjs +14 -14
- package/client/esm2020/lib/long-press.directive.mjs +111 -111
- package/client/esm2020/lib/project.mjs +413 -413
- package/client/esm2020/lib/resize-service.mjs +19 -19
- package/client/esm2020/public-api.mjs +1 -1
- package/client/esm2020/tnp-helpers.mjs +4 -4
- package/client/fesm2015/tnp-helpers.mjs +1535 -1535
- package/client/fesm2020/tnp-helpers.mjs +1527 -1527
- package/client/lib/base-component.d.ts +17 -17
- package/client/lib/base-formly-component.d.ts +31 -31
- package/client/lib/condition-wait.d.ts +8 -8
- package/client/lib/constants.d.ts +6 -6
- package/client/lib/dual-component-ctrl.d.ts +25 -25
- package/client/lib/helpers-array-obj.d.ts +15 -15
- package/client/lib/helpers-environment.d.ts +5 -5
- package/client/lib/helpers-numbers.d.ts +3 -3
- package/client/lib/helpers-strings-regexes.d.ts +28 -28
- package/client/lib/helpers-strings.d.ts +50 -50
- package/client/lib/helpers.d.ts +28 -28
- package/client/lib/index.d.ts +9 -9
- package/client/lib/long-press.directive.d.ts +25 -25
- package/client/lib/project.d.ts +91 -91
- package/client/lib/resize-service.d.ts +9 -9
- package/client/public-api.d.ts +1 -1
- package/client/tnp-helpers.d.ts +5 -5
- package/index.d.ts +1 -1
- package/index.js +5 -5
- package/lib/angular.helper.d.ts +22 -22
- package/lib/angular.helper.js +50 -50
- package/lib/base-component.d.ts +1 -1
- package/lib/base-component.js +6 -6
- package/lib/base-formly-component.d.ts +1 -1
- package/lib/base-formly-component.js +6 -6
- package/lib/condition-wait.d.ts +8 -8
- package/lib/condition-wait.js +94 -94
- package/lib/constants.d.ts +6 -6
- package/lib/constants.js +10 -10
- package/lib/dual-component-ctrl.d.ts +1 -1
- package/lib/dual-component-ctrl.js +6 -6
- package/lib/git-project.d.ts +44 -44
- package/lib/git-project.js +195 -195
- package/lib/helpers-array-obj.d.ts +15 -15
- package/lib/helpers-array-obj.js +88 -88
- package/lib/helpers-cli-tool.backend.d.ts +45 -45
- package/lib/helpers-cli-tool.backend.js +293 -293
- package/lib/helpers-dependencies.backend.d.ts +11 -11
- package/lib/helpers-dependencies.backend.js +105 -105
- package/lib/helpers-environment.d.ts +5 -5
- package/lib/helpers-environment.js +35 -35
- package/lib/helpers-file-folders.backend.d.ts +84 -84
- package/lib/helpers-file-folders.backend.js +736 -736
- package/lib/helpers-git.backend.d.ts +35 -35
- package/lib/helpers-git.backend.js +541 -545
- package/lib/helpers-git.backend.js.map +1 -1
- package/lib/helpers-json5.backend.d.ts +16 -16
- package/lib/helpers-json5.backend.js +70 -70
- package/lib/helpers-morphi-framework.backend.d.ts +4 -4
- package/lib/helpers-morphi-framework.backend.js +49 -49
- package/lib/helpers-network.backend.d.ts +8 -8
- package/lib/helpers-network.backend.js +48 -48
- package/lib/helpers-npm.backend.d.ts +3 -3
- package/lib/helpers-npm.backend.js +17 -17
- package/lib/helpers-numbers.d.ts +3 -3
- package/lib/helpers-numbers.js +11 -11
- package/lib/helpers-path.backend.d.ts +7 -7
- package/lib/helpers-path.backend.js +36 -36
- package/lib/helpers-process.backend.d.ts +57 -57
- package/lib/helpers-process.backend.js +529 -529
- package/lib/helpers-process.backend.js.map +1 -1
- package/lib/helpers-strings-regexes.d.ts +28 -28
- package/lib/helpers-strings-regexes.js +60 -60
- package/lib/helpers-strings.d.ts +50 -50
- package/lib/helpers-strings.js +265 -265
- package/lib/helpers-system-terminal.backend.d.ts +5 -5
- package/lib/helpers-system-terminal.backend.js +53 -53
- package/lib/helpers.d.ts +67 -67
- package/lib/helpers.js +391 -391
- package/lib/index.d.ts +6 -6
- package/lib/index.js +19 -19
- package/lib/long-press.directive.d.ts +22 -22
- package/lib/long-press.directive.js +141 -141
- package/lib/merge-helpers.backend.d.ts +23 -23
- package/lib/merge-helpers.backend.js +108 -108
- package/lib/project.d.ts +94 -94
- package/lib/project.js +422 -422
- package/lib/resize-service.d.ts +6 -6
- package/lib/resize-service.js +24 -24
- package/lib/ts-code/index.d.ts +1 -1
- package/lib/ts-code/index.js +7 -7
- package/lib/ts-code/ts-code-extractor.d.ts +14 -14
- package/lib/ts-code/ts-code-extractor.js +46 -46
- package/lib/ts-code/ts-code-modifier.backend.d.ts +12 -12
- package/lib/ts-code/ts-code-modifier.backend.js +70 -70
- package/package.json +5 -5
- package/package.json_devDependencies.json +222 -222
- package/package.json_tnp.json5 +41 -41
- package/tmp-environment.json +22 -21
- package/websql/README.md +24 -24
- package/websql/esm2020/lib/base-component.mjs +47 -47
- package/websql/esm2020/lib/base-formly-component.mjs +125 -125
- package/websql/esm2020/lib/condition-wait.mjs +53 -53
- package/websql/esm2020/lib/constants.mjs +6 -6
- package/websql/esm2020/lib/dual-component-ctrl.mjs +120 -120
- package/websql/esm2020/lib/helpers-array-obj.mjs +80 -80
- package/websql/esm2020/lib/helpers-environment.mjs +30 -30
- package/websql/esm2020/lib/helpers-numbers.mjs +5 -5
- package/websql/esm2020/lib/helpers-strings-regexes.mjs +48 -48
- package/websql/esm2020/lib/helpers-strings.mjs +257 -257
- package/websql/esm2020/lib/helpers.mjs +276 -276
- package/websql/esm2020/lib/index.mjs +14 -14
- package/websql/esm2020/lib/long-press.directive.mjs +111 -111
- package/websql/esm2020/lib/project.mjs +413 -413
- package/websql/esm2020/lib/resize-service.mjs +19 -19
- package/websql/esm2020/public-api.mjs +1 -1
- package/websql/esm2020/tnp-helpers.mjs +4 -4
- package/websql/fesm2015/tnp-helpers.mjs +1535 -1535
- package/websql/fesm2020/tnp-helpers.mjs +1527 -1527
- package/websql/lib/base-component.d.ts +17 -17
- package/websql/lib/base-formly-component.d.ts +31 -31
- package/websql/lib/condition-wait.d.ts +8 -8
- package/websql/lib/constants.d.ts +6 -6
- package/websql/lib/dual-component-ctrl.d.ts +25 -25
- package/websql/lib/helpers-array-obj.d.ts +15 -15
- package/websql/lib/helpers-environment.d.ts +5 -5
- package/websql/lib/helpers-numbers.d.ts +3 -3
- package/websql/lib/helpers-strings-regexes.d.ts +28 -28
- package/websql/lib/helpers-strings.d.ts +50 -50
- package/websql/lib/helpers.d.ts +28 -28
- package/websql/lib/index.d.ts +9 -9
- package/websql/lib/long-press.directive.d.ts +25 -25
- package/websql/lib/project.d.ts +91 -91
- package/websql/lib/resize-service.d.ts +9 -9
- package/websql/public-api.d.ts +1 -1
- package/websql/tnp-helpers.d.ts +5 -5
|
@@ -1,258 +1,258 @@
|
|
|
1
|
-
import { _ } from 'tnp-core/browser';
|
|
2
|
-
import { Helpers } from './index';
|
|
3
|
-
export class HelpersStrings {
|
|
4
|
-
/**
|
|
5
|
-
* Example:
|
|
6
|
-
*
|
|
7
|
-
* const result = interpolateString("I'm {age} years old!")
|
|
8
|
-
* .withParameters({ age: 29 });
|
|
9
|
-
*
|
|
10
|
-
* const result = interpolateString("The {a} says {n}, {n}, {n}!")
|
|
11
|
-
* .withParameters({ a: 'cow', n: 'moo' });
|
|
12
|
-
*
|
|
13
|
-
*
|
|
14
|
-
* @param value string to interpolate
|
|
15
|
-
* @param parameters object with parametes
|
|
16
|
-
*/
|
|
17
|
-
interpolateString(value) {
|
|
18
|
-
if (typeof value !== 'string') {
|
|
19
|
-
Helpers.warn('[ss-logic][helper] Value for interpolation is not string: ', value);
|
|
20
|
-
return value;
|
|
21
|
-
}
|
|
22
|
-
return {
|
|
23
|
-
withParameters(parameters) {
|
|
24
|
-
if (typeof parameters !== 'object') {
|
|
25
|
-
Helpers.log(parameters);
|
|
26
|
-
Helpers.warn('[ss-logic][helper] Parameters are not a object: ');
|
|
27
|
-
return value;
|
|
28
|
-
}
|
|
29
|
-
return value.replace(/{([^{}]*)}/g, function (a, b) {
|
|
30
|
-
var r = parameters[b];
|
|
31
|
-
return typeof r === 'string' || typeof r === 'number' ? r : a;
|
|
32
|
-
});
|
|
33
|
-
}
|
|
34
|
-
};
|
|
35
|
-
}
|
|
36
|
-
numValue(pixelsCss) {
|
|
37
|
-
// tslint:disable-next-line:radix
|
|
38
|
-
return parseInt(pixelsCss.replace('px', ''));
|
|
39
|
-
}
|
|
40
|
-
/**
|
|
41
|
-
* examples:
|
|
42
|
-
* 'aa bb bb' => ['aa','bb','cc'],
|
|
43
|
-
* 'aa' => ['aa']
|
|
44
|
-
* ['aa'] => ['aa']
|
|
45
|
-
*/
|
|
46
|
-
splitIfNeed(stringOrArr) {
|
|
47
|
-
let res = [];
|
|
48
|
-
if (_.isArray(stringOrArr)) {
|
|
49
|
-
res = stringOrArr.map(s => {
|
|
50
|
-
return s.trim();
|
|
51
|
-
});
|
|
52
|
-
}
|
|
53
|
-
if (_.isString(stringOrArr)) {
|
|
54
|
-
res = stringOrArr.split(/\s*[\s,]\s*/);
|
|
55
|
-
}
|
|
56
|
-
return res.filter(f => !!f && (f.trim() !== ''));
|
|
57
|
-
}
|
|
58
|
-
removeDoubleOrMoreEmptyLines(s) {
|
|
59
|
-
s = s?.split('\n').map(f => f.trimRight()).join('\n');
|
|
60
|
-
return s?.replace(/(\r\n|\r|\n){2,}/g, '$1\n');
|
|
61
|
-
}
|
|
62
|
-
/**
|
|
63
|
-
*
|
|
64
|
-
* https://stackoverflow.com/a/57129703/1345101
|
|
65
|
-
*
|
|
66
|
-
* Returns the plural of an English word.
|
|
67
|
-
*
|
|
68
|
-
* @export
|
|
69
|
-
* @param {string} word
|
|
70
|
-
* @param {number} [amount]
|
|
71
|
-
* @returns {string}
|
|
72
|
-
*/
|
|
73
|
-
plural(word, amount) {
|
|
74
|
-
if (amount !== undefined && amount === 1) {
|
|
75
|
-
return word;
|
|
76
|
-
}
|
|
77
|
-
const plural = {
|
|
78
|
-
'(quiz)$': "$1zes",
|
|
79
|
-
'^(ox)$': "$1en",
|
|
80
|
-
'([m|l])ouse$': "$1ice",
|
|
81
|
-
'(matr|vert|ind)ix|ex$': "$1ices",
|
|
82
|
-
'(x|ch|ss|sh)$': "$1es",
|
|
83
|
-
'([^aeiouy]|qu)y$': "$1ies",
|
|
84
|
-
'(hive)$': "$1s",
|
|
85
|
-
'(?:([^f])fe|([lr])f)$': "$1$2ves",
|
|
86
|
-
'(shea|lea|loa|thie)f$': "$1ves",
|
|
87
|
-
'sis$': "ses",
|
|
88
|
-
'([ti])um$': "$1a",
|
|
89
|
-
'(tomat|potat|ech|her|vet)o$': "$1oes",
|
|
90
|
-
'(bu)s$': "$1ses",
|
|
91
|
-
'(alias)$': "$1es",
|
|
92
|
-
'(octop)us$': "$1i",
|
|
93
|
-
'(ax|test)is$': "$1es",
|
|
94
|
-
'(us)$': "$1es",
|
|
95
|
-
'([^s]+)$': "$1s"
|
|
96
|
-
};
|
|
97
|
-
const irregular = {
|
|
98
|
-
'move': 'moves',
|
|
99
|
-
'foot': 'feet',
|
|
100
|
-
'goose': 'geese',
|
|
101
|
-
'sex': 'sexes',
|
|
102
|
-
'child': 'children',
|
|
103
|
-
'man': 'men',
|
|
104
|
-
'tooth': 'teeth',
|
|
105
|
-
'person': 'people'
|
|
106
|
-
};
|
|
107
|
-
const uncountable = [
|
|
108
|
-
'sheep',
|
|
109
|
-
'fish',
|
|
110
|
-
'deer',
|
|
111
|
-
'moose',
|
|
112
|
-
'series',
|
|
113
|
-
'species',
|
|
114
|
-
'money',
|
|
115
|
-
'rice',
|
|
116
|
-
'information',
|
|
117
|
-
'equipment',
|
|
118
|
-
'bison',
|
|
119
|
-
'cod',
|
|
120
|
-
'offspring',
|
|
121
|
-
'pike',
|
|
122
|
-
'salmon',
|
|
123
|
-
'shrimp',
|
|
124
|
-
'swine',
|
|
125
|
-
'trout',
|
|
126
|
-
'aircraft',
|
|
127
|
-
'hovercraft',
|
|
128
|
-
'spacecraft',
|
|
129
|
-
'sugar',
|
|
130
|
-
'tuna',
|
|
131
|
-
'you',
|
|
132
|
-
'wood'
|
|
133
|
-
];
|
|
134
|
-
// save some time in the case that singular and plural are the same
|
|
135
|
-
if (uncountable.indexOf(word.toLowerCase()) >= 0) {
|
|
136
|
-
return word;
|
|
137
|
-
}
|
|
138
|
-
// check for irregular forms
|
|
139
|
-
for (const w in irregular) {
|
|
140
|
-
const pattern = new RegExp(`${w}$`, 'i');
|
|
141
|
-
const replace = irregular[w];
|
|
142
|
-
if (pattern.test(word)) {
|
|
143
|
-
return word.replace(pattern, replace);
|
|
144
|
-
}
|
|
145
|
-
}
|
|
146
|
-
// check for matches using regular expressions
|
|
147
|
-
for (const reg in plural) {
|
|
148
|
-
const pattern = new RegExp(reg, 'i');
|
|
149
|
-
if (pattern.test(word)) {
|
|
150
|
-
return word.replace(pattern, plural[reg]);
|
|
151
|
-
}
|
|
152
|
-
}
|
|
153
|
-
return word;
|
|
154
|
-
}
|
|
155
|
-
/**
|
|
156
|
-
* https://stackoverflow.com/a/57129703/1345101
|
|
157
|
-
*
|
|
158
|
-
* Returns the singular of an English word.
|
|
159
|
-
*
|
|
160
|
-
* @export
|
|
161
|
-
* @param {string} word
|
|
162
|
-
* @param {number} [amount]
|
|
163
|
-
* @returns {string}
|
|
164
|
-
*/
|
|
165
|
-
singular(word, amount) {
|
|
166
|
-
if (amount !== undefined && amount !== 1) {
|
|
167
|
-
return word;
|
|
168
|
-
}
|
|
169
|
-
const singular = {
|
|
170
|
-
'(quiz)zes$': "$1",
|
|
171
|
-
'(matr)ices$': "$1ix",
|
|
172
|
-
'(vert|ind)ices$': "$1ex",
|
|
173
|
-
'^(ox)en$': "$1",
|
|
174
|
-
'(alias)es$': "$1",
|
|
175
|
-
'(octop|vir)i$': "$1us",
|
|
176
|
-
'(cris|ax|test)es$': "$1is",
|
|
177
|
-
'(shoe)s$': "$1",
|
|
178
|
-
'(o)es$': "$1",
|
|
179
|
-
'(bus)es$': "$1",
|
|
180
|
-
'([m|l])ice$': "$1ouse",
|
|
181
|
-
'(x|ch|ss|sh)es$': "$1",
|
|
182
|
-
'(m)ovies$': "$1ovie",
|
|
183
|
-
'(s)eries$': "$1eries",
|
|
184
|
-
'([^aeiouy]|qu)ies$': "$1y",
|
|
185
|
-
'([lr])ves$': "$1f",
|
|
186
|
-
'(tive)s$': "$1",
|
|
187
|
-
'(hive)s$': "$1",
|
|
188
|
-
'(li|wi|kni)ves$': "$1fe",
|
|
189
|
-
'(shea|loa|lea|thie)ves$': "$1f",
|
|
190
|
-
'(^analy)ses$': "$1sis",
|
|
191
|
-
'((a)naly|(b)a|(d)iagno|(p)arenthe|(p)rogno|(s)ynop|(t)he)ses$': "$1$2sis",
|
|
192
|
-
'([ti])a$': "$1um",
|
|
193
|
-
'(n)ews$': "$1ews",
|
|
194
|
-
'(h|bl)ouses$': "$1ouse",
|
|
195
|
-
'(corpse)s$': "$1",
|
|
196
|
-
'(us)es$': "$1",
|
|
197
|
-
's$': ""
|
|
198
|
-
};
|
|
199
|
-
const irregular = {
|
|
200
|
-
'move': 'moves',
|
|
201
|
-
'foot': 'feet',
|
|
202
|
-
'goose': 'geese',
|
|
203
|
-
'sex': 'sexes',
|
|
204
|
-
'child': 'children',
|
|
205
|
-
'man': 'men',
|
|
206
|
-
'tooth': 'teeth',
|
|
207
|
-
'person': 'people'
|
|
208
|
-
};
|
|
209
|
-
const uncountable = [
|
|
210
|
-
'sheep',
|
|
211
|
-
'fish',
|
|
212
|
-
'deer',
|
|
213
|
-
'moose',
|
|
214
|
-
'series',
|
|
215
|
-
'species',
|
|
216
|
-
'money',
|
|
217
|
-
'rice',
|
|
218
|
-
'information',
|
|
219
|
-
'equipment',
|
|
220
|
-
'bison',
|
|
221
|
-
'cod',
|
|
222
|
-
'offspring',
|
|
223
|
-
'pike',
|
|
224
|
-
'salmon',
|
|
225
|
-
'shrimp',
|
|
226
|
-
'swine',
|
|
227
|
-
'trout',
|
|
228
|
-
'aircraft',
|
|
229
|
-
'hovercraft',
|
|
230
|
-
'spacecraft',
|
|
231
|
-
'sugar',
|
|
232
|
-
'tuna',
|
|
233
|
-
'you',
|
|
234
|
-
'wood'
|
|
235
|
-
];
|
|
236
|
-
// save some time in the case that singular and plural are the same
|
|
237
|
-
if (uncountable.indexOf(word.toLowerCase()) >= 0) {
|
|
238
|
-
return word;
|
|
239
|
-
}
|
|
240
|
-
// check for irregular forms
|
|
241
|
-
for (const w in irregular) {
|
|
242
|
-
const pattern = new RegExp(`${irregular[w]}$`, 'i');
|
|
243
|
-
const replace = w;
|
|
244
|
-
if (pattern.test(word)) {
|
|
245
|
-
return word.replace(pattern, replace);
|
|
246
|
-
}
|
|
247
|
-
}
|
|
248
|
-
// check for matches using regular expressions
|
|
249
|
-
for (const reg in singular) {
|
|
250
|
-
const pattern = new RegExp(reg, 'i');
|
|
251
|
-
if (pattern.test(word)) {
|
|
252
|
-
return word.replace(pattern, singular[reg]);
|
|
253
|
-
}
|
|
254
|
-
}
|
|
255
|
-
return word;
|
|
256
|
-
}
|
|
257
|
-
}
|
|
1
|
+
import { _ } from 'tnp-core/browser';
|
|
2
|
+
import { Helpers } from './index';
|
|
3
|
+
export class HelpersStrings {
|
|
4
|
+
/**
|
|
5
|
+
* Example:
|
|
6
|
+
*
|
|
7
|
+
* const result = interpolateString("I'm {age} years old!")
|
|
8
|
+
* .withParameters({ age: 29 });
|
|
9
|
+
*
|
|
10
|
+
* const result = interpolateString("The {a} says {n}, {n}, {n}!")
|
|
11
|
+
* .withParameters({ a: 'cow', n: 'moo' });
|
|
12
|
+
*
|
|
13
|
+
*
|
|
14
|
+
* @param value string to interpolate
|
|
15
|
+
* @param parameters object with parametes
|
|
16
|
+
*/
|
|
17
|
+
interpolateString(value) {
|
|
18
|
+
if (typeof value !== 'string') {
|
|
19
|
+
Helpers.warn('[ss-logic][helper] Value for interpolation is not string: ', value);
|
|
20
|
+
return value;
|
|
21
|
+
}
|
|
22
|
+
return {
|
|
23
|
+
withParameters(parameters) {
|
|
24
|
+
if (typeof parameters !== 'object') {
|
|
25
|
+
Helpers.log(parameters);
|
|
26
|
+
Helpers.warn('[ss-logic][helper] Parameters are not a object: ');
|
|
27
|
+
return value;
|
|
28
|
+
}
|
|
29
|
+
return value.replace(/{([^{}]*)}/g, function (a, b) {
|
|
30
|
+
var r = parameters[b];
|
|
31
|
+
return typeof r === 'string' || typeof r === 'number' ? r : a;
|
|
32
|
+
});
|
|
33
|
+
}
|
|
34
|
+
};
|
|
35
|
+
}
|
|
36
|
+
numValue(pixelsCss) {
|
|
37
|
+
// tslint:disable-next-line:radix
|
|
38
|
+
return parseInt(pixelsCss.replace('px', ''));
|
|
39
|
+
}
|
|
40
|
+
/**
|
|
41
|
+
* examples:
|
|
42
|
+
* 'aa bb bb' => ['aa','bb','cc'],
|
|
43
|
+
* 'aa' => ['aa']
|
|
44
|
+
* ['aa'] => ['aa']
|
|
45
|
+
*/
|
|
46
|
+
splitIfNeed(stringOrArr) {
|
|
47
|
+
let res = [];
|
|
48
|
+
if (_.isArray(stringOrArr)) {
|
|
49
|
+
res = stringOrArr.map(s => {
|
|
50
|
+
return s.trim();
|
|
51
|
+
});
|
|
52
|
+
}
|
|
53
|
+
if (_.isString(stringOrArr)) {
|
|
54
|
+
res = stringOrArr.split(/\s*[\s,]\s*/);
|
|
55
|
+
}
|
|
56
|
+
return res.filter(f => !!f && (f.trim() !== ''));
|
|
57
|
+
}
|
|
58
|
+
removeDoubleOrMoreEmptyLines(s) {
|
|
59
|
+
s = s?.split('\n').map(f => f.trimRight()).join('\n');
|
|
60
|
+
return s?.replace(/(\r\n|\r|\n){2,}/g, '$1\n');
|
|
61
|
+
}
|
|
62
|
+
/**
|
|
63
|
+
*
|
|
64
|
+
* https://stackoverflow.com/a/57129703/1345101
|
|
65
|
+
*
|
|
66
|
+
* Returns the plural of an English word.
|
|
67
|
+
*
|
|
68
|
+
* @export
|
|
69
|
+
* @param {string} word
|
|
70
|
+
* @param {number} [amount]
|
|
71
|
+
* @returns {string}
|
|
72
|
+
*/
|
|
73
|
+
plural(word, amount) {
|
|
74
|
+
if (amount !== undefined && amount === 1) {
|
|
75
|
+
return word;
|
|
76
|
+
}
|
|
77
|
+
const plural = {
|
|
78
|
+
'(quiz)$': "$1zes",
|
|
79
|
+
'^(ox)$': "$1en",
|
|
80
|
+
'([m|l])ouse$': "$1ice",
|
|
81
|
+
'(matr|vert|ind)ix|ex$': "$1ices",
|
|
82
|
+
'(x|ch|ss|sh)$': "$1es",
|
|
83
|
+
'([^aeiouy]|qu)y$': "$1ies",
|
|
84
|
+
'(hive)$': "$1s",
|
|
85
|
+
'(?:([^f])fe|([lr])f)$': "$1$2ves",
|
|
86
|
+
'(shea|lea|loa|thie)f$': "$1ves",
|
|
87
|
+
'sis$': "ses",
|
|
88
|
+
'([ti])um$': "$1a",
|
|
89
|
+
'(tomat|potat|ech|her|vet)o$': "$1oes",
|
|
90
|
+
'(bu)s$': "$1ses",
|
|
91
|
+
'(alias)$': "$1es",
|
|
92
|
+
'(octop)us$': "$1i",
|
|
93
|
+
'(ax|test)is$': "$1es",
|
|
94
|
+
'(us)$': "$1es",
|
|
95
|
+
'([^s]+)$': "$1s"
|
|
96
|
+
};
|
|
97
|
+
const irregular = {
|
|
98
|
+
'move': 'moves',
|
|
99
|
+
'foot': 'feet',
|
|
100
|
+
'goose': 'geese',
|
|
101
|
+
'sex': 'sexes',
|
|
102
|
+
'child': 'children',
|
|
103
|
+
'man': 'men',
|
|
104
|
+
'tooth': 'teeth',
|
|
105
|
+
'person': 'people'
|
|
106
|
+
};
|
|
107
|
+
const uncountable = [
|
|
108
|
+
'sheep',
|
|
109
|
+
'fish',
|
|
110
|
+
'deer',
|
|
111
|
+
'moose',
|
|
112
|
+
'series',
|
|
113
|
+
'species',
|
|
114
|
+
'money',
|
|
115
|
+
'rice',
|
|
116
|
+
'information',
|
|
117
|
+
'equipment',
|
|
118
|
+
'bison',
|
|
119
|
+
'cod',
|
|
120
|
+
'offspring',
|
|
121
|
+
'pike',
|
|
122
|
+
'salmon',
|
|
123
|
+
'shrimp',
|
|
124
|
+
'swine',
|
|
125
|
+
'trout',
|
|
126
|
+
'aircraft',
|
|
127
|
+
'hovercraft',
|
|
128
|
+
'spacecraft',
|
|
129
|
+
'sugar',
|
|
130
|
+
'tuna',
|
|
131
|
+
'you',
|
|
132
|
+
'wood'
|
|
133
|
+
];
|
|
134
|
+
// save some time in the case that singular and plural are the same
|
|
135
|
+
if (uncountable.indexOf(word.toLowerCase()) >= 0) {
|
|
136
|
+
return word;
|
|
137
|
+
}
|
|
138
|
+
// check for irregular forms
|
|
139
|
+
for (const w in irregular) {
|
|
140
|
+
const pattern = new RegExp(`${w}$`, 'i');
|
|
141
|
+
const replace = irregular[w];
|
|
142
|
+
if (pattern.test(word)) {
|
|
143
|
+
return word.replace(pattern, replace);
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
// check for matches using regular expressions
|
|
147
|
+
for (const reg in plural) {
|
|
148
|
+
const pattern = new RegExp(reg, 'i');
|
|
149
|
+
if (pattern.test(word)) {
|
|
150
|
+
return word.replace(pattern, plural[reg]);
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
return word;
|
|
154
|
+
}
|
|
155
|
+
/**
|
|
156
|
+
* https://stackoverflow.com/a/57129703/1345101
|
|
157
|
+
*
|
|
158
|
+
* Returns the singular of an English word.
|
|
159
|
+
*
|
|
160
|
+
* @export
|
|
161
|
+
* @param {string} word
|
|
162
|
+
* @param {number} [amount]
|
|
163
|
+
* @returns {string}
|
|
164
|
+
*/
|
|
165
|
+
singular(word, amount) {
|
|
166
|
+
if (amount !== undefined && amount !== 1) {
|
|
167
|
+
return word;
|
|
168
|
+
}
|
|
169
|
+
const singular = {
|
|
170
|
+
'(quiz)zes$': "$1",
|
|
171
|
+
'(matr)ices$': "$1ix",
|
|
172
|
+
'(vert|ind)ices$': "$1ex",
|
|
173
|
+
'^(ox)en$': "$1",
|
|
174
|
+
'(alias)es$': "$1",
|
|
175
|
+
'(octop|vir)i$': "$1us",
|
|
176
|
+
'(cris|ax|test)es$': "$1is",
|
|
177
|
+
'(shoe)s$': "$1",
|
|
178
|
+
'(o)es$': "$1",
|
|
179
|
+
'(bus)es$': "$1",
|
|
180
|
+
'([m|l])ice$': "$1ouse",
|
|
181
|
+
'(x|ch|ss|sh)es$': "$1",
|
|
182
|
+
'(m)ovies$': "$1ovie",
|
|
183
|
+
'(s)eries$': "$1eries",
|
|
184
|
+
'([^aeiouy]|qu)ies$': "$1y",
|
|
185
|
+
'([lr])ves$': "$1f",
|
|
186
|
+
'(tive)s$': "$1",
|
|
187
|
+
'(hive)s$': "$1",
|
|
188
|
+
'(li|wi|kni)ves$': "$1fe",
|
|
189
|
+
'(shea|loa|lea|thie)ves$': "$1f",
|
|
190
|
+
'(^analy)ses$': "$1sis",
|
|
191
|
+
'((a)naly|(b)a|(d)iagno|(p)arenthe|(p)rogno|(s)ynop|(t)he)ses$': "$1$2sis",
|
|
192
|
+
'([ti])a$': "$1um",
|
|
193
|
+
'(n)ews$': "$1ews",
|
|
194
|
+
'(h|bl)ouses$': "$1ouse",
|
|
195
|
+
'(corpse)s$': "$1",
|
|
196
|
+
'(us)es$': "$1",
|
|
197
|
+
's$': ""
|
|
198
|
+
};
|
|
199
|
+
const irregular = {
|
|
200
|
+
'move': 'moves',
|
|
201
|
+
'foot': 'feet',
|
|
202
|
+
'goose': 'geese',
|
|
203
|
+
'sex': 'sexes',
|
|
204
|
+
'child': 'children',
|
|
205
|
+
'man': 'men',
|
|
206
|
+
'tooth': 'teeth',
|
|
207
|
+
'person': 'people'
|
|
208
|
+
};
|
|
209
|
+
const uncountable = [
|
|
210
|
+
'sheep',
|
|
211
|
+
'fish',
|
|
212
|
+
'deer',
|
|
213
|
+
'moose',
|
|
214
|
+
'series',
|
|
215
|
+
'species',
|
|
216
|
+
'money',
|
|
217
|
+
'rice',
|
|
218
|
+
'information',
|
|
219
|
+
'equipment',
|
|
220
|
+
'bison',
|
|
221
|
+
'cod',
|
|
222
|
+
'offspring',
|
|
223
|
+
'pike',
|
|
224
|
+
'salmon',
|
|
225
|
+
'shrimp',
|
|
226
|
+
'swine',
|
|
227
|
+
'trout',
|
|
228
|
+
'aircraft',
|
|
229
|
+
'hovercraft',
|
|
230
|
+
'spacecraft',
|
|
231
|
+
'sugar',
|
|
232
|
+
'tuna',
|
|
233
|
+
'you',
|
|
234
|
+
'wood'
|
|
235
|
+
];
|
|
236
|
+
// save some time in the case that singular and plural are the same
|
|
237
|
+
if (uncountable.indexOf(word.toLowerCase()) >= 0) {
|
|
238
|
+
return word;
|
|
239
|
+
}
|
|
240
|
+
// check for irregular forms
|
|
241
|
+
for (const w in irregular) {
|
|
242
|
+
const pattern = new RegExp(`${irregular[w]}$`, 'i');
|
|
243
|
+
const replace = w;
|
|
244
|
+
if (pattern.test(word)) {
|
|
245
|
+
return word.replace(pattern, replace);
|
|
246
|
+
}
|
|
247
|
+
}
|
|
248
|
+
// check for matches using regular expressions
|
|
249
|
+
for (const reg in singular) {
|
|
250
|
+
const pattern = new RegExp(reg, 'i');
|
|
251
|
+
if (pattern.test(word)) {
|
|
252
|
+
return word.replace(pattern, singular[reg]);
|
|
253
|
+
}
|
|
254
|
+
}
|
|
255
|
+
return word;
|
|
256
|
+
}
|
|
257
|
+
}
|
|
258
258
|
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaGVscGVycy1zdHJpbmdzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vdG1wLWxpYnMtZm9yLWJ1bmRsZS90bnAtaGVscGVycy9wcm9qZWN0cy90bnAtaGVscGVycy9zcmMvbGliL2hlbHBlcnMtc3RyaW5ncy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsQ0FBQyxFQUFFLE1BQU0sa0JBQWtCLENBQUM7QUFDckMsT0FBTyxFQUFFLE9BQU8sRUFBRSxNQUFNLFNBQVMsQ0FBQztBQUVsQyxNQUFNLE9BQU8sY0FBYztJQUV6Qjs7Ozs7Ozs7Ozs7O09BWUc7SUFDSCxpQkFBaUIsQ0FBVSxLQUFhO1FBQ3RDLElBQUksT0FBTyxLQUFLLEtBQUssUUFBUSxFQUFFO1lBQzdCLE9BQU8sQ0FBQyxJQUFJLENBQUMsNERBQTRELEVBQUUsS0FBSyxDQUFDLENBQUM7WUFDbEYsT0FBTyxLQUFLLENBQUM7U0FDZDtRQUVELE9BQU87WUFDTCxjQUFjLENBQUMsVUFBYTtnQkFDMUIsSUFBSSxPQUFPLFVBQVUsS0FBSyxRQUFRLEVBQUU7b0JBQ2xDLE9BQU8sQ0FBQyxHQUFHLENBQUMsVUFBaUIsQ0FBQyxDQUFDO29CQUMvQixPQUFPLENBQUMsSUFBSSxDQUFDLGtEQUFrRCxDQUFDLENBQUM7b0JBQ2pFLE9BQU8sS0FBSyxDQUFDO2lCQUNkO2dCQUNELE9BQU8sS0FBSyxDQUFDLE9BQU8sQ0FBQyxhQUFhLEVBQUUsVUFBVSxDQUFDLEVBQUUsQ0FBQztvQkFDaEQsSUFBSSxDQUFDLEdBQUcsVUFBVSxDQUFDLENBQUMsQ0FBQyxDQUFDO29CQUN0QixPQUFPLE9BQU8sQ0FBQyxLQUFLLFFBQVEsSUFBSSxPQUFPLENBQUMsS0FBSyxRQUFRLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDO2dCQUNoRSxDQUFRLENBQUMsQ0FBQztZQUNaLENBQUM7U0FDRixDQUFBO0lBRUgsQ0FBQztJQUVELFFB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QUFRO1lBQ1IsT0FBTztZQUNQLE9BQU87WUFDUCxVQUFVO1lBQ1YsWUFBWTtZQUNaLFlBQVk7WUFDWixPQUFPO1lBQ1AsTUFBTTtZQUNOLEtBQUs7WUFDTCxNQUFNO1NBQ1AsQ0FBQTtRQUNELG1FQUFtRTtRQUNuRSxJQUFJLFdBQVcsQ0FBQyxPQUFPLENBQUMsSUFBSSxDQUFDLFdBQVcsRUFBRSxDQUFDLElBQUksQ0FBQyxFQUFFO1lBQ2hELE9BQU8sSUFBSSxDQUFBO1NBQ1o7UUFDRCw0QkFBNEI7UUFDNUIsS0FBSyxNQUFNLENBQUMsSUFBSSxTQUFTLEVBQUU7WUFDekIsTUFBTSxPQUFPLEdBQUcsSUFBSSxNQUFNLENBQUMsR0FBRyxTQUFTLENBQUMsQ0FBQyxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsQ0FBQTtZQUNuRCxNQUFNLE9BQU8sR0FBRyxDQUFDLENBQUE7WUFDakIsSUFBSSxPQUFPLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxFQUFFO2dCQUN0QixPQUFPLElBQUksQ0FBQyxPQUFPLENBQUMsT0FBTyxFQUFFLE9BQU8sQ0FBQyxDQUFBO2FBQ3RDO1NBQ0Y7UUFDRCw4Q0FBOEM7UUFDOUMsS0FBSyxNQUFNLEdBQUcsSUFBSSxRQUFRLEVBQUU7WUFDMUIsTUFBTSxPQUFPLEdBQUcsSUFBSSxNQUFNLENBQUMsR0FBRyxFQUFFLEdBQUcsQ0FBQyxDQUFBO1lBQ3BDLElBQUksT0FBTyxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsRUFBRTtnQkFDdEIsT0FBTyxJQUFJLENBQUMsT0FBTyxDQUFDLE9BQU8sRUFBRSxRQUFRLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQTthQUM1QztTQUNGO1FBQ0QsT0FBTyxJQUFJLENBQUE7SUFDYixDQUFDO0NBRUYiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBfIH0gZnJvbSAndG5wLWNvcmUvYnJvd3Nlcic7XG5pbXBvcnQgeyBIZWxwZXJzIH0gZnJvbSAnLi9pbmRleCc7XG5cbmV4cG9ydCBjbGFzcyBIZWxwZXJzU3RyaW5ncyB7XG5cbiAgLyoqXG4gICAqIEV4YW1wbGU6XG4gICAqXG4gICAqIGNvbnN0IHJlc3VsdCA9IGludGVycG9sYXRlU3RyaW5nKFwiSSdtIHthZ2V9IHllYXJzIG9sZCFcIilcbiAgICogLndpdGhQYXJhbWV0ZXJzKHsgYWdlOiAyOSB9KTtcbiAgICpcbiAgICogY29uc3QgcmVzdWx0ID0gaW50ZXJwb2xhdGVTdHJpbmcoXCJUaGUge2F9IHNheXMge259LCB7bn0sIHtufSFcIilcbiAgICogLndpdGhQYXJhbWV0ZXJzKHsgYTogJ2NvdycsIG46ICdtb28nIH0pO1xuICAgKlxuICAgKlxuICAgKiBAcGFyYW0gdmFsdWUgc3RyaW5nIHRvIGludGVycG9sYXRlXG4gICAqIEBwYXJhbSBwYXJhbWV0ZXJzIG9iamVjdCB3aXRoIHBhcmFtZXRlc1xuICAgKi9cbiAgaW50ZXJwb2xhdGVTdHJpbmc8VCA9IGFueT4odmFsdWU6IHN0cmluZykge1xuICAgIGlmICh0eXBlb2YgdmFsdWUgIT09ICdzdHJpbmcnKSB7XG4gICAgICBIZWxwZXJzLndhcm4oJ1tzcy1sb2dpY11baGVscGVyXSBWYWx1ZSBmb3IgaW50ZXJwb2xhdGlvbiBpcyBub3Qgc3RyaW5nOiAnLCB2YWx1ZSk7XG4gICAgICByZXR1cm4gdmFsdWU7XG4gICAgfVxuXG4gICAgcmV0dXJuIHtcbiAgICAgIHdpdGhQYXJhbWV0ZXJzKHBhcmFtZXRlcnM6IFQpIHtcbiAgICAgICAgaWYgKHR5cGVvZiBwYXJhbWV0ZXJzICE9PSAnb2JqZWN0Jykge1xuICAgICAgICAgIEhlbHBlcnMubG9nKHBhcmFtZXRlcnMgYXMgYW55KTtcbiAgICAgICAgICBIZWxwZXJzLndhcm4oJ1tzcy1sb2dpY11baGVscGVyXSBQYXJhbWV0ZXJzIGFyZSBub3QgYSBvYmplY3Q6ICcpO1xuICAgICAgICAgIHJldHVybiB2YWx1ZTtcbiAgICAgICAgfVxuICAgICAgICByZXR1cm4gdmFsdWUucmVwbGFjZSgveyhbXnt9XSopfS9nLCBmdW5jdGlvbiAoYSwgYikge1xuICAgICAgICAgIHZhciByID0gcGFyYW1ldGVyc1tiXTtcbiAgICAgICAgICByZXR1cm4gdHlwZW9mIHIgPT09ICdzdHJpbmcnIHx8IHR5cGVvZiByID09PSAnbnVtYmVyJyA/IHIgOiBhO1xuICAgICAgICB9IGFzIGFueSk7XG4gICAgICB9XG4gICAgfVxuXG4gIH1cblxuICBudW1WYWx1ZShwaXhlbHNDc3M6IHN0cmluZykge1xuICAgIC8vIHRzbGludDpkaXNhYmxlLW5leHQtbGluZTpyYWRpeFxuICAgIHJldHVybiBwYXJzZUludChwaXhlbHNDc3MucmVwbGFjZSgncHgnLCAnJykpO1xuICB9XG5cbiAgLyoqXG4gICAqIGV4YW1wbGVzOlxuICAgKiAnYWEgYmIgYmInID0+IFsnYWEnLCdiYicsJ2NjJ10sXG4gICAqICdhYScgPT4gWydhYSddXG4gICAqIFsnYWEnXSA9PiBbJ2FhJ11cbiAgICovXG4gIHNwbGl0SWZOZWVkKHN0cmluZ09yQXJyOiBzdHJpbmcgfCBzdHJpbmdbXSk6IHN0cmluZ1tdIHtcbiAgICBsZXQgcmVzID0gW107XG4gICAgaWYgKF8uaXNBcnJheShzdHJpbmdPckFycikpIHtcbiAgICAgIHJlcyA9IHN0cmluZ09yQXJyLm1hcChzID0+IHtcbiAgICAgICAgcmV0dXJuIHMudHJpbSgpO1xuICAgICAgfSlcbiAgICB9XG4gICAgaWYgKF8uaXNTdHJpbmcoc3RyaW5nT3JBcnIpKSB7XG4gICAgICByZXMgPSBzdHJpbmdPckFyci5zcGxpdCgvXFxzKltcXHMsXVxccyovKTtcbiAgICB9XG4gICAgcmV0dXJuIHJlcy5maWx0ZXIoZiA9PiAhIWYgJiYgKGYudHJpbSgpICE9PSAnJykpO1xuICB9XG5cbiAgcmVtb3ZlRG91YmxlT3JNb3JlRW1wdHlMaW5lcyhzOiBzdHJpbmcpIHtcbiAgICBzID0gcz8uc3BsaXQoJ1xcbicpLm1hcChmID0+IGYudHJpbVJpZ2h0KCkpLmpvaW4oJ1xcbicpO1xuICAgIHJldHVybiBzPy5yZXBsYWNlKC8oXFxyXFxufFxccnxcXG4pezIsfS9nLCAnJDFcXG4nKTtcbiAgfVxuXG4gIC8qKlxuICAgKlxuICAgKiBodHRwczovL3N0YWNrb3ZlcmZsb3cuY29tL2EvNTcxMjk3MDMvMTM0NTEwMVxuICAgKlxuICAgKiBSZXR1cm5zIHRoZSBwbHVyYWwgb2YgYW4gRW5nbGlzaCB3b3JkLlxuICAgKlxuICAgKiBAZXhwb3J0XG4gICAqIEBwYXJhbSB7c3RyaW5nfSB3b3JkXG4gICAqIEBwYXJhbSB7bnVtYmVyfSBbYW1vdW50XVxuICAgKiBAcmV0dXJucyB7c3RyaW5nfVxuICAgKi9cbiAgcGx1cmFsKHdvcmQ6IHN0cmluZywgYW1vdW50PzogbnVtYmVyKTogc3RyaW5nIHtcbiAgICBpZiAoYW1vdW50ICE9PSB1bmRlZmluZWQgJiYgYW1vdW50ID09PSAxKSB7XG4gICAgICByZXR1cm4gd29yZFxuICAgIH1cbiAgICBjb25zdCBwbHVyYWw6IHsgW2tleTogc3RyaW5nXTogc3RyaW5nIH0gPSB7XG4gICAgICAnKHF1aXopJCc6IFwiJDF6ZXNcIixcbiAgICAgICdeKG94KSQnOiBcIiQxZW5cIixcbiAgICAgICcoW218bF0pb3VzZSQnOiBcIiQxaWNlXCIsXG4gICAgICAnKG1hdHJ8dmVydHxpbmQpaXh8ZXgkJzogXCIkMWljZXNcIixcbiAgICAgICcoeHxjaHxzc3xzaCkkJzogXCIkMWVzXCIsXG4gICAgICAnKFteYWVpb3V5XXxxdSl5JCc6IFwiJDFpZXNcIixcbiAgICAgICcoaGl2ZSkkJzogXCIkMXNcIixcbiAgICAgICcoPzooW15mXSlmZXwoW2xyXSlmKSQnOiBcIiQxJDJ2ZXNcIixcbiAgICAgICcoc2hlYXxsZWF8bG9hfHRoaWUpZiQnOiBcIiQxdmVzXCIsXG4gICAgICAnc2lzJCc6IFwic2VzXCIsXG4gICAgICAnKFt0aV0pdW0kJzogXCIkMWFcIixcbiAgICAgICcodG9tYXR8cG90YXR8ZWNofGhlcnx2ZXQpbyQnOiBcIiQxb2VzXCIsXG4gICAgICAnKGJ1KXMkJzogXCIkMXNlc1wiLFxuICAgICAgJyhhbGlhcykkJzogXCIkMWVzXCIsXG4gICAgICAnKG9jdG9wKXVzJCc6IFwiJDFpXCIsXG4gICAgICAnKGF4fHRlc3QpaXMkJzogXCIkMWVzXCIsXG4gICAgICAnKHVzKSQnOiBcIiQxZXNcIixcbiAgICAgICcoW15zXSspJCc6IFwiJDFzXCJcbiAgICB9XG4gICAgY29uc3QgaXJyZWd1bGFyOiB7IFtrZXk6IHN0cmluZ106IHN0cmluZyB9ID0ge1xuICAgICAgJ21vdmUnOiAnbW92ZXMnLFxuICAgICAgJ2Zvb3QnOiAnZmVldCcsXG4gICAgICAnZ29vc2UnOiAnZ2Vlc2UnLFxuICAgICAgJ3NleCc6ICdzZXhlcycsXG4gICAgICAnY2hpbGQnOiAnY2hpbGRyZW4nLFxuICAgICAgJ21hbic6ICdtZW4nLFxuICAgICAgJ3Rvb3RoJzogJ3RlZXRoJyxcbiAgICAgICdwZXJzb24nOiAncGVvcGxlJ1xuICAgIH1cbiAgICBjb25zdCB1bmNvdW50YWJsZTogc3RyaW5nW10gPSBbXG4gICAgICAnc2hlZXAnLFxuICAgICAgJ2Zpc2gnLFxuICAgICAgJ2RlZXInLFxuICAgICAgJ21vb3NlJyxcbiAgICAgICdzZXJpZXMnLFxuICAgICAgJ3NwZWNpZXMnLFxuICAgICAgJ21vbmV5JyxcbiAgICAgICdyaWNlJyxcbiAgICAgICdpbmZvcm1hdGlvbicsXG4gICAgICAnZXF1aXBtZW50JyxcbiAgICAgICdiaXNvbicsXG4gICAgICAnY29kJyxcbiAgICAgICdvZmZzcHJpbmcnLFxuICAgICAgJ3Bpa2UnLFxuICAgICAgJ3NhbG1vbicsXG4gICAgICAnc2hyaW1wJyxcbiAgICAgICdzd2luZScsXG4gICAgICAndHJvdXQnLFxuICAgICAgJ2FpcmNyYWZ0JyxcbiAgICAgICdob3ZlcmNyYWZ0JyxcbiAgICAgICdzcGFjZWNyYWZ0JyxcbiAgICAgICdzdWdhcicsXG4gICAgICAndHVuYScsXG4gICAgICAneW91JyxcbiAgICAgICd3b29kJ1xuICAgIF1cbiAgICAvLyBzYXZlIHNvbWUgdGltZSBpbiB0aGUgY2FzZSB0aGF0IHNpbmd1bGFyIGFuZCBwbHVyYWwgYXJlIHRoZSBzYW1lXG4gICAgaWYgKHVuY291bnRhYmxlLmluZGV4T2Yod29yZC50b0xvd2VyQ2FzZSgpKSA+PSAwKSB7XG4gICAgICByZXR1cm4gd29yZFxuICAgIH1cbiAgICAvLyBjaGVjayBmb3IgaXJyZWd1bGFyIGZvcm1zXG4gICAgZm9yIChjb25zdCB3IGluIGlycmVndWxhcikge1xuICAgICAgY29uc3QgcGF0dGVybiA9IG5ldyBSZWdFeHAoYCR7d30kYCwgJ2knKVxuICAgICAgY29uc3QgcmVwbGFjZSA9IGlycmVndWxhclt3XVxuICAgICAgaWYgKHBhdHRlcm4udGVzdCh3b3JkKSkge1xuICAgICAgICByZXR1cm4gd29yZC5yZXBsYWNlKHBhdHRlcm4sIHJlcGxhY2UpXG4gICAgICB9XG4gICAgfVxuICAgIC8vIGNoZWNrIGZvciBtYXRjaGVzIHVzaW5nIHJlZ3VsYXIgZXhwcmVzc2lvbnNcbiAgICBmb3IgKGNvbnN0IHJlZyBpbiBwbHVyYWwpIHtcbiAgICAgIGNvbnN0IHBhdHRlcm4gPSBuZXcgUmVnRXhwKHJlZywgJ2knKVxuICAgICAgaWYgKHBhdHRlcm4udGVzdCh3b3JkKSkge1xuICAgICAgICByZXR1cm4gd29yZC5yZXBsYWNlKHBhdHRlcm4sIHBsdXJhbFtyZWddKVxuICAgICAgfVxuICAgIH1cbiAgICByZXR1cm4gd29yZFxuICB9XG5cblxuICAvKipcbiAgICogaHR0cHM6Ly9zdGFja292ZXJmbG93LmNvbS9hLzU3MTI5NzAzLzEzNDUxMDFcbiAgICpcbiAgKiBSZXR1cm5zIHRoZSBzaW5ndWxhciBvZiBhbiBFbmdsaXNoIHdvcmQuXG4gICpcbiAgKiBAZXhwb3J0XG4gICogQHBhcmFtIHtzdHJpbmd9IHdvcmRcbiAgKiBAcGFyYW0ge251bWJlcn0gW2Ftb3VudF1cbiAgKiBAcmV0dXJucyB7c3RyaW5nfVxuICAqL1xuICBzaW5ndWxhcih3b3JkOiBzdHJpbmcsIGFtb3VudD86IG51bWJlcik6IHN0cmluZyB7XG4gICAgaWYgKGFtb3VudCAhPT0gdW5kZWZpbmVkICYmIGFtb3VudCAhPT0gMSkge1xuICAgICAgcmV0dXJuIHdvcmRcbiAgICB9XG4gICAgY29uc3Qgc2luZ3VsYXI6IHsgW2tleTogc3RyaW5nXTogc3RyaW5nIH0gPSB7XG4gICAgICAnKHF1aXopemVzJCc6IFwiJDFcIixcbiAgICAgICcobWF0cilpY2VzJCc6IFwiJDFpeFwiLFxuICAgICAgJyh2ZXJ0fGluZClpY2VzJCc6IFwiJDFleFwiLFxuICAgICAgJ14ob3gpZW4kJzogXCIkMVwiLFxuICAgICAgJyhhbGlhcyllcyQnOiBcIiQxXCIsXG4gICAgICAnKG9jdG9wfHZpcilpJCc6IFwiJDF1c1wiLFxuICAgICAgJyhjcmlzfGF4fHRlc3QpZXMkJzogXCIkMWlzXCIsXG4gICAgICAnKHNob2UpcyQnOiBcIiQxXCIsXG4gICAgICAnKG8pZXMkJzogXCIkMVwiLFxuICAgICAgJyhidXMpZXMkJzogXCIkMVwiLFxuICAgICAgJyhbbXxsXSlpY2UkJzogXCIkMW91c2VcIixcbiAgICAgICcoeHxjaHxzc3xzaCllcyQnOiBcIiQxXCIsXG4gICAgICAnKG0pb3ZpZXMkJzogXCIkMW92aWVcIixcbiAgICAgICcocyllcmllcyQnOiBcIiQxZXJpZXNcIixcbiAgICAgICcoW15hZWlvdXldfHF1KWllcyQnOiBcIiQxeVwiLFxuICAgICAgJyhbbHJdKXZlcyQnOiBcIiQxZlwiLFxuICAgICAgJyh0aXZlKXMkJzogXCIkMVwiLFxuICAgICAgJyhoaXZlKXMkJzogXCIkMVwiLFxuICAgICAgJyhsaXx3aXxrbmkpdmVzJCc6IFwiJDFmZVwiLFxuICAgICAgJyhzaGVhfGxvYXxsZWF8dGhpZSl2ZXMkJzogXCIkMWZcIixcbiAgICAgICcoXmFuYWx5KXNlcyQnOiBcIiQxc2lzXCIsXG4gICAgICAnKChhKW5hbHl8KGIpYXwoZClpYWdub3wocClhcmVudGhlfChwKXJvZ25vfChzKXlub3B8KHQpaGUpc2VzJCc6IFwiJDEkMnNpc1wiLFxuICAgICAgJyhbdGldKWEkJzogXCIkMXVtXCIsXG4gICAgICAnKG4pZXdzJCc6IFwiJDFld3NcIixcbiAgICAgICcoaHxibClvdXNlcyQnOiBcIiQxb3VzZVwiLFxuICAgICAgJyhjb3Jwc2UpcyQnOiBcIiQxXCIsXG4gICAgICAnKHVzKWVzJCc6IFwiJDFcIixcbiAgICAgICdzJCc6IFwiXCJcbiAgICB9XG4gICAgY29uc3QgaXJyZWd1bGFyOiB7IFtrZXk6IHN0cmluZ106IHN0cmluZyB9ID0ge1xuICAgICAgJ21vdmUnOiAnbW92ZXMnLFxuICAgICAgJ2Zvb3QnOiAnZmVldCcsXG4gICAgICAnZ29vc2UnOiAnZ2Vlc2UnLFxuICAgICAgJ3NleCc6ICdzZXhlcycsXG4gICAgICAnY2hpbGQnOiAnY2hpbGRyZW4nLFxuICAgICAgJ21hbic6ICdtZW4nLFxuICAgICAgJ3Rvb3RoJzogJ3RlZXRoJyxcbiAgICAgICdwZXJzb24nOiAncGVvcGxlJ1xuICAgIH1cbiAgICBjb25zdCB1bmNvdW50YWJsZTogc3RyaW5nW10gPSBbXG4gICAgICAnc2hlZXAnLFxuICAgICAgJ2Zpc2gnLFxuICAgICAgJ2RlZXInLFxuICAgICAgJ21vb3NlJyxcbiAgICAgICdzZXJpZXMnLFxuICAgICAgJ3NwZWNpZXMnLFxuICAgICAgJ21vbmV5JyxcbiAgICAgICdyaWNlJyxcbiAgICAgICdpbmZvcm1hdGlvbicsXG4gICAgICAnZXF1aXBtZW50JyxcbiAgICAgICdiaXNvbicsXG4gICAgICAnY29kJyxcbiAgICAgICdvZmZzcHJpbmcnLFxuICAgICAgJ3Bpa2UnLFxuICAgICAgJ3NhbG1vbicsXG4gICAgICAnc2hyaW1wJyxcbiAgICAgICdzd2luZScsXG4gICAgICAndHJvdXQnLFxuICAgICAgJ2FpcmNyYWZ0JyxcbiAgICAgICdob3ZlcmNyYWZ0JyxcbiAgICAgICdzcGFjZWNyYWZ0JyxcbiAgICAgICdzdWdhcicsXG4gICAgICAndHVuYScsXG4gICAgICAneW91JyxcbiAgICAgICd3b29kJ1xuICAgIF1cbiAgICAvLyBzYXZlIHNvbWUgdGltZSBpbiB0aGUgY2FzZSB0aGF0IHNpbmd1bGFyIGFuZCBwbHVyYWwgYXJlIHRoZSBzYW1lXG4gICAgaWYgKHVuY291bnRhYmxlLmluZGV4T2Yod29yZC50b0xvd2VyQ2FzZSgpKSA+PSAwKSB7XG4gICAgICByZXR1cm4gd29yZFxuICAgIH1cbiAgICAvLyBjaGVjayBmb3IgaXJyZWd1bGFyIGZvcm1zXG4gICAgZm9yIChjb25zdCB3IGluIGlycmVndWxhcikge1xuICAgICAgY29uc3QgcGF0dGVybiA9IG5ldyBSZWdFeHAoYCR7aXJyZWd1bGFyW3ddfSRgLCAnaScpXG4gICAgICBjb25zdCByZXBsYWNlID0gd1xuICAgICAgaWYgKHBhdHRlcm4udGVzdCh3b3JkKSkge1xuICAgICAgICByZXR1cm4gd29yZC5yZXBsYWNlKHBhdHRlcm4sIHJlcGxhY2UpXG4gICAgICB9XG4gICAgfVxuICAgIC8vIGNoZWNrIGZvciBtYXRjaGVzIHVzaW5nIHJlZ3VsYXIgZXhwcmVzc2lvbnNcbiAgICBmb3IgKGNvbnN0IHJlZyBpbiBzaW5ndWxhcikge1xuICAgICAgY29uc3QgcGF0dGVybiA9IG5ldyBSZWdFeHAocmVnLCAnaScpXG4gICAgICBpZiAocGF0dGVybi50ZXN0KHdvcmQpKSB7XG4gICAgICAgIHJldHVybiB3b3JkLnJlcGxhY2UocGF0dGVybiwgc2luZ3VsYXJbcmVnXSlcbiAgICAgIH1cbiAgICB9XG4gICAgcmV0dXJuIHdvcmRcbiAgfVxuXG59XG5cbiJdfQ==
|