@fluidframework/telemetry-utils 2.0.0-dev.2.3.0.115467 → 2.0.0-dev.3.1.0.125672

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 (103) hide show
  1. package/.eslintrc.js +11 -13
  2. package/.mocharc.js +2 -2
  3. package/api-extractor.json +2 -2
  4. package/dist/config.d.ts.map +1 -1
  5. package/dist/config.js +3 -3
  6. package/dist/config.js.map +1 -1
  7. package/dist/debugLogger.d.ts.map +1 -1
  8. package/dist/debugLogger.js.map +1 -1
  9. package/dist/errorLogging.d.ts +8 -7
  10. package/dist/errorLogging.d.ts.map +1 -1
  11. package/dist/errorLogging.js +23 -20
  12. package/dist/errorLogging.js.map +1 -1
  13. package/dist/eventEmitterWithErrorHandling.d.ts.map +1 -1
  14. package/dist/eventEmitterWithErrorHandling.js.map +1 -1
  15. package/dist/events.d.ts.map +1 -1
  16. package/dist/events.js.map +1 -1
  17. package/dist/fluidErrorBase.d.ts.map +1 -1
  18. package/dist/fluidErrorBase.js +4 -4
  19. package/dist/fluidErrorBase.js.map +1 -1
  20. package/dist/index.d.ts +3 -2
  21. package/dist/index.d.ts.map +1 -1
  22. package/dist/index.js.map +1 -1
  23. package/dist/logger.d.ts +15 -4
  24. package/dist/logger.d.ts.map +1 -1
  25. package/dist/logger.js +69 -11
  26. package/dist/logger.js.map +1 -1
  27. package/dist/mockLogger.d.ts +12 -2
  28. package/dist/mockLogger.d.ts.map +1 -1
  29. package/dist/mockLogger.js +24 -2
  30. package/dist/mockLogger.js.map +1 -1
  31. package/dist/packageVersion.d.ts +1 -1
  32. package/dist/packageVersion.js +1 -1
  33. package/dist/packageVersion.js.map +1 -1
  34. package/dist/sampledTelemetryHelper.d.ts.map +1 -1
  35. package/dist/sampledTelemetryHelper.js.map +1 -1
  36. package/dist/telemetryTypes.d.ts +81 -0
  37. package/dist/telemetryTypes.d.ts.map +1 -0
  38. package/dist/telemetryTypes.js +7 -0
  39. package/dist/telemetryTypes.js.map +1 -0
  40. package/dist/thresholdCounter.d.ts.map +1 -1
  41. package/dist/thresholdCounter.js.map +1 -1
  42. package/dist/utils.d.ts.map +1 -1
  43. package/dist/utils.js.map +1 -1
  44. package/lib/config.d.ts.map +1 -1
  45. package/lib/config.js +3 -3
  46. package/lib/config.js.map +1 -1
  47. package/lib/debugLogger.d.ts.map +1 -1
  48. package/lib/debugLogger.js +1 -1
  49. package/lib/debugLogger.js.map +1 -1
  50. package/lib/errorLogging.d.ts +8 -7
  51. package/lib/errorLogging.d.ts.map +1 -1
  52. package/lib/errorLogging.js +23 -20
  53. package/lib/errorLogging.js.map +1 -1
  54. package/lib/eventEmitterWithErrorHandling.d.ts.map +1 -1
  55. package/lib/eventEmitterWithErrorHandling.js.map +1 -1
  56. package/lib/events.d.ts.map +1 -1
  57. package/lib/events.js.map +1 -1
  58. package/lib/fluidErrorBase.d.ts.map +1 -1
  59. package/lib/fluidErrorBase.js +4 -4
  60. package/lib/fluidErrorBase.js.map +1 -1
  61. package/lib/index.d.ts +3 -2
  62. package/lib/index.d.ts.map +1 -1
  63. package/lib/index.js +2 -2
  64. package/lib/index.js.map +1 -1
  65. package/lib/logger.d.ts +15 -4
  66. package/lib/logger.d.ts.map +1 -1
  67. package/lib/logger.js +69 -12
  68. package/lib/logger.js.map +1 -1
  69. package/lib/mockLogger.d.ts +12 -2
  70. package/lib/mockLogger.d.ts.map +1 -1
  71. package/lib/mockLogger.js +24 -2
  72. package/lib/mockLogger.js.map +1 -1
  73. package/lib/packageVersion.d.ts +1 -1
  74. package/lib/packageVersion.js +1 -1
  75. package/lib/packageVersion.js.map +1 -1
  76. package/lib/sampledTelemetryHelper.d.ts.map +1 -1
  77. package/lib/sampledTelemetryHelper.js.map +1 -1
  78. package/lib/telemetryTypes.d.ts +81 -0
  79. package/lib/telemetryTypes.d.ts.map +1 -0
  80. package/lib/telemetryTypes.js +6 -0
  81. package/lib/telemetryTypes.js.map +1 -0
  82. package/lib/thresholdCounter.d.ts.map +1 -1
  83. package/lib/thresholdCounter.js.map +1 -1
  84. package/lib/utils.d.ts.map +1 -1
  85. package/lib/utils.js.map +1 -1
  86. package/package.json +14 -12
  87. package/prettier.config.cjs +1 -1
  88. package/src/config.ts +173 -172
  89. package/src/debugLogger.ts +118 -111
  90. package/src/errorLogging.ts +306 -300
  91. package/src/eventEmitterWithErrorHandling.ts +16 -12
  92. package/src/events.ts +26 -26
  93. package/src/fluidErrorBase.ts +42 -38
  94. package/src/index.ts +29 -9
  95. package/src/logger.ts +568 -500
  96. package/src/mockLogger.ts +118 -88
  97. package/src/packageVersion.ts +1 -1
  98. package/src/sampledTelemetryHelper.ts +122 -122
  99. package/src/telemetryTypes.ts +96 -0
  100. package/src/thresholdCounter.ts +34 -34
  101. package/src/utils.ts +15 -15
  102. package/tsconfig.esnext.json +6 -6
  103. package/tsconfig.json +9 -13
package/src/logger.ts CHANGED
@@ -4,48 +4,53 @@
4
4
  */
5
5
 
