2023-02-22 21:08:47 -08:00
|
|
|
/**
|
|
|
|
* Copyright (c) Microsoft Corporation.
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2023-02-27 22:31:47 -08:00
|
|
|
import fs from 'fs';
|
|
|
|
import type EventEmitter from 'events';
|
2023-03-15 22:33:40 -07:00
|
|
|
import type { ClientSideCallMetadata, SerializedError, StackFrame } from '@protocol/channels';
|
2023-03-14 15:58:55 -07:00
|
|
|
import type { SerializedClientSideCallMetadata, SerializedStack, SerializedStackFrame } from './isomorphic/traceUtils';
|
2023-02-27 22:31:47 -08:00
|
|
|
import { yazl, yauzl } from '../zipBundle';
|
|
|
|
import { ManualPromise } from './manualPromise';
|
2023-03-15 22:33:40 -07:00
|
|
|
import type { AfterActionTraceEvent, BeforeActionTraceEvent, TraceEvent } from '@trace/trace';
|
2023-02-28 13:26:23 -08:00
|
|
|
import { calculateSha1 } from './crypto';
|
|
|
|
import { monotonicTime } from './time';
|
2023-02-22 21:08:47 -08:00
|
|
|
|
|
|
|
export function serializeClientSideCallMetadata(metadatas: ClientSideCallMetadata[]): SerializedClientSideCallMetadata {
|
2023-02-23 09:10:09 -08:00
|
|
|
const fileNames = new Map<string, number>();
|
|
|
|
const stacks: SerializedStack[] = [];
|
2023-02-22 21:08:47 -08:00
|
|
|
for (const m of metadatas) {
|
|
|
|
if (!m.stack || !m.stack.length)
|
|
|
|
continue;
|
2023-02-23 09:10:09 -08:00
|
|
|
const stack: SerializedStackFrame[] = [];
|
2023-02-22 21:08:47 -08:00
|
|
|
for (const frame of m.stack) {
|
2023-02-23 09:10:09 -08:00
|
|
|
let ordinal = fileNames.get(frame.file);
|
2023-02-22 21:08:47 -08:00
|
|
|
if (typeof ordinal !== 'number') {
|
2023-02-23 09:10:09 -08:00
|
|
|
ordinal = fileNames.size;
|
|
|
|
fileNames.set(frame.file, ordinal);
|
2023-02-22 21:08:47 -08:00
|
|
|
}
|
2023-02-23 09:10:09 -08:00
|
|
|
const stackFrame: SerializedStackFrame = [ordinal, frame.line || 0, frame.column || 0, frame.function || ''];
|
|
|
|
stack.push(stackFrame);
|
2023-02-22 21:08:47 -08:00
|
|
|
}
|
|
|
|
stacks.push([m.id, stack]);
|
|
|
|
}
|
2023-02-23 09:10:09 -08:00
|
|
|
return { files: [...fileNames.keys()], stacks };
|
2023-02-22 21:08:47 -08:00
|
|
|
}
|
2023-02-27 22:31:47 -08:00
|
|
|
|
|
|
|
export async function mergeTraceFiles(fileName: string, temporaryTraceFiles: string[]) {
|
|
|
|
if (temporaryTraceFiles.length === 1) {
|
|
|
|
await fs.promises.rename(temporaryTraceFiles[0], fileName);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const mergePromise = new ManualPromise();
|
|
|
|
const zipFile = new yazl.ZipFile();
|
|
|
|
const entryNames = new Set<string>();
|
|
|
|
(zipFile as any as EventEmitter).on('error', error => mergePromise.reject(error));
|
|
|
|
|
|
|
|
for (let i = 0; i < temporaryTraceFiles.length; ++i) {
|
|
|
|
const tempFile = temporaryTraceFiles[i];
|
|
|
|
const promise = new ManualPromise<void>();
|
|
|
|
yauzl.open(tempFile, (err, inZipFile) => {
|
|
|
|
if (err) {
|
|
|
|
promise.reject(err);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
let pendingEntries = inZipFile.entryCount;
|
|
|
|
inZipFile.on('entry', entry => {
|
|
|
|
let entryName = entry.fileName;
|
2023-05-23 09:36:35 -07:00
|
|
|
if (entry.fileName.match(/[\d-]*trace./))
|
2023-02-27 22:31:47 -08:00
|
|
|
entryName = i + '-' + entry.fileName;
|
|
|
|
inZipFile.openReadStream(entry, (err, readStream) => {
|
|
|
|
if (err) {
|
|
|
|
promise.reject(err);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (!entryNames.has(entryName)) {
|
|
|
|
entryNames.add(entryName);
|
|
|
|
zipFile.addReadStream(readStream!, entryName);
|
|
|
|
}
|
|
|
|
if (--pendingEntries === 0)
|
|
|
|
promise.resolve();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
await promise;
|
|
|
|
}
|
|
|
|
|
|
|
|
zipFile.end(undefined, () => {
|
|
|
|
zipFile.outputStream.pipe(fs.createWriteStream(fileName)).on('close', () => {
|
|
|
|
Promise.all(temporaryTraceFiles.map(tempFile => fs.promises.unlink(tempFile))).then(() => {
|
|
|
|
mergePromise.resolve();
|
|
|
|
});
|
2023-06-30 13:36:50 -07:00
|
|
|
}).on('error', error => mergePromise.reject(error));
|
2023-02-27 22:31:47 -08:00
|
|
|
});
|
|
|
|
await mergePromise;
|
|
|
|
}
|
2023-02-28 13:26:23 -08:00
|
|
|
|
2023-03-15 22:33:40 -07:00
|
|
|
export async function saveTraceFile(fileName: string, traceEvents: TraceEvent[], saveSources: boolean) {
|
2023-02-28 13:26:23 -08:00
|
|
|
const zipFile = new yazl.ZipFile();
|
|
|
|
|
|
|
|
if (saveSources) {
|
|
|
|
const sourceFiles = new Set<string>();
|
|
|
|
for (const event of traceEvents) {
|
2023-03-15 22:33:40 -07:00
|
|
|
if (event.type === 'before') {
|
|
|
|
for (const frame of event.stack || [])
|
|
|
|
sourceFiles.add(frame.file);
|
|
|
|
}
|
2023-02-28 13:26:23 -08:00
|
|
|
}
|
|
|
|
for (const sourceFile of sourceFiles) {
|
|
|
|
await fs.promises.readFile(sourceFile, 'utf8').then(source => {
|
|
|
|
zipFile.addBuffer(Buffer.from(source), 'resources/src@' + calculateSha1(sourceFile) + '.txt');
|
|
|
|
}).catch(() => {});
|
|
|
|
}
|
|
|
|
}
|
2023-04-25 17:38:12 -07:00
|
|
|
|
|
|
|
const sha1s = new Set<string>();
|
|
|
|
for (const event of traceEvents.filter(e => e.type === 'after') as AfterActionTraceEvent[]) {
|
2023-05-11 16:32:32 -07:00
|
|
|
for (const attachment of (event.attachments || [])) {
|
2023-07-25 14:32:56 -07:00
|
|
|
let contentPromise: Promise<Buffer | undefined> | undefined;
|
2023-05-11 16:32:32 -07:00
|
|
|
if (attachment.path)
|
2023-07-25 14:32:56 -07:00
|
|
|
contentPromise = fs.promises.readFile(attachment.path).catch(() => undefined);
|
2023-05-11 16:32:32 -07:00
|
|
|
else if (attachment.base64)
|
|
|
|
contentPromise = Promise.resolve(Buffer.from(attachment.base64, 'base64'));
|
|
|
|
|
|
|
|
const content = await contentPromise;
|
2023-07-25 14:32:56 -07:00
|
|
|
if (content === undefined)
|
|
|
|
continue;
|
|
|
|
|
2023-05-11 16:32:32 -07:00
|
|
|
const sha1 = calculateSha1(content);
|
|
|
|
attachment.sha1 = sha1;
|
|
|
|
delete attachment.path;
|
|
|
|
delete attachment.base64;
|
|
|
|
if (sha1s.has(sha1))
|
|
|
|
continue;
|
|
|
|
sha1s.add(sha1);
|
|
|
|
zipFile.addBuffer(content, 'resources/' + sha1);
|
2023-04-25 17:38:12 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const traceContent = Buffer.from(traceEvents.map(e => JSON.stringify(e)).join('\n'));
|
|
|
|
zipFile.addBuffer(traceContent, 'trace.trace');
|
|
|
|
|
2023-02-28 13:26:23 -08:00
|
|
|
await new Promise(f => {
|
|
|
|
zipFile.end(undefined, () => {
|
|
|
|
zipFile.outputStream.pipe(fs.createWriteStream(fileName)).on('close', f);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-05-05 15:12:18 -07:00
|
|
|
export function createBeforeActionTraceEventForStep(callId: string, parentId: string | undefined, apiName: string, params: Record<string, any> | undefined, wallTime: number, stack: StackFrame[]): BeforeActionTraceEvent {
|
2023-02-28 13:26:23 -08:00
|
|
|
return {
|
2023-03-15 22:33:40 -07:00
|
|
|
type: 'before',
|
|
|
|
callId,
|
2023-05-05 15:12:18 -07:00
|
|
|
parentId,
|
2023-04-21 10:07:23 -07:00
|
|
|
wallTime,
|
2023-02-28 13:26:23 -08:00
|
|
|
startTime: monotonicTime(),
|
|
|
|
class: 'Test',
|
|
|
|
method: 'step',
|
|
|
|
apiName,
|
2023-05-05 15:12:18 -07:00
|
|
|
params: Object.fromEntries(Object.entries(params || {}).map(([name, value]) => [name, generatePreview(value)])),
|
2023-02-28 13:26:23 -08:00
|
|
|
stack,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2023-05-05 15:12:18 -07:00
|
|
|
export function createAfterActionTraceEventForStep(callId: string, attachments: AfterActionTraceEvent['attachments'], error?: SerializedError['error']): AfterActionTraceEvent {
|
2023-03-15 22:33:40 -07:00
|
|
|
return {
|
|
|
|
type: 'after',
|
|
|
|
callId,
|
|
|
|
endTime: monotonicTime(),
|
|
|
|
log: [],
|
2023-04-25 17:38:12 -07:00
|
|
|
attachments,
|
2023-03-15 22:33:40 -07:00
|
|
|
error,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2023-02-28 13:26:23 -08:00
|
|
|
function generatePreview(value: any, visited = new Set<any>()): string {
|
|
|
|
if (visited.has(value))
|
|
|
|
return '';
|
|
|
|
visited.add(value);
|
|
|
|
if (typeof value === 'string')
|
|
|
|
return value;
|
|
|
|
if (typeof value === 'number')
|
|
|
|
return value.toString();
|
|
|
|
if (typeof value === 'boolean')
|
|
|
|
return value.toString();
|
|
|
|
if (value === null)
|
|
|
|
return 'null';
|
|
|
|
if (value === undefined)
|
|
|
|
return 'undefined';
|
|
|
|
if (Array.isArray(value))
|
|
|
|
return '[' + value.map(v => generatePreview(v, visited)).join(', ') + ']';
|
|
|
|
if (typeof value === 'object')
|
|
|
|
return 'Object';
|
|
|
|
return String(value);
|
|
|
|
}
|