@kwiz/common 1.0.78 → 1.0.80

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.
Files changed (118) hide show
  1. package/.github/workflows/npm-publish.yml +24 -0
  2. package/.madgerc +2 -2
  3. package/LICENSE +21 -21
  4. package/fix-folder-imports.js +26 -26
  5. package/lib/cjs/helpers/sharepoint.js +5 -1
  6. package/lib/cjs/helpers/sharepoint.js.map +1 -1
  7. package/lib/cjs/helpers/typecheckers.js +5 -1
  8. package/lib/cjs/helpers/typecheckers.js.map +1 -1
  9. package/lib/cjs/types/libs/msal.types.js +26 -26
  10. package/lib/cjs/utils/sharepoint.rest/list.js +1 -1
  11. package/lib/cjs/utils/sharepoint.rest/list.js.map +1 -1
  12. package/lib/cjs/utils/sharepoint.rest/user.js +11 -11
  13. package/lib/esm/helpers/sharepoint.js +3 -0
  14. package/lib/esm/helpers/sharepoint.js.map +1 -1
  15. package/lib/esm/helpers/typecheckers.js +3 -0
  16. package/lib/esm/helpers/typecheckers.js.map +1 -1
  17. package/lib/esm/types/libs/msal.types.js +26 -26
  18. package/lib/esm/utils/sharepoint.rest/list.js +2 -2
  19. package/lib/esm/utils/sharepoint.rest/list.js.map +1 -1
  20. package/lib/esm/utils/sharepoint.rest/user.js +11 -11
  21. package/lib/types/helpers/sharepoint.d.ts +1 -0
  22. package/lib/types/helpers/typecheckers.d.ts +1 -0
  23. package/package.json +77 -77
  24. package/readme.md +17 -17
  25. package/src/_dependencies.ts +12 -12
  26. package/src/config.ts +17 -17
  27. package/src/helpers/Guid.ts +181 -181
  28. package/src/helpers/base64.ts +173 -173
  29. package/src/helpers/browser.test.js +13 -13
  30. package/src/helpers/browser.ts +1348 -1348
  31. package/src/helpers/browserinfo.ts +292 -292
  32. package/src/helpers/collections.base.test.js +25 -25
  33. package/src/helpers/collections.base.ts +437 -437
  34. package/src/helpers/collections.ts +107 -107
  35. package/src/helpers/color.ts +54 -54
  36. package/src/helpers/cookies.ts +59 -59
  37. package/src/helpers/date.test.js +119 -119
  38. package/src/helpers/date.ts +188 -188
  39. package/src/helpers/debug.ts +186 -186
  40. package/src/helpers/emails.ts +6 -6
  41. package/src/helpers/eval.ts +5 -5
  42. package/src/helpers/file.test.js +50 -50
  43. package/src/helpers/file.ts +58 -58
  44. package/src/helpers/flatted.ts +149 -149
  45. package/src/helpers/functions.ts +16 -16
  46. package/src/helpers/graph/calendar.types.ts +10 -10
  47. package/src/helpers/http.ts +69 -69
  48. package/src/helpers/images.ts +22 -22
  49. package/src/helpers/json.ts +38 -38
  50. package/src/helpers/md5.ts +189 -189
  51. package/src/helpers/objects.test.js +33 -33
  52. package/src/helpers/objects.ts +270 -270
  53. package/src/helpers/promises.test.js +37 -37
  54. package/src/helpers/promises.ts +165 -165
  55. package/src/helpers/random.ts +27 -27
  56. package/src/helpers/scheduler/scheduler.test.js +103 -103
  57. package/src/helpers/scheduler/scheduler.ts +131 -131
  58. package/src/helpers/sharepoint.ts +776 -772
  59. package/src/helpers/strings.test.js +101 -101
  60. package/src/helpers/strings.ts +317 -317
  61. package/src/helpers/typecheckers.test.js +34 -34
  62. package/src/helpers/typecheckers.ts +266 -262
  63. package/src/helpers/url.test.js +43 -43
  64. package/src/helpers/url.ts +207 -207
  65. package/src/helpers/urlhelper.ts +111 -111
  66. package/src/index.ts +6 -6
  67. package/src/types/auth.ts +54 -54
  68. package/src/types/common.types.ts +15 -15
  69. package/src/types/flatted.types.ts +59 -59
  70. package/src/types/globals.types.ts +6 -6
  71. package/src/types/graph/calendar.types.ts +80 -80
  72. package/src/types/knownscript.types.ts +18 -18
  73. package/src/types/libs/datajs.types.ts +28 -28
  74. package/src/types/libs/ics.types.ts +30 -30
  75. package/src/types/libs/msal.types.ts +49 -49
  76. package/src/types/locales.ts +124 -124
  77. package/src/types/localstoragecache.types.ts +8 -8
  78. package/src/types/location.types.ts +27 -27
  79. package/src/types/moment.ts +11 -11
  80. package/src/types/regex.types.ts +16 -16
  81. package/src/types/rest.types.ts +95 -95
  82. package/src/types/sharepoint.types.ts +1465 -1465
  83. package/src/types/sharepoint.utils.types.ts +287 -287
  84. package/src/utils/auth/common.ts +74 -74
  85. package/src/utils/auth/discovery.test.js +12 -12
  86. package/src/utils/auth/discovery.ts +132 -132
  87. package/src/utils/base64.ts +27 -27
  88. package/src/utils/consolelogger.ts +320 -320
  89. package/src/utils/date.ts +35 -35
  90. package/src/utils/emails.ts +24 -24
  91. package/src/utils/knownscript.ts +286 -286
  92. package/src/utils/localstoragecache.ts +441 -441
  93. package/src/utils/rest.ts +501 -501
  94. package/src/utils/script.ts +170 -170
  95. package/src/utils/sharepoint.rest/common.ts +154 -154
  96. package/src/utils/sharepoint.rest/date.ts +62 -62
  97. package/src/utils/sharepoint.rest/file.folder.ts +598 -598
  98. package/src/utils/sharepoint.rest/item.ts +547 -547
  99. package/src/utils/sharepoint.rest/list.ts +1388 -1388
  100. package/src/utils/sharepoint.rest/listutils/GetListItemsByCaml.ts +774 -774
  101. package/src/utils/sharepoint.rest/listutils/GetListItemsById.ts +275 -275
  102. package/src/utils/sharepoint.rest/listutils/common.ts +206 -206
  103. package/src/utils/sharepoint.rest/location.ts +141 -141
  104. package/src/utils/sharepoint.rest/navigation-links.ts +86 -86
  105. package/src/utils/sharepoint.rest/user-search.ts +252 -252
  106. package/src/utils/sharepoint.rest/user.ts +491 -491
  107. package/src/utils/sharepoint.rest/web.ts +1384 -1384
  108. package/src/utils/sod.ts +194 -194
  109. package/lib/cjs/helpers/_dependencies.js +0 -21
  110. package/lib/cjs/helpers/_dependencies.js.map +0 -1
  111. package/lib/cjs/utils/_dependencies.js +0 -24
  112. package/lib/cjs/utils/_dependencies.js.map +0 -1
  113. package/lib/esm/helpers/_dependencies.js +0 -3
  114. package/lib/esm/helpers/_dependencies.js.map +0 -1
  115. package/lib/esm/utils/_dependencies.js +0 -4
  116. package/lib/esm/utils/_dependencies.js.map +0 -1
  117. package/lib/types/helpers/_dependencies.d.ts +0 -2
  118. package/lib/types/utils/_dependencies.d.ts +0 -3
