/** * 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. */ import type { StackFrame } from '@protocol/channels'; import type { ActionTraceEvent } from '@trace/trace'; import { SplitView } from '@web/components/splitView'; import * as React from 'react'; import { useAsyncMemo } from './helpers'; import './sourceTab.css'; import { StackTraceView } from './stackTrace'; import { CodeMirrorWrapper } from '@web/components/codeMirrorWrapper'; type StackInfo = { frames: StackFrame[]; fileContent: Map; }; export const SourceTab: React.FunctionComponent<{ action: ActionTraceEvent | undefined, }> = ({ action }) => { const [lastAction, setLastAction] = React.useState(); const [selectedFrame, setSelectedFrame] = React.useState(0); if (lastAction !== action) { setLastAction(action); setSelectedFrame(0); } const stackInfo = React.useMemo(() => { if (!action) return { frames: [], fileContent: new Map() }; const frames = action.stack || []; return { frames, fileContent: new Map(), }; }, [action]); const content = useAsyncMemo(async () => { const filePath = stackInfo.frames[selectedFrame]?.file; if (!filePath) return ''; if (!stackInfo.fileContent.has(filePath)) { const sha1 = await calculateSha1(filePath); try { let response = await fetch(`sha1/src@${sha1}.txt`); if (response.status === 404) response = await fetch(`file?path=${filePath}`); stackInfo.fileContent.set(filePath, await response.text()); } catch { stackInfo.fileContent.set(filePath, ``); } } return stackInfo.fileContent.get(filePath)!; }, [stackInfo, selectedFrame], ''); const targetLine = stackInfo.frames[selectedFrame]?.line || 0; const error = action?.error?.message; return ; }; export async function calculateSha1(text: string): Promise { const buffer = new TextEncoder().encode(text); const hash = await crypto.subtle.digest('SHA-1', buffer); const hexCodes = []; const view = new DataView(hash); for (let i = 0; i < view.byteLength; i += 1) { const byte = view.getUint8(i).toString(16).padStart(2, '0'); hexCodes.push(byte); } return hexCodes.join(''); }