6
6
  import {
7
- ITelemetryBaseEvent,
8
- ITelemetryBaseLogger,
9
- ITelemetryErrorEvent,
10
- ITelemetryGenericEvent,
11
- ITelemetryLogger,
12
- ITelemetryPerformanceEvent,
13
- ITelemetryProperties,
14
- TelemetryEventPropertyType,
15
- ITaggedTelemetryPropertyType,
16
- TelemetryEventCategory,
7
+ ITelemetryBaseEvent,
8
+ ITelemetryBaseLogger,
9
+ ITelemetryErrorEvent,
10
+ ITelemetryGenericEvent,
11
+ ITelemetryLogger,
12
+ ITelemetryPerformanceEvent,
13
+ ITelemetryProperties,
14
+ TelemetryEventPropertyType,
15
+ ITaggedTelemetryPropertyType,
16
+ TelemetryEventCategory,
17
17
  } from "@fluidframework/common-definitions";
18
18
  import { performance } from "@fluidframework/common-utils";
19
+ import { CachedConfigProvider, loggerIsMonitoringContext, mixinMonitoringContext } from "./config";
19
20
  import {
20
- CachedConfigProvider,
21
- loggerIsMonitoringContext,
22
- mixinMonitoringContext,
23
- } from "./config";
24
- import {
25
- isILoggingError,
26
- extractLogSafeErrorProperties,
27
- generateStack,
21
+ isILoggingError,
22
+ extractLogSafeErrorProperties,
23
+ generateStack,
24
+ isTaggedTelemetryPropertyValue,
28
25
  } from "./errorLogging";
26
+ import {
27
+ ITaggedTelemetryPropertyTypeExt,
28
+ ITelemetryEventExt,
29
+ ITelemetryGenericEventExt,
30
+ ITelemetryLoggerExt,
31
+ ITelemetryPerformanceEventExt,
32
+ TelemetryEventPropertyTypeExt,
33
+ } from "./telemetryTypes";
29
34
 
30
35
  /**
31
36
  * Broad classifications to be applied to individual properties as they're prepared to be logged to telemetry.
32
37
  * Please do not modify existing entries for backwards compatibility.
33
38
  */
34
39
  export enum TelemetryDataTag {
35
- /** Data containing terms or IDs from code packages that may have been dynamically loaded */
36
- CodeArtifact = "CodeArtifact",
37
- /** Personal data of a variety of classifications that pertains to the user */
38
- UserData = "UserData",
40
+ /** Data containing terms or IDs from code packages that may have been dynamically loaded */
41
+ CodeArtifact = "CodeArtifact",
42
+ /** Personal data of a variety of classifications that pertains to the user */
43
+ UserData = "UserData",
39
44
  }
40
45
 
41
46
  export type TelemetryEventPropertyTypes = TelemetryEventPropertyType | ITaggedTelemetryPropertyType;
42
47
 
43
48
  export interface ITelemetryLoggerPropertyBag {
44
- [index: string]: TelemetryEventPropertyTypes | (() => TelemetryEventPropertyTypes);
49
+ [index: string]: TelemetryEventPropertyTypes | (() => TelemetryEventPropertyTypes);
45
50
  }
