1fb726d48Sopenharmony_ci/*
2fb726d48Sopenharmony_ci * Copyright (C) 2024 Shenzhen Kaihong Digital Industry Development Co., Ltd.
3fb726d48Sopenharmony_ci * Licensed under the Apache License, Version 2.0 (the "License");
4fb726d48Sopenharmony_ci * you may not use this file except in compliance with the License.
5fb726d48Sopenharmony_ci * You may obtain a copy of the License at
6fb726d48Sopenharmony_ci *
7fb726d48Sopenharmony_ci *     http://www.apache.org/licenses/LICENSE-2.0
8fb726d48Sopenharmony_ci *
9fb726d48Sopenharmony_ci * Unless required by applicable law or agreed to in writing, software
10fb726d48Sopenharmony_ci * distributed under the License is distributed on an "AS IS" BASIS,
11fb726d48Sopenharmony_ci * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12fb726d48Sopenharmony_ci * See the License for the specific language governing permissions and
13fb726d48Sopenharmony_ci * limitations under the License.
14fb726d48Sopenharmony_ci */
15fb726d48Sopenharmony_ci
16fb726d48Sopenharmony_ciimport { SpSystemTrace } from '../SpSystemTrace';
17fb726d48Sopenharmony_ciimport { TraceRow } from '../trace/base/TraceRow';
18fb726d48Sopenharmony_ciimport { renders } from '../../database/ui-worker/ProcedureWorker';
19fb726d48Sopenharmony_ciimport { info } from '../../../log/Log';
20fb726d48Sopenharmony_ciimport { HangStruct } from '../../database/ui-worker/ProcedureWorkerHang';
21fb726d48Sopenharmony_ciimport { EmptyRender } from '../../database/ui-worker/cpu/ProcedureWorkerCPU';
22fb726d48Sopenharmony_ciimport { queryHangData } from '../../database/sql/Hang.sql';
23fb726d48Sopenharmony_ciimport { hangDataSender } from '../../database/data-trafic/HangDataSender';
24fb726d48Sopenharmony_ciimport { BaseStruct } from '../../bean/BaseStruct';
25fb726d48Sopenharmony_ciimport { Utils } from '../trace/base/Utils';
26fb726d48Sopenharmony_ci
27fb726d48Sopenharmony_ciexport type HangType = 'Instant' | 'Circumstantial' | 'Micro' | 'Severe' | '';
28fb726d48Sopenharmony_ciconst TIME_MS = 1000000;
29fb726d48Sopenharmony_ci/// Hangs聚合泳道
30fb726d48Sopenharmony_ciexport class SpHangChart {
31fb726d48Sopenharmony_ci  private trace: SpSystemTrace;
32fb726d48Sopenharmony_ci  static funcNameMap: Map<number | string, string> = new Map();
33fb726d48Sopenharmony_ci
34fb726d48Sopenharmony_ci  constructor(trace: SpSystemTrace) {
35fb726d48Sopenharmony_ci    this.trace = trace;
36fb726d48Sopenharmony_ci  }
37fb726d48Sopenharmony_ci
38fb726d48Sopenharmony_ci  static calculateHangType(dur: number): HangType {
39fb726d48Sopenharmony_ci    const durMS = dur / TIME_MS;
40fb726d48Sopenharmony_ci    if (durMS < 33) {
41fb726d48Sopenharmony_ci      return '';
42fb726d48Sopenharmony_ci    }
43fb726d48Sopenharmony_ci    else if (durMS < 100) {
44fb726d48Sopenharmony_ci      return 'Instant';
45fb726d48Sopenharmony_ci    }
46fb726d48Sopenharmony_ci    else if (durMS < 250) {
47fb726d48Sopenharmony_ci      return 'Circumstantial';
48fb726d48Sopenharmony_ci    }
49fb726d48Sopenharmony_ci    else if (durMS < 500) {
50fb726d48Sopenharmony_ci      return 'Micro';
51fb726d48Sopenharmony_ci    }
52fb726d48Sopenharmony_ci    else {
53fb726d48Sopenharmony_ci      return 'Severe';
54fb726d48Sopenharmony_ci    }
55fb726d48Sopenharmony_ci  }
56fb726d48Sopenharmony_ci
57fb726d48Sopenharmony_ci  async init(): Promise<void> {
58fb726d48Sopenharmony_ci    SpHangChart.funcNameMap = Utils.getInstance().getCallStatckMap();
59fb726d48Sopenharmony_ci    let folder = await this.initFolder();
60fb726d48Sopenharmony_ci    await this.initData(folder);
61fb726d48Sopenharmony_ci  }
62fb726d48Sopenharmony_ci
63fb726d48Sopenharmony_ci  private hangSupplierFrame(
64fb726d48Sopenharmony_ci    traceRow: TraceRow<HangStruct>,
65fb726d48Sopenharmony_ci    it: {
66fb726d48Sopenharmony_ci      id: number;
67fb726d48Sopenharmony_ci      name: string;
68fb726d48Sopenharmony_ci      num: number;
69fb726d48Sopenharmony_ci    }
70fb726d48Sopenharmony_ci  ): void {
71fb726d48Sopenharmony_ci    traceRow.supplierFrame = (): Promise<HangStruct[]> => {
72fb726d48Sopenharmony_ci      let promiseData = hangDataSender(it.id, traceRow);
73fb726d48Sopenharmony_ci      if (promiseData === null) {
74fb726d48Sopenharmony_ci        return new Promise<Array<HangStruct>>((resolve) => resolve([]));
75fb726d48Sopenharmony_ci      } else {
76fb726d48Sopenharmony_ci        return promiseData.then((resultHang: Array<HangStruct>) =>
77fb726d48Sopenharmony_ci          resultHang.map(hangItem => ({
78fb726d48Sopenharmony_ci            ...hangItem,
79fb726d48Sopenharmony_ci            pname: it.name,
80fb726d48Sopenharmony_ci            type: SpHangChart.calculateHangType(hangItem.dur!),
81fb726d48Sopenharmony_ci            content: SpHangChart.funcNameMap.get(hangItem.id!)
82fb726d48Sopenharmony_ci          }))
83fb726d48Sopenharmony_ci        );
84fb726d48Sopenharmony_ci      }
85fb726d48Sopenharmony_ci    };
86fb726d48Sopenharmony_ci  }
87fb726d48Sopenharmony_ci
88fb726d48Sopenharmony_ci  private hangThreadHandler(
89fb726d48Sopenharmony_ci    traceRow: TraceRow<HangStruct>,
90fb726d48Sopenharmony_ci    it: {
91fb726d48Sopenharmony_ci      id: number;
92fb726d48Sopenharmony_ci      name: string;
93fb726d48Sopenharmony_ci      num: number;
94fb726d48Sopenharmony_ci    },
95fb726d48Sopenharmony_ci    hangId: number
96fb726d48Sopenharmony_ci  ): void {
97fb726d48Sopenharmony_ci    traceRow.onThreadHandler = (useCache): void => {
98fb726d48Sopenharmony_ci      let context: CanvasRenderingContext2D;
99fb726d48Sopenharmony_ci      if (traceRow.currentContext) {
100fb726d48Sopenharmony_ci        context = traceRow.currentContext;
101fb726d48Sopenharmony_ci      } else {
102fb726d48Sopenharmony_ci        context = traceRow.collect ? this.trace.canvasFavoritePanelCtx! : this.trace.canvasPanelCtx!;
103fb726d48Sopenharmony_ci      }
104fb726d48Sopenharmony_ci      traceRow.canvasSave(context);
105fb726d48Sopenharmony_ci      renders.hang.renderMainThread(
106fb726d48Sopenharmony_ci        {
107fb726d48Sopenharmony_ci          context: context,
108fb726d48Sopenharmony_ci          useCache: useCache,
109fb726d48Sopenharmony_ci          type: it.name,
110fb726d48Sopenharmony_ci          index: hangId,
111fb726d48Sopenharmony_ci        },
112fb726d48Sopenharmony_ci        traceRow
113fb726d48Sopenharmony_ci      );
114fb726d48Sopenharmony_ci      traceRow.canvasRestore(context, this.trace);
115fb726d48Sopenharmony_ci    };
116fb726d48Sopenharmony_ci  }
117fb726d48Sopenharmony_ci
118fb726d48Sopenharmony_ci  /// 初始化聚合泳道信息
119fb726d48Sopenharmony_ci  async initData(folder: TraceRow<BaseStruct>): Promise<void> {
120fb726d48Sopenharmony_ci    let hangStartTime = new Date().getTime();
121fb726d48Sopenharmony_ci    let hangList = await queryHangData();
122fb726d48Sopenharmony_ci    if (hangList.length === 0) {
123fb726d48Sopenharmony_ci      return;
124fb726d48Sopenharmony_ci    }
125fb726d48Sopenharmony_ci    this.trace.rowsEL?.appendChild(folder);
126fb726d48Sopenharmony_ci    for (let i = 0; i < hangList.length; i++) {
127fb726d48Sopenharmony_ci      const it: {
128fb726d48Sopenharmony_ci        id: number,
129fb726d48Sopenharmony_ci        name: string,
130fb726d48Sopenharmony_ci        num: number,
131fb726d48Sopenharmony_ci      } = hangList[i];
132fb726d48Sopenharmony_ci      let traceRow = TraceRow.skeleton<HangStruct>();
133fb726d48Sopenharmony_ci      traceRow.rowId = `${it.name ?? 'Process'} ${it.id}`;
134fb726d48Sopenharmony_ci      traceRow.rowType = TraceRow.ROW_TYPE_HANG;
135fb726d48Sopenharmony_ci      traceRow.rowParentId = folder.rowId;
136fb726d48Sopenharmony_ci      traceRow.style.height = '40px';
137fb726d48Sopenharmony_ci      traceRow.name = `${it.name ?? 'Process'} ${it.id}`;
138fb726d48Sopenharmony_ci      traceRow.rowHidden = !folder.expansion;
139fb726d48Sopenharmony_ci      traceRow.setAttribute('children', '');
140fb726d48Sopenharmony_ci      traceRow.favoriteChangeHandler = this.trace.favoriteChangeHandler;
141fb726d48Sopenharmony_ci      traceRow.selectChangeHandler = this.trace.selectChangeHandler;
142fb726d48Sopenharmony_ci      this.hangSupplierFrame(traceRow, it);
143fb726d48Sopenharmony_ci      traceRow.getCacheData = (args: unknown): Promise<Array<unknown>> => hangDataSender(it.id, traceRow, args);
144fb726d48Sopenharmony_ci      traceRow.focusHandler = (ev): void => {
145fb726d48Sopenharmony_ci        let hangStruct = HangStruct.hoverHangStruct;
146fb726d48Sopenharmony_ci        this.trace?.displayTip(
147fb726d48Sopenharmony_ci          traceRow, hangStruct,
148fb726d48Sopenharmony_ci          `<span>${hangStruct?.type} ${hangStruct?.dur! / TIME_MS}ms</span>`
149fb726d48Sopenharmony_ci        );
150fb726d48Sopenharmony_ci      };
151fb726d48Sopenharmony_ci      traceRow.findHoverStruct = (): void => {
152fb726d48Sopenharmony_ci        HangStruct.hoverHangStruct = traceRow.getHoverStruct();
153fb726d48Sopenharmony_ci      };
154fb726d48Sopenharmony_ci      this.hangThreadHandler(traceRow, it, i);
155fb726d48Sopenharmony_ci      folder.addChildTraceRow(traceRow);
156fb726d48Sopenharmony_ci    }
157fb726d48Sopenharmony_ci    let durTime = new Date().getTime() - hangStartTime;
158fb726d48Sopenharmony_ci    info('The time to load the HangData is: ', durTime);
159fb726d48Sopenharmony_ci  }
160fb726d48Sopenharmony_ci
161fb726d48Sopenharmony_ci  async initFolder(): Promise<TraceRow<BaseStruct>> {
162fb726d48Sopenharmony_ci    let hangFolder = TraceRow.skeleton();
163fb726d48Sopenharmony_ci    hangFolder.rowId = 'Hangs';
164fb726d48Sopenharmony_ci    hangFolder.index = 0;
165fb726d48Sopenharmony_ci    hangFolder.rowType = TraceRow.ROW_TYPE_HANG_GROUP;
166fb726d48Sopenharmony_ci    hangFolder.rowParentId = '';
167fb726d48Sopenharmony_ci    hangFolder.style.height = '40px';
168fb726d48Sopenharmony_ci    hangFolder.folder = true;
169fb726d48Sopenharmony_ci    hangFolder.name = 'Hangs';
170fb726d48Sopenharmony_ci    hangFolder.favoriteChangeHandler = this.trace.favoriteChangeHandler;
171fb726d48Sopenharmony_ci    hangFolder.selectChangeHandler = this.trace.selectChangeHandler; // @ts-ignore
172fb726d48Sopenharmony_ci    hangFolder.supplier = (): Promise<unknown[]> => new Promise<Array<unknown>>((resolve) => resolve([]));
173fb726d48Sopenharmony_ci    hangFolder.onThreadHandler = (useCache): void => {
174fb726d48Sopenharmony_ci      hangFolder.canvasSave(this.trace.canvasPanelCtx!);
175fb726d48Sopenharmony_ci      if (hangFolder.expansion) {
176fb726d48Sopenharmony_ci        // @ts-ignore
177fb726d48Sopenharmony_ci        this.trace.canvasPanelCtx?.clearRect(0, 0, hangFolder.frame.width, hangFolder.frame.height);
178fb726d48Sopenharmony_ci      } else {
179fb726d48Sopenharmony_ci        (renders.empty as EmptyRender).renderMainThread(
180fb726d48Sopenharmony_ci          {
181fb726d48Sopenharmony_ci            context: this.trace.canvasPanelCtx,
182fb726d48Sopenharmony_ci            useCache: useCache,
183fb726d48Sopenharmony_ci            type: '',
184fb726d48Sopenharmony_ci          },
185fb726d48Sopenharmony_ci          hangFolder
186fb726d48Sopenharmony_ci        );
187fb726d48Sopenharmony_ci      }
188fb726d48Sopenharmony_ci      hangFolder.canvasRestore(this.trace.canvasPanelCtx!, this.trace);
189fb726d48Sopenharmony_ci    };
190fb726d48Sopenharmony_ci    return hangFolder;
191fb726d48Sopenharmony_ci  }
192fb726d48Sopenharmony_ci}
193