Skip to content

Commit 78db1fa

Browse files
ericllndgodinez-dh
andauthored
refactor: Cherry-pick latest log export changes into v0.85 (#2399)
Cherry-pick changes made to `LogExport.ts` in `main` into `v0.85` to address [DH-19079](https://deephaven.atlassian.net/browse/DH-19079) - fix: avoid exceeding call stack when exporting logs with large number of queries (#2382) - feat: Allow wildcards for logs blacklist (#2396) [DH-19079]: https://deephaven.atlassian.net/browse/DH-19079?atlOrigin=eyJpIjoiNWRkNTljNzYxNjVmNDY3MDlhMDU5Y2ZhYzA5YTRkZjUiLCJwIjoiZ2l0aHViLWNvbS1KU1cifQ --------- Co-authored-by: dgodinez-dh <[email protected]>
1 parent f1a02ff commit 78db1fa

File tree

2 files changed

+122
-17
lines changed

2 files changed

+122
-17
lines changed

packages/log/src/LogExport.test.ts

+90
Original file line numberDiff line numberDiff line change
@@ -87,4 +87,94 @@ describe('getReduxDataString', () => {
8787
);
8888
expect(result).toBe(expected);
8989
});
90+
91+
it('should handle wildcards in blacklist paths', () => {
92+
const reduxData = {
93+
key1: 'not blacklisted',
94+
key2: {
95+
keyA: {
96+
key1: 'blacklisted',
97+
},
98+
keyB: {
99+
key1: 'blacklisted',
100+
},
101+
keyC: {
102+
key1: 'blacklisted',
103+
},
104+
},
105+
};
106+
const result = getReduxDataString(reduxData, [['key2', '*', 'key1']]);
107+
const expected = JSON.stringify(
108+
{
109+
key1: 'not blacklisted',
110+
key2: {
111+
keyA: {},
112+
keyB: {},
113+
keyC: {},
114+
},
115+
},
116+
null,
117+
2
118+
);
119+
expect(result).toBe(expected);
120+
});
121+
122+
it('should handle nested wildcards in blacklist paths', () => {
123+
const reduxData = {
124+
key1: 'not blacklisted',
125+
key2: {
126+
keyA: {
127+
key1: 'blacklisted',
128+
key2: {
129+
key3: 'blacklisted',
130+
},
131+
},
132+
keyB: {
133+
key1: 'blacklisted',
134+
key2: {
135+
key3: 'blacklisted',
136+
key4: 'blacklisted',
137+
},
138+
},
139+
},
140+
};
141+
const result = getReduxDataString(reduxData, [['key2', '*', '*']]);
142+
const expected = JSON.stringify(
143+
{
144+
key1: 'not blacklisted',
145+
key2: {
146+
keyA: {},
147+
keyB: {},
148+
},
149+
},
150+
null,
151+
2
152+
);
153+
expect(result).toBe(expected);
154+
});
155+
156+
it('should handle wildcard blacklist paths with no matches', () => {
157+
const reduxData = {
158+
key1: 'not blacklisted',
159+
key2: {
160+
keyA: {
161+
key1: 'not blacklisted',
162+
key2: {
163+
key3: 'not blacklisted',
164+
},
165+
},
166+
},
167+
};
168+
const result = getReduxDataString(reduxData, [['*', '*', '*', '*', '*']]); // Matching more than the depth of the object
169+
const expected = JSON.stringify(reduxData, null, 2);
170+
expect(result).toBe(expected);
171+
});
172+
173+
it('root wildcard should blacklist all', () => {
174+
const reduxData = {
175+
key1: 'should not be blacklisted',
176+
};
177+
const result = getReduxDataString(reduxData, [['*']]);
178+
expect(result).toBe('{}');
179+
});
90180
});

packages/log/src/LogExport.ts

+32-17
Original file line numberDiff line numberDiff line change
@@ -3,11 +3,12 @@ import type LogHistory from './LogHistory';
33

44
// List of objects to blacklist
55
// '' represents the root object
6+
// * represents a wildcard key
67
export const DEFAULT_PATH_BLACKLIST: string[][] = [
78
['api'],
89
['client'],
9-
['dashboardData', 'default', 'connection'],
10-
['dashboardData', 'default', 'sessionWrapper', 'dh'],
10+
['dashboardData', '*', 'connection'],
11+
['dashboardData', '*', 'sessionWrapper'],
1112
['layoutStorage'],
1213
['storage'],
1314
];
@@ -26,14 +27,8 @@ function stringifyReplacer(blacklist: string[][]) {
2627
currPath.shift();
2728

2829
// check blacklists
29-
for (let i = 0; i < blacklist.length; i += 1) {
30-
if (
31-
currPath.length === blacklist[i].length &&
32-
currPath.every((v, index) => v === blacklist[i][index])
33-
) {
34-
// blacklist match
35-
return undefined;
36-
}
30+
if (isOnBlackList(currPath, blacklist)) {
31+
return undefined;
3732
}
3833

3934
if (value instanceof Map) {
@@ -45,6 +40,21 @@ function stringifyReplacer(blacklist: string[][]) {
4540
};
4641
}
4742

43+
function isOnBlackList(currPath: string[], blacklist: string[][]): boolean {
44+
for (let i = 0; i < blacklist.length; i += 1) {
45+
if (
46+
currPath.length === blacklist[i].length &&
47+
currPath.every(
48+
(v, index) => blacklist[i][index] === '*' || v === blacklist[i][index]
49+
)
50+
) {
51+
// blacklist match
52+
return true;
53+
}
54+
}
55+
return false;
56+
}
57+
4858
/**
4959
* Returns a new object that is safe to stringify
5060
* All circular references are replaced by the path to the value creating a circular ref
@@ -90,12 +100,17 @@ function makeSafeToStringify(
90100
// The ref could point to this object or just to another child
91101
const curPath = `${path}.${key}`;
92102
potentiallyCircularValues.set(valRecord, curPath);
93-
output[key] = makeSafeToStringify(
94-
val as Record<string, unknown>,
95-
blacklist,
96-
curPath,
97-
potentiallyCircularValues
98-
);
103+
// Convert the path to an array and remove the root
104+
const curPathArray = curPath.split('.').slice(1);
105+
// If the path is on the blacklist, it will eventually be replaced by undefined, so avoid the recursive call
106+
if (!isOnBlackList(curPathArray, blacklist)) {
107+
output[key] = makeSafeToStringify(
108+
val as Record<string, unknown>,
109+
blacklist,
110+
curPath,
111+
potentiallyCircularValues
112+
);
113+
}
99114
}
100115
}
101116
});
@@ -138,7 +153,7 @@ function formatDate(date: Date): string {
138153
* @param logHistory Log history to include in the console.txt file
139154
* @param metadata Additional metadata to include in the metadata.json file
140155
* @param reduxData Redux data to include in the redux.json file
141-
* @param blacklist List of JSON paths to blacklist in redux data. A JSON path is a list representing the path to that value (e.g. client.data would be `['client', 'data']`)
156+
* @param blacklist List of JSON paths to blacklist in redux data. A JSON path is a list representing the path to that value (e.g. client.data would be `['client', 'data']`). Wildcards (*) are accepted in the path.
142157
* @param fileNamePrefix The zip file name without the .zip extension. Ex: test will be saved as test.zip
143158
* @returns A promise that resolves successfully if the log archive is created and downloaded successfully, rejected if there's an error
144159
*/

0 commit comments

Comments
 (0)