@@ -1,321 +1,321 @@
1
- import { BuildNumber, ReleaseStatus } from "../_dependencies";
2
- import { getSecondsElapsed } from "../helpers/date";
3
- import { consoleLoggerFilter, isDebug } from "../helpers/debug";
4
- import { getGlobal } from "../helpers/objects";
5
- import { padLeft, padRight } from "../helpers/strings";
6
- import { isFunction, isNullOrEmptyString, isNumber, isNumeric, isString } from "../helpers/typecheckers";
7
- import { IDictionary } from "../types/common.types";
8
-
9
- const DEFAULT_LOGGER_NAME = "DEFAULT";
10
- const LoggerPrefix = "[kw]";
11
-
12
- interface ILoggerGlobal {
13
- loggers: IDictionary<ConsoleLogger>;
14
- loggedBuild: boolean;
15
- }
16
-
17
- // eslint-disable-next-line no-shadow
18
- export enum LoggerLevel {
19
- VERBOSE = 0,
20
- DEBUG = 1,
21
- INFO = 2,
22
- LOG = 3,
23
- /** shows when debug=off */
24
- WARN = 4,
25
- /** shows when debug=off */
26
- TRACE = 5,
27
- /** shows when debug=off */
28
- ERROR = 6,
29
- OFF = 10
30
- }
31
-
32
- export type LoggerContext = {
33
- //allow to not set logger level, so it will be set by isDebug always. But - dev can set it for a specific logger instance if they want to override
34
- filterLevel?: LoggerLevel;
35
- name?: string;
36
- //allow to have a different prefix
37
- prefix?: string;
38
- };
39
-
40
- export class ConsoleLogger {
41
- public context: LoggerContext;
42
-
43
- protected constructor(context: LoggerContext) {
44
- this.context = context;
45
- }
46
-
47
- public static get(name: string, prefix?: string) {
48
- var global = ConsoleLogger._getGlobal();
49
- var loggers = global.loggers;
50
-
51
- if (!global.loggedBuild) {
52
- global.loggedBuild = true;
53
- console.debug(`${ConsoleLogger.commonPrefix()} KWIZ build ${ReleaseStatus}.${BuildNumber}`);
54
- }
55
-
56
- return loggers[name] || (loggers[name] = new ConsoleLogger({ name: name, prefix:prefix }));
57
- }
58
-
59
- private static _getGlobal() {
60
- var global: ILoggerGlobal = getGlobal("loggers", {
61
- loggedBuild: false,
62
- loggers: {}
63
- }, true);
64
- return global;
65
- }
66
-
67
- private static _getDefaultLogger() {
68
- return ConsoleLogger.get(DEFAULT_LOGGER_NAME);
69
- }
70
-
71
- public static setLevel(newLevel: LoggerLevel) {
72
- ConsoleLogger._getDefaultLogger().setLevel(newLevel);
73
- }
74
-
75
- public static getLevel() {
76
- return ConsoleLogger._getDefaultLogger().getLevel();
77
- }
78
-
79
- public static debug(message: string) {
80
- ConsoleLogger._getDefaultLogger().debug(message);
81
- }
82
-
83
- public static info(message: string) {
84
- ConsoleLogger._getDefaultLogger().info(message);
85
- }
86
-
87
- public static log(message: string) {
88
- ConsoleLogger._getDefaultLogger().log(message);
89
- }
90
-
91
- public static warn(message: string) {
92
- ConsoleLogger._getDefaultLogger().warn(message);
93
- }
94
-
95
- public static error(message: string) {
96
- ConsoleLogger._getDefaultLogger().error(message);
97
- }
98
-
99
- public static trace(message: string) {
100
- ConsoleLogger._getDefaultLogger().trace(message);
101
- }
102
-
103
- public static commonPrefix(prefix?: string) {
104
- var d = new Date();
105
- var timestamp = padLeft(d.getHours().toString(), 2, "0")
106
- + ":" + padLeft(d.getMinutes().toString(), 2, "0")
107
- + ":" + padLeft(d.getSeconds().toString(), 2, "0")
108
- + "." + padRight(d.getMilliseconds().toString(), 3, "0");
109
-
110
- return `[${timestamp}] ${prefix || LoggerPrefix}`;
111
- }
112
- private contextPrefix() {
113
- return `${ConsoleLogger.commonPrefix(this.context.prefix)} [${this.context.name}]`;
114
- }
115
- private format(message: string) {
116
- return `${this.contextPrefix()} ${message}`;
117
- }
118
-
119
- public setLevel(newLevel: LoggerLevel) {
120
- if (isNumeric(newLevel)) {
121
- this.context.filterLevel = newLevel;
122
- }
123
- }
124
-
125
- public getLevel() {
126
- if (isNumeric(this.context.filterLevel))
127
- return this.context.filterLevel;
128
- else return isDebug() ? LoggerLevel.VERBOSE : LoggerLevel.WARN;
129
- }
130
-
131
- public enabledFor(level: LoggerLevel) {
132
- if (consoleLoggerFilter().indexOf(this.context.name) >= 0) return false;
133
- var filterLevel = this.getLevel();
134
- return level >= filterLevel;
135
- }
136
-
137
- public debug(message: any) {
138
- this.logWithLevel(LoggerLevel.DEBUG, message);
139
- }
140
-
141
- public info(message: string) {
142
- this.logWithLevel(LoggerLevel.INFO, message);
143
- }
144
-
145
- public log(message: string) {
146
- this.logWithLevel(LoggerLevel.LOG, message);
147
- }
148
-
149
- /** output a message when debug is off */
150
- public warn(message: string) {
151
- this.logWithLevel(LoggerLevel.WARN, message);
152
- }
153
- /** output a message when debug is off */
154
- public error(message: string) {
155
- this.logWithLevel(LoggerLevel.ERROR, message);
156
- }
157
- /** output a message when debug is off */
158
- public trace(message: string) {
159
- this.logWithLevel(LoggerLevel.TRACE, message);
160
- }
161
-
162
- /**start timer on a label, call timeEnd with the same label to print out the time that passed */
163
- public time(label: string) {
164
- if (this.enabledFor(LoggerLevel.DEBUG) && isFunction(console.time))
165
- console.time(`[timer] [kw] [${this.context.name}] ${label}`);
166
- }
167
- /**start timer on a label, call timeEnd with the same label to print out the time that passed */
168
- public timeEnd(label: string) {
169
- if (this.enabledFor(LoggerLevel.DEBUG) && isFunction(console.timeEnd))
170
- console.timeEnd(`[timer] [kw] [${this.context.name}] ${label}`);
171
- }
172
- /**prints an array or dictionary to the console inside a group */
173
- public table(data: any, groupLabel?: string, groupCollapsed?: boolean) {
174
- if (this.enabledFor(LoggerLevel.DEBUG) && isFunction(console.table)) {
175
- this.group(() => console.table(data), groupLabel, groupCollapsed);
176
- }
177
- }
178
- /**prints a JSON object to the console inside a group */
179
- public json(data: any, groupLabel?: string, groupCollapsed?: boolean) {
180
- if (this.enabledFor(LoggerLevel.DEBUG) && isFunction(console.dir)) {
181
- this.group(() => console.dir(data), groupLabel, groupCollapsed);
182
- }
183
- }
184
- /**prints an XML object to the console inside a group. If data is string that looks like an XML - will try to parse it. */
185
- public xml(data: any, groupLabel?: string, groupCollapsed?: boolean) {
186
- if (this.enabledFor(LoggerLevel.DEBUG) && isFunction(console.dirxml)) {
187
- this.group(() => {
188
- if (isString(data) && data.startsWith('<')) {
189
- try {
190
- //maybe this string is an html element?
191
- data = new DOMParser().parseFromString(data, "text/html");
192
- } catch (e) { }
193
- }
194
- console.dirxml(data);
195
- }, groupLabel, groupCollapsed);
196
- }
197
- }
198
- /** render messages inside a group, and closes the group when done. if a label is not provided - a group will not be rendered */
199
- public group(renderContent: () => void, label?: string, collapsed?: boolean) {
200
- let hadGroup = false;
201
- if (this.enabledFor(LoggerLevel.DEBUG) && isFunction(console.group) && !isNullOrEmptyString(label)) {
202
- if (collapsed) {
203
- console.groupCollapsed(`${this.contextPrefix()} ${label}`);
204
- } else {
205
- console.group(`${this.contextPrefix()} ${label}`);
206
- }
207
- hadGroup = true;
208
- }
209
-
210
- if (hadGroup) this.time(label);
211
- //we must run render content even if no groups - since this might hold other code the caller needs to run
212
- renderContent();
213
- if (hadGroup) {
214
- this.timeEnd(label);
215
- console.groupEnd();
216
- }
217
- }
218
-
219
- public groupSync<ReturnType>(label: string, renderContent: (log: (message: string) => void) => ReturnType, options?: {
220
- expand?: boolean;
221
- /** do not write to log */
222
- supress?: boolean;
223
- }) {
224
- if (isNullOrEmptyString(label)) label = "SyncGroup";
225
- let { logMessages, start, lastMessage } = this.$startGroup();
226
-
227
- let result: ReturnType;
228
- try {
229
- result = renderContent(message => {
230
- logMessages.push({ message: message, seconds: getSecondsElapsed(lastMessage) });
231
- lastMessage = new Date();
232
- });
233
- } catch (e) {
234
- logMessages.push({ message: `Unhandled exception: ${e}`, seconds: getSecondsElapsed(lastMessage) });
235
- throw this.$finishGroup(label, e, start, logMessages, options);
236
- }
237
-
238
- return this.$finishGroup(label, result, start, logMessages, options);
239
- }
240
- public async groupAsync<ReturnType>(label: string, renderContent: (log: (message: string) => void) => Promise<ReturnType>, options?: {
241
- expand?: boolean;
242
- /** do not write to log */
243
- supress?: boolean;
244
- }) {
245
- if (isNullOrEmptyString(label)) label = "AsyncGroup";
246
- let { logMessages, start, lastMessage } = this.$startGroup();
247
-
248
- let result: ReturnType;
249
- try {
250
- result = await renderContent(message => {
251
- logMessages.push({ message: message, seconds: getSecondsElapsed(lastMessage) });
252
- lastMessage = new Date();
253
- });
254
- } catch (e) {
255
- logMessages.push({ message: `Unhandled exception: ${e}`, seconds: getSecondsElapsed(lastMessage) });
256
- throw this.$finishGroup(label, e, start, logMessages, options);
257
- }
258
-
259
- return this.$finishGroup(label, result, start, logMessages, options);
260
- }
261
-
262
- private $startGroup() {
263
- let logMessages: { seconds: number, message: string }[] = [];
264
-
265
- let start = new Date();
266
- let lastMessage = start;
267
- return { logMessages, start, lastMessage };
268
- }
269
- private $finishGroup<ReturnType>(label: string, result: ReturnType, start: Date, logMessages: { seconds: number, message: string }[], options?: {
270
- expand?: boolean;
271
- /** do not write to log */
272
- supress?: boolean;
273
- }) {
274
- if (options && options.supress) return result;
275
- label = `${label} (${getSecondsElapsed(start)}s)`;
276
-
277
- if (this.enabledFor(LoggerLevel.DEBUG) && isFunction(console.group) && !isNullOrEmptyString(label)) {
278
- if (options && options.expand) {
279
- console.group(`${this.contextPrefix()} ${label}`);
280
- } else {
281
- console.groupCollapsed(`${this.contextPrefix()} ${label}`);
282
- }
283
- }
284
- else return result;
285
-
286
- //drop directly, without a prefix, in the group
287
- logMessages.forEach(m => console.debug(`(${m.seconds}s) ${m.message}`));
288
- console.groupEnd();
289
- return result;
290
- }
291
-
292
- private logWithLevel(level, message) {
293
- try {
294
- if (this.enabledFor(level)) {
295
- var isSimpleObject = isString(message) || isNumber(message);
296
- var logMessage = this.format(isSimpleObject ? "%s" : "%o");
297
- switch (level) {
298
- case LoggerLevel.DEBUG:
299
- console.debug(logMessage, message);
300
- break;
301
- case LoggerLevel.ERROR:
302
- console.error(logMessage, message);
303
- break;
304
- case LoggerLevel.WARN:
305
- console.warn(logMessage, message);
306
- break;
307
- case LoggerLevel.INFO:
308
- console.info(logMessage, message);
309
- break;
310
- case LoggerLevel.TRACE:
311
- console.trace(logMessage, message);
312
- break;
313
- default:
314
- console.log(logMessage, message);
315
- }
316
- }
317
- } catch (ex) {
318
- //empty
319
- }
320
- }
1
+ import { BuildNumber, ReleaseStatus } from "../_dependencies";
2
+ import { getSecondsElapsed } from "../helpers/date";
3
+ import { consoleLoggerFilter, isDebug } from "../helpers/debug";
4
+ import { getGlobal } from "../helpers/objects";
5
+ import { padLeft, padRight } from "../helpers/strings";
6
+ import { isFunction, isNullOrEmptyString, isNumber, isNumeric, isString } from "../helpers/typecheckers";
7
+ import { IDictionary } from "../types/common.types";
8
+
9
+ const DEFAULT_LOGGER_NAME = "DEFAULT";
10
+ const LoggerPrefix = "[kw]";
11
+
12
+ interface ILoggerGlobal {
13
+ loggers: IDictionary<ConsoleLogger>;
14
+ loggedBuild: boolean;
15
+ }
16
+
17
+ // eslint-disable-next-line no-shadow
18
+ export enum LoggerLevel {
19
+ VERBOSE = 0,
20
+ DEBUG = 1,
21
+ INFO = 2,
22
+ LOG = 3,
23
+ /** shows when debug=off */
24
+ WARN = 4,
25
+ /** shows when debug=off */
26
+ TRACE = 5,
27
+ /** shows when debug=off */
28
+ ERROR = 6,
29
+ OFF = 10
30
+ }
31
+
32
+ export type LoggerContext = {
33
+ //allow to not set logger level, so it will be set by isDebug always. But - dev can set it for a specific logger instance if they want to override
34
+ filterLevel?: LoggerLevel;
35
+ name?: string;
36
+ //allow to have a different prefix
37
+ prefix?: string;
38
+ };
39
+
40
+ export class ConsoleLogger {
41
+ public context: LoggerContext;
42
+
43
+ protected constructor(context: LoggerContext) {
44
+ this.context = context;
45
+ }
46
+
47
+ public static get(name: string, prefix?: string) {
48
+ var global = ConsoleLogger._getGlobal();
49
+ var loggers = global.loggers;
50
+
51
+ if (!global.loggedBuild) {
52
+ global.loggedBuild = true;
53
+ console.debug(`${ConsoleLogger.commonPrefix()} KWIZ build ${ReleaseStatus}.${BuildNumber}`);
54
+ }
55
+
56
+ return loggers[name] || (loggers[name] = new ConsoleLogger({ name: name, prefix:prefix }));
57
+ }
58
+
59
+ private static _getGlobal() {
60
+ var global: ILoggerGlobal = getGlobal("loggers", {
61
+ loggedBuild: false,
62
+ loggers: {}
63
+ }, true);
64
+ return global;
65
+ }
66
+
67
+ private static _getDefaultLogger() {
68
+ return ConsoleLogger.get(DEFAULT_LOGGER_NAME);
69
+ }
70
+
71
+ public static setLevel(newLevel: LoggerLevel) {
72
+ ConsoleLogger._getDefaultLogger().setLevel(newLevel);
73
+ }
74
+
75
+ public static getLevel() {
76
+ return ConsoleLogger._getDefaultLogger().getLevel();
77
+ }
78
+
79
+ public static debug(message: string) {
80
+ ConsoleLogger._getDefaultLogger().debug(message);
81
+ }
82
+
83
+ public static info(message: string) {
84
+ ConsoleLogger._getDefaultLogger().info(message);
85
+ }
86
+
87
+ public static log(message: string) {
88
+ ConsoleLogger._getDefaultLogger().log(message);
89
+ }
90
+
91
+ public static warn(message: string) {
92
+ ConsoleLogger._getDefaultLogger().warn(message);
93
+ }
94
+
95
+ public static error(message: string) {
96
+ ConsoleLogger._getDefaultLogger().error(message);
97
+ }
98
+
99
+ public static trace(message: string) {
100
+ ConsoleLogger._getDefaultLogger().trace(message);
101
+ }
102
+
103
+ public static commonPrefix(prefix?: string) {
104
+ var d = new Date();
105
+ var timestamp = padLeft(d.getHours().toString(), 2, "0")
106
+ + ":" + padLeft(d.getMinutes().toString(), 2, "0")
107
+ + ":" + padLeft(d.getSeconds().toString(), 2, "0")
108
+ + "." + padRight(d.getMilliseconds().toString(), 3, "0");
109
+
110
+ return `[${timestamp}] ${prefix || LoggerPrefix}`;
111
+ }
112
+ private contextPrefix() {
113
+ return `${ConsoleLogger.commonPrefix(this.context.prefix)} [${this.context.name}]`;
114
+ }
115
+ private format(message: string) {
116
+ return `${this.contextPrefix()} ${message}`;
117
+ }
118
+
119
+ public setLevel(newLevel: LoggerLevel) {
120
+ if (isNumeric(newLevel)) {
121
+ this.context.filterLevel = newLevel;
122
+ }
123
+ }
124
+
125
+ public getLevel() {
126
+ if (isNumeric(this.context.filterLevel))
127
+ return this.context.filterLevel;
128
+ else return isDebug() ? LoggerLevel.VERBOSE : LoggerLevel.WARN;
129
+ }
130
+
131
+ public enabledFor(level: LoggerLevel) {
132
+ if (consoleLoggerFilter().indexOf(this.context.name) >= 0) return false;
133
+ var filterLevel = this.getLevel();
134
+ return level >= filterLevel;
135
+ }
136
+
137
+ public debug(message: any) {
138
+ this.logWithLevel(LoggerLevel.DEBUG, message);
139
+ }
140
+
141
+ public info(message: string) {
142
+ this.logWithLevel(LoggerLevel.INFO, message);
143
+ }
144
+
145
+ public log(message: string) {
146
+ this.logWithLevel(LoggerLevel.LOG, message);
147
+ }
148
+
149
+ /** output a message when debug is off */
150
+ public warn(message: string) {
151
+ this.logWithLevel(LoggerLevel.WARN, message);
152
+ }
153
+ /** output a message when debug is off */
154
+ public error(message: string) {
155
+ this.logWithLevel(LoggerLevel.ERROR, message);
156
+ }
157
+ /** output a message when debug is off */
158
+ public trace(message: string) {
159
+ this.logWithLevel(LoggerLevel.TRACE, message);
160
+ }
161
+
162
+ /**start timer on a label, call timeEnd with the same label to print out the time that passed */
163
+ public time(label: string) {
164
+ if (this.enabledFor(LoggerLevel.DEBUG) && isFunction(console.time))
165
+ console.time(`[timer] [kw] [${this.context.name}] ${label}`);
166
+ }
167
+ /**start timer on a label, call timeEnd with the same label to print out the time that passed */
168
+ public timeEnd(label: string) {
169
+ if (this.enabledFor(LoggerLevel.DEBUG) && isFunction(console.timeEnd))
170
+ console.timeEnd(`[timer] [kw] [${this.context.name}] ${label}`);
171
+ }
172
+ /**prints an array or dictionary to the console inside a group */
173
+ public table(data: any, groupLabel?: string, groupCollapsed?: boolean) {
174
+ if (this.enabledFor(LoggerLevel.DEBUG) && isFunction(console.table)) {
175
+ this.group(() => console.table(data), groupLabel, groupCollapsed);
176
+ }
177
+ }
178
+ /**prints a JSON object to the console inside a group */
179
+ public json(data: any, groupLabel?: string, groupCollapsed?: boolean) {
180
+ if (this.enabledFor(LoggerLevel.DEBUG) && isFunction(console.dir)) {
181
+ this.group(() => console.dir(data), groupLabel, groupCollapsed);
182
+ }
183
+ }
184
+ /**prints an XML object to the console inside a group. If data is string that looks like an XML - will try to parse it. */
185
+ public xml(data: any, groupLabel?: string, groupCollapsed?: boolean) {
186
+ if (this.enabledFor(LoggerLevel.DEBUG) && isFunction(console.dirxml)) {
187
+ this.group(() => {
188
+ if (isString(data) && data.startsWith('<')) {
189
+ try {
190
+ //maybe this string is an html element?
191
+ data = new DOMParser().parseFromString(data, "text/html");
192
+ } catch (e) { }
193
+ }
194
+ console.dirxml(data);
195
+ }, groupLabel, groupCollapsed);
196
+ }
197
+ }
198
+ /** render messages inside a group, and closes the group when done. if a label is not provided - a group will not be rendered */
199
+ public group(renderContent: () => void, label?: string, collapsed?: boolean) {
200
+ let hadGroup = false;
201
+ if (this.enabledFor(LoggerLevel.DEBUG) && isFunction(console.group) && !isNullOrEmptyString(label)) {
202
+ if (collapsed) {
203
+ console.groupCollapsed(`${this.contextPrefix()} ${label}`);
204
+ } else {
205
+ console.group(`${this.contextPrefix()} ${label}`);
206
+ }
207
+ hadGroup = true;
208
+ }
209
+
210
+ if (hadGroup) this.time(label);
211
+ //we must run render content even if no groups - since this might hold other code the caller needs to run
212
+ renderContent();
213
+ if (hadGroup) {
214
+ this.timeEnd(label);
215
+ console.groupEnd();
216
+ }
217
+ }
218
+
219
+ public groupSync<ReturnType>(label: string, renderContent: (log: (message: string) => void) => ReturnType, options?: {
220
+ expand?: boolean;
221
+ /** do not write to log */
222
+ supress?: boolean;
223
+ }) {
224
+ if (isNullOrEmptyString(label)) label = "SyncGroup";
225
+ let { logMessages, start, lastMessage } = this.$startGroup();
226
+
227
+ let result: ReturnType;
228
+ try {
229
+ result = renderContent(message => {
230
+ logMessages.push({ message: message, seconds: getSecondsElapsed(lastMessage) });
231
+ lastMessage = new Date();
232
+ });
233
+ } catch (e) {
234
+ logMessages.push({ message: `Unhandled exception: ${e}`, seconds: getSecondsElapsed(lastMessage) });
235
+ throw this.$finishGroup(label, e, start, logMessages, options);
236
+ }
237
+
238
+ return this.$finishGroup(label, result, start, logMessages, options);
239
+ }
240
+ public async groupAsync<ReturnType>(label: string, renderContent: (log: (message: string) => void) => Promise<ReturnType>, options?: {
241
+ expand?: boolean;
242
+ /** do not write to log */
243
+ supress?: boolean;
244
+ }) {
245
+ if (isNullOrEmptyString(label)) label = "AsyncGroup";
246
+ let { logMessages, start, lastMessage } = this.$startGroup();
247
+
248
+ let result: ReturnType;
249
+ try {
250
+ result = await renderContent(message => {
251
+ logMessages.push({ message: message, seconds: getSecondsElapsed(lastMessage) });
252
+ lastMessage = new Date();
253
+ });
254
+ } catch (e) {
255
+ logMessages.push({ message: `Unhandled exception: ${e}`, seconds: getSecondsElapsed(lastMessage) });
256
+ throw this.$finishGroup(label, e, start, logMessages, options);
257
+ }
258
+
259
+ return this.$finishGroup(label, result, start, logMessages, options);
260
+ }
261
+
262
+ private $startGroup() {
263
+ let logMessages: { seconds: number, message: string }[] = [];
264
+
265
+ let start = new Date();
266
+ let lastMessage = start;
267
+ return { logMessages, start, lastMessage };
268
+ }
269
+ private $finishGroup<ReturnType>(label: string, result: ReturnType, start: Date, logMessages: { seconds: number, message: string }[], options?: {
270
+ expand?: boolean;
271
+ /** do not write to log */
272
+ supress?: boolean;
273
+ }) {
274
+ if (options && options.supress) return result;
275
+ label = `${label} (${getSecondsElapsed(start)}s)`;
276
+
277
+ if (this.enabledFor(LoggerLevel.DEBUG) && isFunction(console.group) && !isNullOrEmptyString(label)) {
278
+ if (options && options.expand) {
279
+ console.group(`${this.contextPrefix()} ${label}`);
280
+ } else {
281
+ console.groupCollapsed(`${this.contextPrefix()} ${label}`);
282
+ }
283
+ }
284
+ else return result;
285
+
286
+ //drop directly, without a prefix, in the group
287
+ logMessages.forEach(m => console.debug(`(${m.seconds}s) ${m.message}`));
288
+ console.groupEnd();
289
+ return result;
290
+ }
291
+
292
+ private logWithLevel(level, message) {
293
+ try {
294
+ if (this.enabledFor(level)) {
295
+ var isSimpleObject = isString(message) || isNumber(message);
296
+ var logMessage = this.format(isSimpleObject ? "%s" : "%o");
297
+ switch (level) {
298
+ case LoggerLevel.DEBUG:
299
+ console.debug(logMessage, message);
300
+ break;
301
+ case LoggerLevel.ERROR:
302
+ console.error(logMessage, message);
303
+ break;
304
+ case LoggerLevel.WARN:
305
+ console.warn(logMessage, message);
306
+ break;
307
+ case LoggerLevel.INFO:
308
+ console.info(logMessage, message);
309
+ break;
310
+ case LoggerLevel.TRACE:
311
+ console.trace(logMessage, message);
312
+ break;
313
+ default:
314
+ console.log(logMessage, message);
315
+ }
316
+ }
317
+ } catch (ex) {
318
+ //empty
319
+ }
320
+ }
321
321
  }