46
- export interface ITelemetryLoggerPropertyBags{
47
- all?: ITelemetryLoggerPropertyBag;
48
- error?: ITelemetryLoggerPropertyBag;
51
+ export interface ITelemetryLoggerPropertyBags {
52
+ all?: ITelemetryLoggerPropertyBag;
53
+ error?: ITelemetryLoggerPropertyBag;
49
54
  }
50
55
 
51
56
  /**
@@ -53,170 +58,178 @@ export interface ITelemetryLoggerPropertyBags{
53
58
  * encoding in one place schemas for various types of Fluid telemetry events.
54
59
  * Creates sub-logger that appends properties to all events
55
60
  */
56
- export abstract class TelemetryLogger implements ITelemetryLogger {
57
- public static readonly eventNamespaceSeparator = ":";
58
-
59
- public static formatTick(tick: number): number {
60
- return Math.floor(tick);
61
- }
62
-
63
- /**
64
- * Attempts to parse number from string.
65
- * If fails,returns original string.
66
- * Used to make telemetry data typed (and support math operations, like comparison),
67
- * in places where we do expect numbers (like contentsize/duration property in http header)
68
- */
69
- public static numberFromString(str: string | null | undefined): string | number | undefined {
70
- if (str === undefined || str === null) {
71
- return undefined;
72
- }
73
- const num = Number(str);
74
- return Number.isNaN(num) ? str : num;
75
- }
76
-
77
- public static sanitizePkgName(name: string) {
78
- return name.replace("@", "").replace("/", "-");
79
- }
80
-
81
- /**
82
- * Take an unknown error object and add the appropriate info from it to the event. Message and stack will be copied
83
- * over from the error object, along with other telemetry properties if it's an ILoggingError.
84
- * @param event - Event being logged
85
- * @param error - Error to extract info from
86
- * @param fetchStack - Whether to fetch the current callstack if error.stack is undefined
87
- */
88
- public static prepareErrorObject(event: ITelemetryBaseEvent, error: any, fetchStack: boolean) {
89
- const { message, errorType, stack } = extractLogSafeErrorProperties(error, true /* sanitizeStack */);
90
- // First, copy over error message, stack, and errorType directly (overwrite if present on event)
91
- event.stack = stack;
92
- event.error = message; // Note that the error message goes on the 'error' field
93
- event.errorType = errorType;
94
-
95
- if (isILoggingError(error)) {
96
- // Add any other telemetry properties from the LoggingError
97
- const telemetryProp = error.getTelemetryProperties();
98
- for (const key of Object.keys(telemetryProp)) {
99
- if (event[key] !== undefined) {
100
- // Don't overwrite existing properties on the event
101
- continue;
102
- }
103
- event[key] = telemetryProp[key];
104
- }
105
- }
106
-
107
- // Collect stack if we were not able to extract it from error
108
- if (event.stack === undefined && fetchStack) {
109
- event.stack = generateStack();
110
- }
111
- }
112
-
113
- public constructor(
114
- protected readonly namespace?: string,
115
- protected readonly properties?: ITelemetryLoggerPropertyBags) {
116
- }
117
-
118
- /**
119
- * Send an event with the logger
120
- *
121
- * @param event - the event to send
122
- */
123
- public abstract send(event: ITelemetryBaseEvent): void;
124
-
125
- /**
126
- * Send a telemetry event with the logger
127
- *
128
- * @param event - the event to send
129
- * @param error - optional error object to log
130
- */
131
- public sendTelemetryEvent(event: ITelemetryGenericEvent, error?: any) {
132
- this.sendTelemetryEventCore({ ...event, category: event.category ?? "generic" }, error);
133
- }
134
-
135
- /**
136
- * Send a telemetry event with the logger
137
- *
138
- * @param event - the event to send
139
- * @param error - optional error object to log
140
- */
141
- protected sendTelemetryEventCore(
142
- event: ITelemetryGenericEvent & { category: TelemetryEventCategory; },
143
- error?: any) {
144
- const newEvent = { ...event };
145
- if (error !== undefined) {
146
- TelemetryLogger.prepareErrorObject(newEvent, error, false);
147
- }
148
-
149
- // Will include Nan & Infinity, but probably we do not care
150
- if (typeof newEvent.duration === "number") {
151
- newEvent.duration = TelemetryLogger.formatTick(newEvent.duration);
152
- }
153
-
154
- this.send(newEvent);
155
- }
156
-
157
- /**
158
- * Send an error telemetry event with the logger
159
- *
160
- * @param event - the event to send
161
- * @param error - optional error object to log
162
- */
163
- public sendErrorEvent(event: ITelemetryErrorEvent, error?: any) {
164
- this.sendTelemetryEventCore({
165
- // ensure the error field has some value,
166
- // this can and will be overridden by event, or error
167
- error: event.eventName,
168
- ...event,
169
- category: "error",
170
- }, error);
171
- }
172
-
173
- /**
174
- * Send a performance telemetry event with the logger
175
- *
176
- * @param event - Event to send
177
- * @param error - optional error object to log
178
- */
179
- public sendPerformanceEvent(event: ITelemetryPerformanceEvent, error?: any): void {
180
- const perfEvent = {
181
- ...event,
182
- category: event.category ?? "performance",
183
- };
184
-
185
- this.sendTelemetryEventCore(perfEvent, error);
186
- }
187
-
188
- protected prepareEvent(event: ITelemetryBaseEvent): ITelemetryBaseEvent {
189
- const includeErrorProps = event.category === "error" || event.error !== undefined;
190
- const newEvent: ITelemetryBaseEvent = {
191
- ...event,
192
- };
193
- if (this.namespace !== undefined) {
194
- newEvent.eventName = `${this.namespace}${TelemetryLogger.eventNamespaceSeparator}${newEvent.eventName}`;
195
- }
196
- if (this.properties) {
197
- const properties: (undefined | ITelemetryLoggerPropertyBag)[] = [];
198
- properties.push(this.properties.all);
199
- if (includeErrorProps) {
200
- properties.push(this.properties.error);
201
- }
202
- for (const props of properties) {
203
- if (props !== undefined) {
204
- for (const key of Object.keys(props)) {
205
- if (event[key] !== undefined) {
206
- continue;
207
- }
208
- const getterOrValue = props[key];
209
- // If this throws, hopefully it is handled elsewhere
210
- const value = typeof getterOrValue === "function" ? getterOrValue() : getterOrValue;
211
- if (value !== undefined) {
212
- newEvent[key] = value;
213
- }
214
- }
215
- }
216
- }
217
- }
218
- return newEvent;
219
- }
61
+ export abstract class TelemetryLogger implements ITelemetryLoggerExt {
62
+ public static readonly eventNamespaceSeparator = ":";
63
+
64
+ public static formatTick(tick: number): number {
65
+ return Math.floor(tick);
66
+ }
67
+
68
+ /**
69
+ * Attempts to parse number from string.
70
+ * If fails,returns original string.
71
+ * Used to make telemetry data typed (and support math operations, like comparison),
72
+ * in places where we do expect numbers (like contentsize/duration property in http header)
73
+ */
74
+ public static numberFromString(str: string | null | undefined): string | number | undefined {
75
+ if (str === undefined || str === null) {
76
+ return undefined;
77
+ }
78
+ const num = Number(str);
79
+ return Number.isNaN(num) ? str : num;
80
+ }
81
+
82
+ public static sanitizePkgName(name: string) {
83
+ return name.replace("@", "").replace("/", "-");
84
+ }
85
+
86
+ /**
87
+ * Take an unknown error object and add the appropriate info from it to the event. Message and stack will be copied
88
+ * over from the error object, along with other telemetry properties if it's an ILoggingError.
89
+ * @param event - Event being logged
90
+ * @param error - Error to extract info from
91
+ * @param fetchStack - Whether to fetch the current callstack if error.stack is undefined
92
+ */
93
+ public static prepareErrorObject(event: ITelemetryBaseEvent, error: any, fetchStack: boolean) {
94
+ const { message, errorType, stack } = extractLogSafeErrorProperties(
95
+ error,
96
+ true /* sanitizeStack */,
97
+ );
98
+ // First, copy over error message, stack, and errorType directly (overwrite if present on event)
99
+ event.stack = stack;
100
+ event.error = message; // Note that the error message goes on the 'error' field
101
+ event.errorType = errorType;
102
+
103
+ if (isILoggingError(error)) {
104
+ // Add any other telemetry properties from the LoggingError
105
+ const telemetryProp = error.getTelemetryProperties();
106
+ for (const key of Object.keys(telemetryProp)) {
107
+ if (event[key] !== undefined) {
108
+ // Don't overwrite existing properties on the event
109
+ continue;
110
+ }
111
+ event[key] = telemetryProp[key];
112
+ }
113
+ }
114
+
115
+ // Collect stack if we were not able to extract it from error
116
+ if (event.stack === undefined && fetchStack) {
117
+ event.stack = generateStack();
118
+ }
119
+ }
120
+
121
+ public constructor(
122
+ protected readonly namespace?: string,
123
+ protected readonly properties?: ITelemetryLoggerPropertyBags,
124
+ ) {}
125
+
126
+ /**
127
+ * Send an event with the logger
128
+ *
129
+ * @param event - the event to send
130
+ */
131
+ public abstract send(event: ITelemetryBaseEvent): void;
132
+
133
+ /**
134
+ * Send a telemetry event with the logger
135
+ *
136
+ * @param event - the event to send
137
+ * @param error - optional error object to log
138
+ */
139
+ public sendTelemetryEvent(event: ITelemetryGenericEventExt, error?: any) {
140
+ this.sendTelemetryEventCore({ ...event, category: event.category ?? "generic" }, error);
141
+ }
142
+
143
+ /**
144
+ * Send a telemetry event with the logger
145
+ *
146
+ * @param event - the event to send
147
+ * @param error - optional error object to log
148
+ */
149
+ protected sendTelemetryEventCore(
150
+ event: ITelemetryGenericEventExt & { category: TelemetryEventCategory },
151
+ error?: any,
152
+ ) {
153
+ const newEvent = convertToBaseEvent(event);
154
+ if (error !== undefined) {
155
+ TelemetryLogger.prepareErrorObject(newEvent, error, false);
156
+ }
157
+
158
+ // Will include Nan & Infinity, but probably we do not care
159
+ if (typeof newEvent.duration === "number") {
160
+ newEvent.duration = TelemetryLogger.formatTick(newEvent.duration);
161
+ }
162
+
163
+ this.send(newEvent);
164
+ }
165
+
166
+ /**
167
+ * Send an error telemetry event with the logger
168
+ *
169
+ * @param event - the event to send
170
+ * @param error - optional error object to log
171
+ */
172
+ public sendErrorEvent(event: ITelemetryErrorEvent, error?: any) {
173
+ this.sendTelemetryEventCore(
174
+ {
175
+ // ensure the error field has some value,
176
+ // this can and will be overridden by event, or error
177
+ error: event.eventName,
178
+ ...event,
179
+ category: "error",
180
+ },
181
+ error,
182
+ );
183
+ }
184
+
185
+ /**
186
+ * Send a performance telemetry event with the logger
187
+ *
188
+ * @param event - Event to send
189
+ * @param error - optional error object to log
190
+ */
191
+ public sendPerformanceEvent(event: ITelemetryPerformanceEventExt, error?: any): void {
192
+ const perfEvent = {
193
+ ...event,
194
+ category: event.category ?? "performance",
195
+ };
196
+
197
+ this.sendTelemetryEventCore(perfEvent, error);
198
+ }
199
+
200
+ protected prepareEvent(event: ITelemetryBaseEvent): ITelemetryBaseEvent {
201
+ const includeErrorProps = event.category === "error" || event.error !== undefined;
202
+ const newEvent: ITelemetryBaseEvent = {
203
+ ...event,
204
+ };
205
+ if (this.namespace !== undefined) {
206
+ newEvent.eventName = `${this.namespace}${TelemetryLogger.eventNamespaceSeparator}${newEvent.eventName}`;
207
+ }
208
+ if (this.properties) {
209
+ const properties: (undefined | ITelemetryLoggerPropertyBag)[] = [];
210
+ properties.push(this.properties.all);
211
+ if (includeErrorProps) {
212
+ properties.push(this.properties.error);
213
+ }
214
+ for (const props of properties) {
215
+ if (props !== undefined) {
216
+ for (const key of Object.keys(props)) {
217
+ if (event[key] !== undefined) {
218
+ continue;
219
+ }
220
+ const getterOrValue = props[key];
221
+ // If this throws, hopefully it is handled elsewhere
222
+ const value =
223
+ typeof getterOrValue === "function" ? getterOrValue() : getterOrValue;
224
+ if (value !== undefined) {
225
+ newEvent[key] = value;
226
+ }
227
+ }
228
+ }
229
+ }
230
+ }
231
+ return newEvent;
232
+ }
220
233
  }
221
234
 
222
235
  /**
@@ -224,46 +237,45 @@ export abstract class TelemetryLogger implements ITelemetryLogger {
224
237
  * container-runtime. Issue: #8191
225
238
  * TaggedLoggerAdapter class can add tag handling to your logger.
226
239
  */
227
- export class TaggedLoggerAdapter implements ITelemetryBaseLogger {
228
- public constructor(
229
- private readonly logger: ITelemetryBaseLogger) {
230
- }
231
-
232
- public send(eventWithTagsMaybe: ITelemetryBaseEvent) {
233
- const newEvent: ITelemetryBaseEvent = {
234
- category: eventWithTagsMaybe.category,
235
- eventName: eventWithTagsMaybe.eventName,
236
- };
237
- for (const key of Object.keys(eventWithTagsMaybe)) {
238
- const taggableProp = eventWithTagsMaybe[key];
239
- const { value, tag } = (typeof taggableProp === "object")
240
- ? taggableProp
241
- : { value: taggableProp, tag: undefined };
242
- switch (tag) {
243
- case undefined:
244
- // No tag means we can log plainly
245
- newEvent[key] = value;
246
- break;
247
- case "PackageData": // For back-compat
248
- case TelemetryDataTag.CodeArtifact:
249
- // For Microsoft applications, CodeArtifact is safe for now
250
- // (we don't load 3P code in 1P apps)
251
- newEvent[key] = value;
252
- break;
253
- case TelemetryDataTag.UserData:
254
- // Strip out anything tagged explicitly as PII.
255
- // Alternate strategy would be to hash these props
256
- newEvent[key] = "REDACTED (UserData)";
257
- break;
258
- default:
259
- // If we encounter a tag we don't recognize
260
- // then we must assume we should scrub.
261
- newEvent[key] = "REDACTED (unknown tag)";
262
- break;
263
- }
264
- }
265
- this.logger.send(newEvent);
266
- }
240
+ export class TaggedLoggerAdapter implements ITelemetryBaseLogger {
241
+ public constructor(private readonly logger: ITelemetryBaseLogger) {}
242
+
243
+ public send(eventWithTagsMaybe: ITelemetryBaseEvent) {
244
+ const newEvent: ITelemetryBaseEvent = {
245
+ category: eventWithTagsMaybe.category,
246
+ eventName: eventWithTagsMaybe.eventName,
247
+ };
248
+ for (const key of Object.keys(eventWithTagsMaybe)) {
249
+ const taggableProp = eventWithTagsMaybe[key];
250
+ const { value, tag } =
251
+ typeof taggableProp === "object"
252
+ ? taggableProp
253
+ : { value: taggableProp, tag: undefined };
254
+ switch (tag) {
255
+ case undefined:
256
+ // No tag means we can log plainly
257
+ newEvent[key] = value;
258
+ break;
259
+ case "PackageData": // For back-compat
260
+ case TelemetryDataTag.CodeArtifact:
261
+ // For Microsoft applications, CodeArtifact is safe for now
262
+ // (we don't load 3P code in 1P apps)
263
+ newEvent[key] = value;
264
+ break;
265
+ case TelemetryDataTag.UserData:
266
+ // Strip out anything tagged explicitly as PII.
267
+ // Alternate strategy would be to hash these props
268
+ newEvent[key] = "REDACTED (UserData)";
269
+ break;
270
+ default:
271
+ // If we encounter a tag we don't recognize
272
+ // then we must assume we should scrub.
273
+ newEvent[key] = "REDACTED (unknown tag)";
274
+ break;
275
+ }
276
+ }
277
+ this.logger.send(newEvent);
278
+ }
267
279
  }
268
280
 
269
281
  /**
@@ -272,81 +284,78 @@ export abstract class TelemetryLogger implements ITelemetryLogger {
272
284
  * Creates sub-logger that appends properties to all events
273
285
  */
274
286
  export class ChildLogger extends TelemetryLogger {
275
- /**
276
- * Create child logger
277
- * @param baseLogger - Base logger to use to output events. If undefined, proper child logger
278
- * is created, but it does not send telemetry events anywhere.
279
- * @param namespace - Telemetry event name prefix to add to all events
280
- * @param properties - Base properties to add to all events
281
- * @param propertyGetters - Getters to add additional properties to all events
282
- */
283
- public static create(
284
- baseLogger?: ITelemetryBaseLogger,
285
- namespace?: string,
286
- properties?: ITelemetryLoggerPropertyBags): TelemetryLogger {
287
- // if we are creating a child of a child, rather than nest, which will increase
288
- // the callstack overhead, just generate a new logger that includes everything from the previous
289
- if (baseLogger instanceof ChildLogger) {
290
- const combinedProperties: ITelemetryLoggerPropertyBags = {};
291
- for (const extendedProps of [baseLogger.properties, properties]) {
292
- if (extendedProps !== undefined) {
293
- if (extendedProps.all !== undefined) {
294
- combinedProperties.all = {
295
- ... combinedProperties.all,
296
- ... extendedProps.all,
297
- };
298
- }
299
- if (extendedProps.error !== undefined) {
300
- combinedProperties.error = {
301
- ... combinedProperties.error,
302
- ... extendedProps.error,
303
- };
304
- }
305
- }
306
- }
307
-
308
- const combinedNamespace = baseLogger.namespace === undefined
309
- ? namespace
310
- : namespace === undefined
311
- ? baseLogger.namespace
312
- : `${baseLogger.namespace}${TelemetryLogger.eventNamespaceSeparator}${namespace}`;
313
-
314
- return new ChildLogger(
315
- baseLogger.baseLogger,
316
- combinedNamespace,
317
- combinedProperties,
318
- );
319
- }
320
-
321
- return new ChildLogger(
322
- baseLogger ? baseLogger : new BaseTelemetryNullLogger(),
323
- namespace,
324
- properties);
325
- }
326
-
327
- private constructor(
328
- protected readonly baseLogger: ITelemetryBaseLogger,
329
- namespace: string | undefined,
330
- properties: ITelemetryLoggerPropertyBags | undefined,
331
- ) {
332
- super(namespace, properties);
333
-
334
- // propagate the monitoring context
335
- if (loggerIsMonitoringContext(baseLogger)) {
336
- mixinMonitoringContext(
337
- this,
338
- new CachedConfigProvider(baseLogger.config));
339
- }
340
- }
341
-
342
- /**
343
- * Send an event with the logger
344
- *
345
- * @param event - the event to send
346
- */
347
- public send(event: ITelemetryBaseEvent): void {
348
- this.baseLogger.send(this.prepareEvent(event));
349
- }
287
+ /**
288
+ * Create child logger
289
+ * @param baseLogger - Base logger to use to output events. If undefined, proper child logger
290
+ * is created, but it does not send telemetry events anywhere.
291
+ * @param namespace - Telemetry event name prefix to add to all events
292
+ * @param properties - Base properties to add to all events
293
+ * @param propertyGetters - Getters to add additional properties to all events
294
+ */
295
+ public static create(
296
+ baseLogger?: ITelemetryBaseLogger,
297
+ namespace?: string,
298
+ properties?: ITelemetryLoggerPropertyBags,
299
+ ): TelemetryLogger {
300
+ // if we are creating a child of a child, rather than nest, which will increase
301
+ // the callstack overhead, just generate a new logger that includes everything from the previous
302
+ if (baseLogger instanceof ChildLogger) {
303
+ const combinedProperties: ITelemetryLoggerPropertyBags = {};
304
+ for (const extendedProps of [baseLogger.properties, properties]) {
305
+ if (extendedProps !== undefined) {
306
+ if (extendedProps.all !== undefined) {
307
+ combinedProperties.all = {
308
+ ...combinedProperties.all,
309
+ ...extendedProps.all,
310
+ };
311
+ }
312
+ if (extendedProps.error !== undefined) {
313
+ combinedProperties.error = {
314
+ ...combinedProperties.error,
315
+ ...extendedProps.error,
316
+ };
317
+ }
318
+ }
319
+ }
320
+
321
+ const combinedNamespace =
322
+ baseLogger.namespace === undefined
323
+ ? namespace
324
+ : namespace === undefined
325
+ ? baseLogger.namespace
326
+ : `${baseLogger.namespace}${TelemetryLogger.eventNamespaceSeparator}${namespace}`;
327
+
328
+ return new ChildLogger(baseLogger.baseLogger, combinedNamespace, combinedProperties);
329
+ }
330
+
331
+ return new ChildLogger(
332
+ baseLogger ? baseLogger : new BaseTelemetryNullLogger(),
333
+ namespace,
334
+ properties,
335
+ );
336
+ }
337
+
338
+ private constructor(
339
+ protected readonly baseLogger: ITelemetryBaseLogger,
340
+ namespace: string | undefined,
341
+ properties: ITelemetryLoggerPropertyBags | undefined,
342
+ ) {
343
+ super(namespace, properties);
344
+
345
+ // propagate the monitoring context
346
+ if (loggerIsMonitoringContext(baseLogger)) {
347
+ mixinMonitoringContext(this, new CachedConfigProvider(baseLogger.config));
348
+ }
349
+ }
350
+
351
+ /**
352
+ * Send an event with the logger
353
+ *
354
+ * @param event - the event to send
355
+ */
356
+ public send(event: ITelemetryBaseEvent): void {
357
+ this.baseLogger.send(this.prepareEvent(event));
358
+ }
350
359
  }
351
360
 
352
361
  /**
@@ -355,41 +364,39 @@ export class ChildLogger extends TelemetryLogger {
355
364
  * Implements ITelemetryBaseLogger (through static create() method)
356
365
  */
357
366
  export class MultiSinkLogger extends TelemetryLogger {
358
- protected loggers: ITelemetryBaseLogger[] = [];
359
-
360
- /**
361
- * Create multiple sink logger (i.e. logger that sends events to multiple sinks)
362
- * @param namespace - Telemetry event name prefix to add to all events
363
- * @param properties - Base properties to add to all events
364
- * @param propertyGetters - Getters to add additional properties to all events
365
- */
366
- constructor(
367
- namespace?: string,
368
- properties?: ITelemetryLoggerPropertyBags) {
369
- super(namespace, properties);
370
- }
371
-
372
- /**
373
- * Add logger to send all events to
374
- * @param logger - Logger to add
375
- */
376
- public addLogger(logger?: ITelemetryBaseLogger) {
377
- if (logger !== undefined && logger !== null) {
378
- this.loggers.push(logger);
379
- }
380
- }
381
-
382
- /**
383
- * Send an event to the loggers
384
- *
385
- * @param event - the event to send to all the registered logger
386
- */
387
- public send(event: ITelemetryBaseEvent): void {
388
- const newEvent = this.prepareEvent(event);
389
- this.loggers.forEach((logger: ITelemetryBaseLogger) => {
390
- logger.send(newEvent);
391
- });
392
- }
367
+ protected loggers: ITelemetryBaseLogger[] = [];
368
+
369
+ /**
370
+ * Create multiple sink logger (i.e. logger that sends events to multiple sinks)
371
+ * @param namespace - Telemetry event name prefix to add to all events
372
+ * @param properties - Base properties to add to all events
373
+ * @param propertyGetters - Getters to add additional properties to all events
374
+ */
375
+ constructor(namespace?: string, properties?: ITelemetryLoggerPropertyBags) {
376
+ super(namespace, properties);
377
+ }
378
+
379
+ /**
380
+ * Add logger to send all events to
381
+ * @param logger - Logger to add
382
+ */
383
+ public addLogger(logger?: ITelemetryBaseLogger) {
384
+ if (logger !== undefined && logger !== null) {
385
+ this.loggers.push(logger);
386
+ }
387
+ }
388
+
389
+ /**
390
+ * Send an event to the loggers
391
+ *
392
+ * @param event - the event to send to all the registered logger
393
+ */
394
+ public send(event: ITelemetryBaseEvent): void {
395
+ const newEvent = this.prepareEvent(event);
396
+ this.loggers.forEach((logger: ITelemetryBaseLogger) => {
397
+ logger.send(newEvent);
398
+ });
399
+ }
393
400
  }
394
401
 
395
402
  /**
@@ -399,129 +406,135 @@ export class MultiSinkLogger extends TelemetryLogger {
399
406
  * success / failure tracking, including duration (on success).
400
407
  */
401
408
  export interface IPerformanceEventMarkers {
402
- start?: true;
403
- end?: true;
404
- cancel?: "generic" | "error"; // tells wether to issue "generic" or "error" category cancel event
409
+ start?: true;
410
+ end?: true;
411
+ cancel?: "generic" | "error"; // tells wether to issue "generic" or "error" category cancel event
405
412
  }
406
413
 
407
414
  /**
408
415
  * Helper class to log performance events
409
416
  */
410
417
  export class PerformanceEvent {
411
- public static start(logger: ITelemetryLogger, event: ITelemetryGenericEvent, markers?: IPerformanceEventMarkers) {
412
- return new PerformanceEvent(logger, event, markers);
413
- }
414
-
415
- public static timedExec<T>(
416
- logger: ITelemetryLogger,
417
- event: ITelemetryGenericEvent,
418
- callback: (event: PerformanceEvent) => T,
419
- markers?: IPerformanceEventMarkers,
420
- ) {
421
- const perfEvent = PerformanceEvent.start(logger, event, markers);
422
- try {
423
- const ret = callback(perfEvent);
424
- perfEvent.autoEnd();
425
- return ret;
426
- } catch (error) {
427
- perfEvent.cancel(undefined, error);
428
- throw error;
429
- }
430
- }
431
-
432
- public static async timedExecAsync<T>(
433
- logger: ITelemetryLogger,
434
- event: ITelemetryGenericEvent,
435
- callback: (event: PerformanceEvent) => Promise<T>,
436
- markers?: IPerformanceEventMarkers,
437
- ) {
438
- const perfEvent = PerformanceEvent.start(logger, event, markers);
439
- try {
440
- const ret = await callback(perfEvent);
441
- perfEvent.autoEnd();
442
- return ret;
443
- } catch (error) {
444
- perfEvent.cancel(undefined, error);
445
- throw error;
446
- }
447
- }
448
-
449
- public get duration() { return performance.now() - this.startTime; }
450
-
451
- private event?: ITelemetryGenericEvent;
452
- private readonly startTime = performance.now();
453
- private startMark?: string;
454
-
455
- protected constructor(
456
- private readonly logger: ITelemetryLogger,
457
- event: ITelemetryGenericEvent,
458
- private readonly markers: IPerformanceEventMarkers = { end: true, cancel: "generic" },
459
- ) {
460
- this.event = { ...event };
461
- if (this.markers.start) {
462
- this.reportEvent("start");
463
- }
464
-
465
- if (typeof window === "object" && window != null && window.performance) {
466
- this.startMark = `${event.eventName}-start`;
467
- window.performance.mark(this.startMark);
468
- }
469
- }
470
-
471
- public reportProgress(props?: ITelemetryProperties, eventNameSuffix: string = "update"): void {
472
- this.reportEvent(eventNameSuffix, props);
473
- }
474
-
475
- private autoEnd() {
476
- // Event might have been cancelled or ended in the callback
477
- if (this.event && this.markers.end) {
478
- this.reportEvent("end");
479
- }
480
- this.performanceEndMark();
481
- this.event = undefined;
482
- }
483
-
484
- public end(props?: ITelemetryProperties): void {
485
- this.reportEvent("end", props);
486
- this.performanceEndMark();
487
- this.event = undefined;
488
- }
489
-
490
- private performanceEndMark() {
491
- if (this.startMark && this.event) {
492
- const endMark = `${this.event.eventName}-end`;
493
- window.performance.mark(endMark);
494
- window.performance.measure(`${this.event.eventName}`, this.startMark, endMark);
495
- this.startMark = undefined;
496
- }
497
- }
498
-
499
- public cancel(props?: ITelemetryProperties, error?: any): void {
500
- if (this.markers.cancel !== undefined) {
501
- this.reportEvent("cancel", { category: this.markers.cancel, ...props }, error);
502
- }
503
- this.event = undefined;
504
- }
505
-
506
- /**
507
- * Report the event, if it hasn't already been reported.
508
- */
509
- public reportEvent(eventNameSuffix: string, props?: ITelemetryProperties, error?: any) {
510
- // There are strange sequences involving multiple Promise chains
511
- // where the event can be cancelled and then later a callback is invoked
512
- // and the caller attempts to end directly, e.g. issue #3936. Just return.
513
- if (!this.event) {
514
- return;
515
- }
516
-
517
- const event: ITelemetryPerformanceEvent = { ...this.event, ...props };
518
- event.eventName = `${event.eventName}_${eventNameSuffix}`;
519
- if (eventNameSuffix !== "start") {
520
- event.duration = this.duration;
521
- }
522
-
523
- this.logger.sendPerformanceEvent(event, error);
524
- }
418
+ public static start(
419
+ logger: ITelemetryLogger,
420
+ event: ITelemetryGenericEvent,
421
+ markers?: IPerformanceEventMarkers,
422
+ ) {
423
+ return new PerformanceEvent(logger, event, markers);
424
+ }
425
+
426
+ public static timedExec<T>(
427
+ logger: ITelemetryLogger,
428
+ event: ITelemetryGenericEvent,
429
+ callback: (event: PerformanceEvent) => T,
430
+ markers?: IPerformanceEventMarkers,
431
+ ) {
432
+ const perfEvent = PerformanceEvent.start(logger, event, markers);
433
+ try {
434
+ const ret = callback(perfEvent);
435
+ perfEvent.autoEnd();
436
+ return ret;
437
+ } catch (error) {
438
+ perfEvent.cancel(undefined, error);
439
+ throw error;
440
+ }
441
+ }
442
+
443
+ public static async timedExecAsync<T>(
444
+ logger: ITelemetryLogger,
445
+ event: ITelemetryGenericEvent,
446
+ callback: (event: PerformanceEvent) => Promise<T>,
447
+ markers?: IPerformanceEventMarkers,
448
+ ) {
449
+ const perfEvent = PerformanceEvent.start(logger, event, markers);
450
+ try {
451
+ const ret = await callback(perfEvent);
452
+ perfEvent.autoEnd();
453
+ return ret;
454
+ } catch (error) {
455
+ perfEvent.cancel(undefined, error);
456
+ throw error;
457
+ }
458
+ }
459
+
460
+ public get duration() {
461
+ return performance.now() - this.startTime;
462
+ }
463
+
464
+ private event?: ITelemetryGenericEvent;
465
+ private readonly startTime = performance.now();
466
+ private startMark?: string;
467
+
468
+ protected constructor(
469
+ private readonly logger: ITelemetryLogger,
470
+ event: ITelemetryGenericEvent,
471
+ private readonly markers: IPerformanceEventMarkers = { end: true, cancel: "generic" },
472
+ ) {
473
+ this.event = { ...event };
474
+ if (this.markers.start) {
475
+ this.reportEvent("start");
476
+ }
477
+
478
+ if (typeof window === "object" && window != null && window.performance?.mark) {
479
+ this.startMark = `${event.eventName}-start`;
480
+ window.performance.mark(this.startMark);
481
+ }
482
+ }
483
+
484
+ public reportProgress(props?: ITelemetryProperties, eventNameSuffix: string = "update"): void {
485
+ this.reportEvent(eventNameSuffix, props);
486
+ }
487
+
488
+ private autoEnd() {
489
+ // Event might have been cancelled or ended in the callback
490
+ if (this.event && this.markers.end) {
491
+ this.reportEvent("end");
492
+ }
493
+ this.performanceEndMark();
494
+ this.event = undefined;
495
+ }
496
+
497
+ public end(props?: ITelemetryProperties): void {
498
+ this.reportEvent("end", props);
499
+ this.performanceEndMark();
500
+ this.event = undefined;
501
+ }
502
+
503
+ private performanceEndMark() {
504
+ if (this.startMark && this.event) {
505
+ const endMark = `${this.event.eventName}-end`;
506
+ window.performance.mark(endMark);
507
+ window.performance.measure(`${this.event.eventName}`, this.startMark, endMark);
508
+ this.startMark = undefined;
509
+ }
510
+ }
511
+
512
+ public cancel(props?: ITelemetryProperties, error?: any): void {
513
+ if (this.markers.cancel !== undefined) {
514
+ this.reportEvent("cancel", { category: this.markers.cancel, ...props }, error);
515
+ }
516
+ this.event = undefined;
517
+ }
518
+
519
+ /**
520
+ * Report the event, if it hasn't already been reported.
521
+ */
522
+ public reportEvent(eventNameSuffix: string, props?: ITelemetryProperties, error?: any) {
523
+ // There are strange sequences involving multiple Promise chains
524
+ // where the event can be cancelled and then later a callback is invoked
525
+ // and the caller attempts to end directly, e.g. issue #3936. Just return.
526
+ if (!this.event) {
527
+ return;
528
+ }
529
+
530
+ const event: ITelemetryPerformanceEvent = { ...this.event, ...props };
531
+ event.eventName = `${event.eventName}_${eventNameSuffix}`;
532
+ if (eventNameSuffix !== "start") {
533
+ event.duration = this.duration;
534
+ }
535
+
536
+ this.logger.sendPerformanceEvent(event, error);
537
+ }
525
538
  }
526
539
 
527
540
  /**
@@ -529,37 +542,34 @@ export class PerformanceEvent {
529
542
  * It can be used in places where logger instance is required, but events should be not send over.
530
543
  */
531
544
  export class TelemetryUTLogger implements ITelemetryLogger {
532
- public send(event: ITelemetryBaseEvent): void {
533
- }
534
- public sendTelemetryEvent(event: ITelemetryGenericEvent, error?: any) {
535
- }
536
- public sendErrorEvent(event: ITelemetryErrorEvent, error?: any) {
537
- this.reportError("errorEvent in UT logger!", event, error);
538
- }
539
- public sendPerformanceEvent(event: ITelemetryPerformanceEvent, error?: any): void {
540
- }
541
- public logGenericError(eventName: string, error: any) {
542
- this.reportError(`genericError in UT logger!`, { eventName }, error);
543
- }
544
- public logException(event: ITelemetryErrorEvent, exception: any): void {
545
- this.reportError("exception in UT logger!", event, exception);
546
- }
547
- public debugAssert(condition: boolean, event?: ITelemetryErrorEvent): void {
548
- this.reportError("debugAssert in UT logger!");
549
- }
550
- public shipAssert(condition: boolean, event?: ITelemetryErrorEvent): void {
551
- this.reportError("shipAssert in UT logger!");
552
- }
553
-
554
- private reportError(message: string, event?: ITelemetryErrorEvent, err?: any) {
555
- const error = new Error(message);
556
- (error as any).error = error;
557
- (error as any).event = event;
558
- // report to console as exception can be eaten
559
- console.error(message);
560
- console.error(error);
561
- throw error;
562
- }
545
+ public send(event: ITelemetryBaseEvent): void {}
546
+ public sendTelemetryEvent(event: ITelemetryGenericEvent, error?: any) {}
547
+ public sendErrorEvent(event: ITelemetryErrorEvent, error?: any) {
548
+ this.reportError("errorEvent in UT logger!", event, error);
549
+ }
550
+ public sendPerformanceEvent(event: ITelemetryPerformanceEvent, error?: any): void {}
551
+ public logGenericError(eventName: string, error: any) {
552
+ this.reportError(`genericError in UT logger!`, { eventName }, error);
553
+ }
554
+ public logException(event: ITelemetryErrorEvent, exception: any): void {
555
+ this.reportError("exception in UT logger!", event, exception);
556
+ }
557
+ public debugAssert(condition: boolean, event?: ITelemetryErrorEvent): void {
558
+ this.reportError("debugAssert in UT logger!");
559
+ }
560
+ public shipAssert(condition: boolean, event?: ITelemetryErrorEvent): void {
561
+ this.reportError("shipAssert in UT logger!");
562
+ }
563
+
564
+ private reportError(message: string, event?: ITelemetryErrorEvent, err?: any) {
565
+ const error = new Error(message);
566
+ (error as any).error = error;
567
+ (error as any).event = event;
568
+ // report to console as exception can be eaten
569
+ console.error(message);
570
+ console.error(error);
571
+ throw error;
572
+ }
563
573
  }
564
574
 
565
575
  /**
@@ -567,14 +577,14 @@ export class TelemetryUTLogger implements ITelemetryLogger {
567
577
  * It can be used in places where logger instance is required, but events should be not send over.
568
578
  */
569
579
  export class BaseTelemetryNullLogger implements ITelemetryBaseLogger {
570
- /**
571
- * Send an event with the logger
572
- *
573
- * @param event - the event to send
574
- */
575
- public send(event: ITelemetryBaseEvent): void {
576
- return;
577
- }
580
+ /**
581
+ * Send an event with the logger
582
+ *
583
+ * @param event - the event to send
584
+ */
585
+ public send(event: ITelemetryBaseEvent): void {
586
+ return;
587
+ }
578
588
  }
579
589
 
580
590
  /**
@@ -582,8 +592,66 @@ export class BaseTelemetryNullLogger implements ITelemetryBaseLogger {
582
592
  * It can be used in places where logger instance is required, but events should be not send over.
583
593
  */
584
594
  export class TelemetryNullLogger implements ITelemetryLogger {
585
- public send(event: ITelemetryBaseEvent): void {}
586
- public sendTelemetryEvent(event: ITelemetryGenericEvent, error?: any): void {}
587
- public sendErrorEvent(event: ITelemetryErrorEvent, error?: any): void {}
588
- public sendPerformanceEvent(event: ITelemetryPerformanceEvent, error?: any): void {}
595
+ public send(event: ITelemetryBaseEvent): void {}
596
+ public sendTelemetryEvent(event: ITelemetryGenericEvent, error?: any): void {}
597
+ public sendErrorEvent(event: ITelemetryErrorEvent, error?: any): void {}
598
+ public sendPerformanceEvent(event: ITelemetryPerformanceEvent, error?: any): void {}
599
+ }
600
+
601
+ /**
602
+ * Takes in an event object, and converts all of its values to a basePropertyType.
603
+ * In the case of an invalid property type, the value will be converted to an error string.
604
+ * @param event - Event with fields you want to stringify.
605
+ */
606
+ function convertToBaseEvent({
607
+ category,
608
+ eventName,
609
+ ...props
610
+ }: ITelemetryEventExt): ITelemetryBaseEvent {
611
+ const newEvent: ITelemetryBaseEvent = { category, eventName };
612
+ for (const key of Object.keys(props)) {
613
+ newEvent[key] = convertToBasePropertyType(props[key]);
614
+ }
615
+ return newEvent;
616
+ }
617
+
618
+ /**
619
+ * Takes in value, and does one of 4 things.
620
+ * if value is of primitive type - returns the original value.
621
+ * If the value is an array of primitives - returns a stringified version of the array.
622
+ * If the value is an object of type ITaggedTelemetryPropertyType - returns the object
623
+ * with its values recursively converted to base property Type.
624
+ * If none of these cases are reached - returns an error string
625
+ * @param x - value passed in to convert to a base property type
626
+ */
627
+ export function convertToBasePropertyType(
628
+ x: TelemetryEventPropertyTypeExt | ITaggedTelemetryPropertyTypeExt,
629
+ ): TelemetryEventPropertyType | ITaggedTelemetryPropertyType {
630
+ return isTaggedTelemetryPropertyValue(x)
631
+ ? {
632
+ value: convertToBasePropertyTypeUntagged(x.value),
633
+ tag: x.tag,
634
+ }
635
+ : convertToBasePropertyTypeUntagged(x);
636
+ }
637
+
638
+ function convertToBasePropertyTypeUntagged(
639
+ x: TelemetryEventPropertyTypeExt,
640
+ ): TelemetryEventPropertyType {
641
+ switch (typeof x) {
642
+ case "string":
643
+ case "number":
644
+ case "boolean":
645
+ case "undefined":
646
+ return x;
647
+ case "object":
648
+ // We assume this is an array based on the input types
649
+ return JSON.stringify(x);
650
+ default:
651
+ // should never reach this case based on the input types
652
+ console.error(
653
+ `convertToBasePropertyTypeUntagged: INVALID PROPERTY (typed as ${typeof x})`,
654
+ );
655
+ return `INVALID PROPERTY (typed as ${typeof x})`;
656
+ }
589
657
  }