/* * Copyright (C) 2022 Huawei Device Co., Ltd. * 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 { BaseElement, element } from '../../../base-ui/BaseElement'; import { LitTable } from '../../../base-ui/table/lit-table'; import { procedurePool } from '../../database/Procedure'; import { info } from '../../../log/Log'; import '../../../base-ui/chart/pie/LitChartPie'; import { LitChartPie } from '../../../base-ui/chart/pie/LitChartPie'; import { LitSelect } from '../../../base-ui/select/LitSelect'; import { LitSelectOption } from '../../../base-ui/select/LitSelectOption'; import '../../../base-ui/progress-bar/LitProgressBar'; import { LitProgressBar } from '../../../base-ui/progress-bar/LitProgressBar'; import './TableNoData'; import { TableNoData } from './TableNoData'; import { getProbablyTime } from '../../database/logic-worker/ProcedureLogicWorkerCommon'; import {queryThreads} from "../../database/sql/ProcessThread.sql"; import { Top20FrequencyThreadHtml } from './Top20FrequencyThread.html'; @element('top20-frequency-thread') export class Top20FrequencyThread extends BaseElement { static threads: { id: number; tid: number; name: string }[] | undefined; traceChange: boolean = false; private frequencyThreadTbl: LitTable | null | undefined; private threadSelect: LitSelect | null | undefined; private frequencyThreadPie: LitChartPie | null | undefined; private currentThread: HTMLDivElement | null | undefined; private frequencyThreadProgress: LitProgressBar | null | undefined; private nodata: TableNoData | null | undefined; private currentTid: number = 0; private frequencyThreadData: Array = []; private sortColumn: string = ''; private sortType: number = 0; initElements(): void { this.nodata = this.shadowRoot!.querySelector('#nodata'); this.frequencyThreadProgress = this.shadowRoot!.querySelector('#loading'); this.frequencyThreadTbl = this.shadowRoot!.querySelector('#tb-process-thread-count'); this.currentThread = this.shadowRoot!.querySelector('#current_thread'); this.threadSelect = this.shadowRoot!.querySelector('#thread_select'); this.frequencyThreadPie = this.shadowRoot!.querySelector('#pie'); this.threadSelect!.onchange = (e) => { this.currentThread!.textContent = (e as any).detail.text; this.currentTid = parseInt((e as any).detail.value); this.frequencyThreadProgress!.loading = true; this.queryData(); }; this.frequencyThreadTbl!.addEventListener('row-click', (evt: any) => { let data = evt.detail.data; data.isSelected = true; if ((evt.detail as any).callBack) { (evt.detail as any).callBack(true); } }); this.frequencyThreadTbl!.addEventListener('column-click', (evt: any) => { this.sortColumn = evt.detail.key; this.sortType = evt.detail.sort; // @ts-ignore this.sortByColumn(evt.detail); }); this.frequencyThreadTbl!.addEventListener('row-hover', (evt: any) => { if (evt.detail.data) { let data = evt.detail.data; data.isHover = true; if ((evt.detail as any).callBack) { (evt.detail as any).callBack(true); } } this.frequencyThreadPie?.showHover(); }); this.frequencyThreadTbl!.itemTextHandleMap.set('freq', (value) => (value === -1 ? 'unknown' : value)); } sortByColumn(detail: any) { // @ts-ignore function compare(frequencyThreadProperty, sort, type) { return function (a: any, b: any) { if (type === 'number') { // @ts-ignore return sort === 2 ? parseFloat(b[frequencyThreadProperty]) - parseFloat(a[frequencyThreadProperty]) : parseFloat(a[frequencyThreadProperty]) - parseFloat(b[frequencyThreadProperty]); } else { if (sort === 2) { return b[frequencyThreadProperty].toString().localeCompare(a[frequencyThreadProperty].toString()); } else { return a[frequencyThreadProperty].toString().localeCompare(b[frequencyThreadProperty].toString()); } } }; } if (detail.key === 'timeStr') { detail.key = 'time'; this.frequencyThreadData.sort(compare(detail.key, detail.sort, 'number')); } else if (detail.key === 'no' || detail.key === 'cpu' || detail.key === 'freq' || detail.key === 'ratio') { this.frequencyThreadData.sort(compare(detail.key, detail.sort, 'number')); } else { this.frequencyThreadData.sort(compare(detail.key, detail.sort, 'string')); } this.frequencyThreadTbl!.recycleDataSource = this.frequencyThreadData; } async init() { if (!this.traceChange) { if (this.frequencyThreadTbl!.recycleDataSource.length > 0) { this.frequencyThreadTbl?.reMeauseHeight(); } return; } this.traceChange = false; this.frequencyThreadProgress!.loading = true; if (Top20FrequencyThread.threads === undefined) { Top20FrequencyThread.threads = (await queryThreads()) || []; this.nodata!.noData = Top20FrequencyThread.threads === undefined || Top20FrequencyThread.threads.length === 0; this.threadSelect!.innerHTML = ''; let threads = Top20FrequencyThread.threads.map((it) => { let option = new LitSelectOption(); option.setAttribute('value', it.tid + ''); option.textContent = it.name; return option; }); this.threadSelect!.append(...threads); this.threadSelect?.initOptions(); this.threadSelect!.value = Top20FrequencyThread.threads[0].tid + ''; this.currentThread!.textContent = Top20FrequencyThread.threads[0].name; this.currentTid = Top20FrequencyThread.threads[0].tid; this.queryData(); } } queryData() { this.queryLogicWorker('scheduling-Thread Freq', 'query Thread Top 20 Frequency Time:', (res) => { this.nodata!.noData = Top20FrequencyThread.threads === undefined || Top20FrequencyThread.threads.length === 0 || res === undefined || res.length === 0; (res as any[]).map((it: any, index: number) => { it.no = index + 1; }); this.frequencyThreadData = res; if (this.sortColumn != '') { this.sortByColumn({ key: this.sortColumn, sort: this.sortType, }); } else { this.frequencyThreadTbl!.recycleDataSource = res; } this.frequencyThreadTbl!.reMeauseHeight(); this.setThreadPieConfig(res); this.frequencyThreadProgress!.loading = false; this.shadowRoot!.querySelector('#tb_vessel')!.scrollTop = 0; }); } private setThreadPieConfig(res: any): void { this.frequencyThreadPie!.config = { appendPadding: 10, data: this.getPieChartData(res), angleField: 'time', colorField: 'freq', colorFieldTransferHandler: (value) => (value === -1 ? 'unknown' : value), radius: 0.8, label: { type: 'outer', }, tip: (obj) => { return `
freq:${obj.obj.freq === -1 ? 'unknown' : obj.obj.freq}
cpu:${obj.obj.cpu}
time:${obj.obj.timeStr}
ratio:${obj.obj.ratio}%
`; }, hoverHandler: (data) => { if (data) { this.frequencyThreadTbl!.setCurrentHover(data); } else { this.frequencyThreadTbl!.mouseOut(); } }, interactions: [ { type: 'element-active', }, ], }; } getPieChartData(res: any[]) { if (res.length > 20) { let pieChartArr: any[] = []; let other: any = { cpu: '-', freq: 'other', time: 0, ratio: '0', totalDur: 0, }; for (let i = 0; i < res.length; i++) { if (i < 19) { pieChartArr.push(res[i]); } else { other.time += res[i].time; other.timeStr = getProbablyTime(other.time); other.totalDur = res[i].totalDur; other.ratio = ((other.time / other.totalDur) * 100).toFixed(2); } } pieChartArr.push(other); return pieChartArr; } return res; } clearData() { this.traceChange = true; this.threadSelect!.innerHTML = ''; this.frequencyThreadPie!.dataSource = []; this.frequencyThreadTbl!.recycleDataSource = []; } queryLogicWorker(option: string, log: string, handler: (res: any) => void) { let frequencyThreadTime = new Date().getTime(); procedurePool.submitWithName('logic0', option, { tid: this.currentTid }, undefined, handler); let durTime = new Date().getTime() - frequencyThreadTime; info(log, durTime); } initHtml(): string { return Top20FrequencyThreadHtml; } }