From ccd861241772d74dac6e9384361c9004500b58c5 Mon Sep 17 00:00:00 2001
From: Marylia Gutierrez
Date: Mon, 30 May 2022 11:36:20 -0400
Subject: [PATCH] ui: new charts to statement details page
The overview for a statement details page now
shows charts instead of just the mean value for:
- Execution and Planning Time
- Rows Processed
- Execution Retries
- Execution Count
- Contention
Fixes #74517
This commit also introduces mock for matchMedia and canvas
used for testing with jest.
Release note (ui change): Statement Details page now shows charts
for: Execution and Planning Time, Rows Processed, Execution Retries,
Execution Count and Contention.
---
pkg/server/combined_statement_stats.go | 1 +
pkg/ui/workspaces/cluster-ui/BUILD.bazel | 1 +
pkg/ui/workspaces/cluster-ui/jest.config.js | 2 +-
pkg/ui/workspaces/cluster-ui/package.json | 1 +
.../cluster-ui/src/graphs/bargraph/bars.ts | 9 +-
.../src/pageConfig/pageConfig.module.scss | 2 +-
.../src/statementDetails/statementDetails.tsx | 343 +++++++-----------
.../src/statementDetails/timeseriesUtils.ts | 92 +++++
.../src/test-utils/matchMedia.mock.js | 25 ++
.../src/timeScaleDropdown/utils.spec.tsx | 4 +-
.../cluster-ui/src/timeScaleDropdown/utils.ts | 7 +-
pkg/ui/workspaces/cluster-ui/yarn.lock | 22 +-
.../db-console/styl/base/layout-vars.styl | 2 +-
pkg/ui/yarn-vendor | 2 +-
14 files changed, 296 insertions(+), 217 deletions(-)
create mode 100644 pkg/ui/workspaces/cluster-ui/src/statementDetails/timeseriesUtils.ts
create mode 100644 pkg/ui/workspaces/cluster-ui/src/test-utils/matchMedia.mock.js
diff --git a/pkg/server/combined_statement_stats.go b/pkg/server/combined_statement_stats.go
index 91871b4cc7b6..c01a24fbd8cc 100644
--- a/pkg/server/combined_statement_stats.go
+++ b/pkg/server/combined_statement_stats.go
@@ -563,6 +563,7 @@ func getStatementDetailsPerAggregatedTs(
GROUP BY
aggregated_ts,
aggregation_interval
+ ORDER BY aggregated_ts ASC
LIMIT $%d`, whereClause, len(args)+1)
args = append(args, limit)
diff --git a/pkg/ui/workspaces/cluster-ui/BUILD.bazel b/pkg/ui/workspaces/cluster-ui/BUILD.bazel
index 0261e4fa27f9..dabf63e7a4fb 100644
--- a/pkg/ui/workspaces/cluster-ui/BUILD.bazel
+++ b/pkg/ui/workspaces/cluster-ui/BUILD.bazel
@@ -83,6 +83,7 @@ DEPENDENCIES = [
"@npm_cluster_ui//highlight.js",
"@npm_cluster_ui//http-proxy-middleware",
"@npm_cluster_ui//identity-obj-proxy",
+ "@npm_cluster_ui//jest-canvas-mock",
"@npm_cluster_ui//jest-environment-enzyme",
"@npm_cluster_ui//jest-enzyme",
"@npm_cluster_ui//jest-fetch-mock",
diff --git a/pkg/ui/workspaces/cluster-ui/jest.config.js b/pkg/ui/workspaces/cluster-ui/jest.config.js
index 7ee8b57dca91..15f63754496c 100644
--- a/pkg/ui/workspaces/cluster-ui/jest.config.js
+++ b/pkg/ui/workspaces/cluster-ui/jest.config.js
@@ -43,7 +43,7 @@ module.exports = {
],
roots: ["/src"],
testEnvironment: "enzyme",
- setupFilesAfterEnv: ["./enzyme.setup.js", "jest-enzyme"],
+ setupFilesAfterEnv: ["./enzyme.setup.js", "./src/test-utils/matchMedia.mock.js", "jest-enzyme", "jest-canvas-mock"],
transform: {
"^.+\\.tsx?$": "ts-jest",
"^.+\\.jsx?$": ['babel-jest', { configFile: path.resolve(__dirname, 'babel.config.js') }],
diff --git a/pkg/ui/workspaces/cluster-ui/package.json b/pkg/ui/workspaces/cluster-ui/package.json
index f05828074e92..dc1419bfee86 100644
--- a/pkg/ui/workspaces/cluster-ui/package.json
+++ b/pkg/ui/workspaces/cluster-ui/package.json
@@ -104,6 +104,7 @@
"http-proxy-middleware": "^1.0.5",
"identity-obj-proxy": "^3.0.0",
"jest": "^27.5.1",
+ "jest-canvas-mock": "^2.4.0",
"jest-cli": "^27.5.1",
"jest-environment-enzyme": "^7.1.2",
"jest-enzyme": "^7.1.2",
diff --git a/pkg/ui/workspaces/cluster-ui/src/graphs/bargraph/bars.ts b/pkg/ui/workspaces/cluster-ui/src/graphs/bargraph/bars.ts
index 14e4f0c9b410..f69305e734ca 100644
--- a/pkg/ui/workspaces/cluster-ui/src/graphs/bargraph/bars.ts
+++ b/pkg/ui/workspaces/cluster-ui/src/graphs/bargraph/bars.ts
@@ -14,8 +14,8 @@ import { AxisUnits, AxisDomain } from "../utils/domain";
import { barTooltipPlugin } from "./plugins";
const seriesPalette = [
- "#475872",
- "#FFCD02",
+ "#003EBD",
+ "#2AAF44",
"#F16969",
"#4E9FD1",
"#49D990",
@@ -24,7 +24,7 @@ const seriesPalette = [
"#A3415B",
"#B59153",
"#C9DB6D",
- "#203D9B",
+ "#475872",
"#748BF2",
"#91C8F2",
"#FF9696",
@@ -158,6 +158,9 @@ export const getBarChartOpts = (
x: {
range: () => [xAxisDomain.extent[0], xAxisDomain.extent[1]],
},
+ yAxis: {
+ range: () => [yAxisDomain.extent[0], yAxisDomain.extent[1]],
+ },
},
axes: [
{
diff --git a/pkg/ui/workspaces/cluster-ui/src/pageConfig/pageConfig.module.scss b/pkg/ui/workspaces/cluster-ui/src/pageConfig/pageConfig.module.scss
index 58eac363b887..3e010f09be21 100644
--- a/pkg/ui/workspaces/cluster-ui/src/pageConfig/pageConfig.module.scss
+++ b/pkg/ui/workspaces/cluster-ui/src/pageConfig/pageConfig.module.scss
@@ -14,7 +14,7 @@
margin-bottom: -10px;
margin-top: -10px;
- z-index: 1;
+ z-index: 2;
background-color: $colors--background;
&__list {
diff --git a/pkg/ui/workspaces/cluster-ui/src/statementDetails/statementDetails.tsx b/pkg/ui/workspaces/cluster-ui/src/statementDetails/statementDetails.tsx
index ac818772e52c..79ebb55e320f 100644
--- a/pkg/ui/workspaces/cluster-ui/src/statementDetails/statementDetails.tsx
+++ b/pkg/ui/workspaces/cluster-ui/src/statementDetails/statementDetails.tsx
@@ -8,19 +8,22 @@
// by the Apache License, Version 2.0, included in the file
// licenses/APL.txt.
+import React, { ReactNode } from "react";
import { Col, Row, Tabs } from "antd";
-import { Text, Heading, InlineAlert } from "@cockroachlabs/ui-components";
-import { PageConfig, PageConfigItem } from "src/pageConfig";
+import { cockroach, google } from "@cockroachlabs/crdb-protobuf-client";
+import { Text, InlineAlert } from "@cockroachlabs/ui-components";
+import { ArrowLeft } from "@cockroachlabs/icons";
+import { Location } from "history";
import _ from "lodash";
-import React, { ReactNode } from "react";
+import Long from "long";
+import { format as d3Format } from "d3-format";
import { Helmet } from "react-helmet";
import { Link, RouteComponentProps } from "react-router-dom";
import classNames from "classnames/bind";
-import { format as d3Format } from "d3-format";
-import { ArrowLeft } from "@cockroachlabs/icons";
-import { cockroach, google } from "@cockroachlabs/crdb-protobuf-client";
-import Long from "long";
-import { Location } from "history";
+import { PageConfig, PageConfigItem } from "src/pageConfig";
+import { BarGraphTimeSeries } from "../graphs/bargraph";
+import { AxisUnits } from "../graphs";
+import { AlignedData, Options } from "uplot";
import {
NumericStat,
@@ -28,9 +31,7 @@ import {
Bytes,
Duration,
FixLong,
- longToInt,
stdDev,
- formatNumberForDisplay,
unique,
queryByName,
appAttr,
@@ -44,11 +45,7 @@ import { SortSetting } from "src/sortedtable";
import { Tooltip } from "@cockroachlabs/ui-components";
import { PlanDetails } from "./planDetails";
import { SummaryCard } from "src/summaryCard";
-import {
- latencyBreakdown,
- genericBarChart,
- formatTwoPlaces,
-} from "src/barCharts";
+import { latencyBreakdown, genericBarChart } from "src/barCharts";
import { DiagnosticsView } from "./diagnostics/diagnosticsView";
import sortedTableStyles from "src/sortedtable/sortedtable.module.scss";
import summaryCardStyles from "src/summaryCard/summaryCard.module.scss";
@@ -68,6 +65,13 @@ import {
ActivateDiagnosticsModalRef,
ActivateStatementDiagnosticsModal,
} from "../statementsDiagnostics";
+import {
+ generateExecCountTimeseries,
+ generateExecRetriesTimeseries,
+ generateExecuteAndPlanningTimeseries,
+ generateRowsProcessedTimeseries,
+ generateContentionTimeseries,
+} from "./timeseriesUtils";
type IDuration = google.protobuf.IDuration;
type StatementDetailsResponse = cockroach.server.serverpb.StatementDetailsResponse;
type IStatementDiagnosticsReport = cockroach.server.serverpb.IStatementDiagnosticsReport;
@@ -332,15 +336,6 @@ export class StatementDetails extends React.Component<
hasDiagnosticReports = (): boolean =>
this.props.diagnosticsReports.length > 0;
- changeSortSetting = (ss: SortSetting): void => {
- this.setState({
- sortSetting: ss,
- });
- if (this.props.onSortingChange) {
- this.props.onSortingChange("Stats By Node", ss.columnTitle, ss.ascending);
- }
- };
-
refreshStatementDetails = (
timeScale: TimeScale,
statementFingerprintID: string,
@@ -470,13 +465,13 @@ export class StatementDetails extends React.Component<
Statements
- Statement Details
+ Statement Fingerprint
@@ -589,15 +584,9 @@ export class StatementDetails extends React.Component<
total_count,
implicit_txn,
} = this.props.statementDetails.statement.metadata;
-
- const { statement } = this.props.statementDetails;
-
- const totalCountBarChart = longToInt(statement.stats.count);
- const firstAttemptsBarChart = longToInt(
- statement.stats.first_attempt_count,
- );
- const retriesBarChart = totalCountBarChart - firstAttemptsBarChart;
- const maxRetriesBarChart = longToInt(statement.stats.max_retries);
+ const {
+ statement_statistics_per_aggregated_ts,
+ } = this.props.statementDetails;
const nodes: string[] = unique(
(stats.nodes || []).map(node => node.toString()),
@@ -606,7 +595,6 @@ export class StatementDetails extends React.Component<
(stats.nodes || []).map(node => nodeRegions[node.toString()]),
).sort();
- const duration = (v: number) => Duration(v * 1e9);
const lastExec =
stats.last_exec_timestamp &&
moment(stats.last_exec_timestamp.seconds.low * 1e3).format(
@@ -614,18 +602,10 @@ export class StatementDetails extends React.Component<
);
const statementSampled = stats.exec_stats.count > Long.fromNumber(0);
const unavailableTooltip = !statementSampled && (
-
- This metric is part of the statement execution and therefore will
- not be available until the statement is sampled via tracing.
-
- }
- >
- unavailable
-
+
+ This metric is part of the statement execution and therefore will not be
+ available until the statement is sampled.
+
);
const db = databases ? (
@@ -633,6 +613,64 @@ export class StatementDetails extends React.Component<
) : (
(unset)
);
+
+ const statsPerAggregatedTs = statement_statistics_per_aggregated_ts.sort(
+ (a, b) =>
+ a.aggregated_ts.seconds < b.aggregated_ts.seconds
+ ? -1
+ : a.aggregated_ts.seconds > b.aggregated_ts.seconds
+ ? 1
+ : 0,
+ );
+
+ const executionAndPlanningTimeseries: AlignedData = generateExecuteAndPlanningTimeseries(
+ statsPerAggregatedTs,
+ );
+ const executionAndPlanningOps: Partial = {
+ axes: [{}, { label: "Time Spent" }],
+ series: [{}, { label: "Execution" }, { label: "Planning" }],
+ width: 735,
+ };
+
+ const rowsProcessedTimeseries: AlignedData = generateRowsProcessedTimeseries(
+ statsPerAggregatedTs,
+ );
+ const rowsProcessedOps: Partial = {
+ axes: [{}, { label: "Rows" }],
+ series: [{}, { label: "Rows Read" }, { label: "Rows Written" }],
+ width: 735,
+ };
+
+ const execRetriesTimeseries: AlignedData = generateExecRetriesTimeseries(
+ statsPerAggregatedTs,
+ );
+ const execRetriesOps: Partial = {
+ axes: [{}, { label: "Retries" }],
+ series: [{}, { label: "Retries" }],
+ legend: { show: false },
+ width: 735,
+ };
+
+ const execCountTimeseries: AlignedData = generateExecCountTimeseries(
+ statsPerAggregatedTs,
+ );
+ const execCountOps: Partial = {
+ axes: [{}, { label: "Execution Counts" }],
+ series: [{}, { label: "Execution Counts" }],
+ legend: { show: false },
+ width: 735,
+ };
+
+ const contentionTimeseries: AlignedData = generateContentionTimeseries(
+ statsPerAggregatedTs,
+ );
+ const contentionOps: Partial = {
+ axes: [{}, { label: "Contention" }],
+ series: [{}, { label: "Contention" }],
+ legend: { show: false },
+ width: 735,
+ };
+
return (
<>
@@ -655,110 +693,6 @@ export class StatementDetails extends React.Component<
-
-
-
- Mean statement time
-
- {formatNumberForDisplay(
- stats.service_lat.mean,
- duration,
- )}
-
-
-
- Planning time
-
- {formatNumberForDisplay(stats.plan_lat.mean, duration)}
-
-
-
-
- Execution time
-
- {formatNumberForDisplay(stats.run_lat.mean, duration)}
-
-
-
-
-
-
-
-
-
-
- Resource usage
-
-
- Mean rows/bytes read
- {statementSampled && (
-
- {formatNumberForDisplay(
- stats.rows_read.mean,
- formatTwoPlaces,
- )}
- {" / "}
- {formatNumberForDisplay(stats.bytes_read.mean, Bytes)}
-
- )}
- {unavailableTooltip}
-
-
- Mean rows written
-
- {formatNumberForDisplay(
- stats.rows_written?.mean,
- formatTwoPlaces,
- )}
-
-
-
- Max memory usage
- {statementSampled && (
-
- {formatNumberForDisplay(
- stats.exec_stats.max_mem_usage.mean,
- Bytes,
- )}
-
- )}
- {unavailableTooltip}
-
-
- Network usage
- {statementSampled && (
-
- {formatNumberForDisplay(
- stats.exec_stats.network_bytes.mean,
- Bytes,
- )}
-
- )}
- {unavailableTooltip}
-
-
- Max scratch disk usage
- {statementSampled && (
-
- {formatNumberForDisplay(
- stats.exec_stats.max_disk_usage.mean,
- Bytes,
- )}
-
- )}
- {unavailableTooltip}
-
-
-
-
-
-
-
- Statement details
{!isTenant && (
@@ -776,16 +710,10 @@ export class StatementDetails extends React.Component<
)}
-
Database
{db}
-
App
@@ -795,16 +723,16 @@ export class StatementDetails extends React.Component<
)}
+
+
+
+
Failed?
{RenderCount(failed_count, total_count)}
- Distributed execution?
- {RenderCount(dist_sql_count, total_count)}
-
-
- Full Scan?
+ Full scan?
{RenderCount(full_scan_count, total_count)}
@@ -819,50 +747,57 @@ export class StatementDetails extends React.Component<
Last execution time
{lastExec}
-
- Execution counts
-
- First attempts
- {firstAttemptsBarChart}
-
-
- Total executions
- {totalCountBarChart}
-
-
- Retries
- 0,
- },
- )}
- >
- {retriesBarChart}
-
-
-
- Max retries
- 0,
- },
- )}
- >
- {maxRetriesBarChart}
-
-
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
>
);
diff --git a/pkg/ui/workspaces/cluster-ui/src/statementDetails/timeseriesUtils.ts b/pkg/ui/workspaces/cluster-ui/src/statementDetails/timeseriesUtils.ts
new file mode 100644
index 000000000000..3f6f5632233f
--- /dev/null
+++ b/pkg/ui/workspaces/cluster-ui/src/statementDetails/timeseriesUtils.ts
@@ -0,0 +1,92 @@
+// Copyright 2022 The Cockroach Authors.
+//
+// Use of this software is governed by the Business Source License
+// included in the file licenses/BSL.txt.
+//
+// As of the Change Date specified in that file, in accordance with
+// the Business Source License, use of this software will be governed
+// by the Apache License, Version 2.0, included in the file
+// licenses/APL.txt.
+
+import { cockroach } from "@cockroachlabs/crdb-protobuf-client";
+import { AlignedData } from "uplot";
+import { longToInt, TimestampToNumber } from "../util";
+
+type statementStatisticsPerAggregatedTs = cockroach.server.serverpb.StatementDetailsResponse.ICollectedStatementGroupedByAggregatedTs;
+
+export function generateExecuteAndPlanningTimeseries(
+ stats: statementStatisticsPerAggregatedTs[],
+): AlignedData {
+ const ts: Array = [];
+ const execution: Array = [];
+ const planning: Array = [];
+
+ stats.forEach(function(stat: statementStatisticsPerAggregatedTs) {
+ ts.push(TimestampToNumber(stat.aggregated_ts) * 1e3);
+ execution.push(stat.stats.run_lat.mean * 1e9);
+ planning.push(stat.stats.plan_lat.mean * 1e9);
+ });
+
+ return [ts, execution, planning];
+}
+
+export function generateRowsProcessedTimeseries(
+ stats: statementStatisticsPerAggregatedTs[],
+): AlignedData {
+ const ts: Array = [];
+ const read: Array = [];
+ const written: Array = [];
+
+ stats.forEach(function(stat: statementStatisticsPerAggregatedTs) {
+ ts.push(TimestampToNumber(stat.aggregated_ts) * 1e3);
+ read.push(stat.stats.rows_read.mean);
+ written.push(stat.stats.rows_written?.mean * 1e3);
+ });
+
+ return [ts, read, written];
+}
+
+export function generateExecRetriesTimeseries(
+ stats: statementStatisticsPerAggregatedTs[],
+): AlignedData {
+ const ts: Array = [];
+ const retries: Array = [];
+
+ stats.forEach(function(stat: statementStatisticsPerAggregatedTs) {
+ ts.push(TimestampToNumber(stat.aggregated_ts) * 1e3);
+
+ const totalCountBarChart = longToInt(stat.stats.count);
+ const firstAttemptsBarChart = longToInt(stat.stats.first_attempt_count);
+ retries.push(totalCountBarChart - firstAttemptsBarChart);
+ });
+
+ return [ts, retries];
+}
+
+export function generateExecCountTimeseries(
+ stats: statementStatisticsPerAggregatedTs[],
+): AlignedData {
+ const ts: Array = [];
+ const count: Array = [];
+
+ stats.forEach(function(stat: statementStatisticsPerAggregatedTs) {
+ ts.push(TimestampToNumber(stat.aggregated_ts) * 1e3);
+ count.push(longToInt(stat.stats.count));
+ });
+
+ return [ts, count];
+}
+
+export function generateContentionTimeseries(
+ stats: statementStatisticsPerAggregatedTs[],
+): AlignedData {
+ const ts: Array = [];
+ const count: Array = [];
+
+ stats.forEach(function(stat: statementStatisticsPerAggregatedTs) {
+ ts.push(TimestampToNumber(stat.aggregated_ts) * 1e3);
+ count.push(stat.stats.exec_stats.contention_time.mean);
+ });
+
+ return [ts, count];
+}
diff --git a/pkg/ui/workspaces/cluster-ui/src/test-utils/matchMedia.mock.js b/pkg/ui/workspaces/cluster-ui/src/test-utils/matchMedia.mock.js
new file mode 100644
index 000000000000..14238af2340f
--- /dev/null
+++ b/pkg/ui/workspaces/cluster-ui/src/test-utils/matchMedia.mock.js
@@ -0,0 +1,25 @@
+// Copyright 2022 The Cockroach Authors.
+//
+// Use of this software is governed by the Business Source License
+// included in the file licenses/BSL.txt.
+//
+// As of the Change Date specified in that file, in accordance with
+// the Business Source License, use of this software will be governed
+// by the Apache License, Version 2.0, included in the file
+// licenses/APL.txt.
+
+Object.defineProperty(window, "matchMedia", {
+ writable: true,
+ value: jest.fn().mockImplementation(query => ({
+ matches: false,
+ media: query,
+ onchange: null,
+ addListener: jest.fn(), // deprecated
+ removeListener: jest.fn(), // deprecated
+ addEventListener: jest.fn(),
+ removeEventListener: jest.fn(),
+ dispatchEvent: jest.fn(),
+ })),
+});
+
+export {};
diff --git a/pkg/ui/workspaces/cluster-ui/src/timeScaleDropdown/utils.spec.tsx b/pkg/ui/workspaces/cluster-ui/src/timeScaleDropdown/utils.spec.tsx
index 160570c49def..d4d860ecc1e9 100644
--- a/pkg/ui/workspaces/cluster-ui/src/timeScaleDropdown/utils.spec.tsx
+++ b/pkg/ui/workspaces/cluster-ui/src/timeScaleDropdown/utils.spec.tsx
@@ -28,7 +28,7 @@ describe("timescale utils", (): void => {
};
const [start, end] = toRoundedDateRange(ts);
assert.equal(start.format("YYYY.MM.DD HH:mm:ss"), "2022.01.05 13:00:00");
- assert.equal(end.format("YYYY.MM.DD HH:mm:ss"), "2022.01.10 14:00:00");
+ assert.equal(end.format("YYYY.MM.DD HH:mm:ss"), "2022.01.10 13:59:59");
});
it("already rounded values", () => {
@@ -40,7 +40,7 @@ describe("timescale utils", (): void => {
};
const [start, end] = toRoundedDateRange(ts);
assert.equal(start.format("YYYY.MM.DD HH:mm:ss"), "2022.01.05 13:00:00");
- assert.equal(end.format("YYYY.MM.DD HH:mm:ss"), "2022.01.10 14:00:00");
+ assert.equal(end.format("YYYY.MM.DD HH:mm:ss"), "2022.01.10 13:59:59");
});
});
diff --git a/pkg/ui/workspaces/cluster-ui/src/timeScaleDropdown/utils.ts b/pkg/ui/workspaces/cluster-ui/src/timeScaleDropdown/utils.ts
index fab8b50db684..eea81a368279 100644
--- a/pkg/ui/workspaces/cluster-ui/src/timeScaleDropdown/utils.ts
+++ b/pkg/ui/workspaces/cluster-ui/src/timeScaleDropdown/utils.ts
@@ -88,10 +88,10 @@ export const toDateRange = (ts: TimeScale): [moment.Moment, moment.Moment] => {
};
// toRoundedDateRange round the TimeScale selected, with the start
-// rounded down and end rounded up one hour.
+// rounded down and end rounded up to the limit before the next hour.
// e.g.
// start: 17:45:23 -> 17:00:00
-// end: 20:14:32 -> 21:00:00
+// end: 20:14:32 -> 20:59:59
export const toRoundedDateRange = (
ts: TimeScale,
): [moment.Moment, moment.Moment] => {
@@ -99,7 +99,8 @@ export const toRoundedDateRange = (
const startRounded = start.set({ minute: 0, second: 0, millisecond: 0 });
const endRounded = end
.set({ minute: 0, second: 0, millisecond: 0 })
- .add(1, "hours");
+ .add(59, "minutes")
+ .add(59, "seconds");
return [startRounded, endRounded];
};
diff --git a/pkg/ui/workspaces/cluster-ui/yarn.lock b/pkg/ui/workspaces/cluster-ui/yarn.lock
index d59bd9774b34..2eed04b28811 100644
--- a/pkg/ui/workspaces/cluster-ui/yarn.lock
+++ b/pkg/ui/workspaces/cluster-ui/yarn.lock
@@ -5186,7 +5186,7 @@ color-name@1.1.3:
resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.3.tgz#a7d0558bd89c42f795dd42328f740831ca53bc25"
integrity sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=
-color-name@~1.1.4:
+color-name@^1.1.4, color-name@~1.1.4:
version "1.1.4"
resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.4.tgz#c2a09a87acbde69543de6f63fa3995c826c536a2"
integrity sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==
@@ -5615,6 +5615,11 @@ cssesc@^3.0.0:
resolved "https://registry.yarnpkg.com/cssesc/-/cssesc-3.0.0.tgz#37741919903b868565e1c09ea747445cd18983ee"
integrity sha512-/Tb/JcjK111nNScGob5MNtsntNM1aCNUDipB/TkwZFhyDrrE47SOx/18wF2bbjgc3ZzCSKW1T5nt5EbFoAz/Vg==
+cssfontparser@^1.2.1:
+ version "1.2.1"
+ resolved "https://registry.yarnpkg.com/cssfontparser/-/cssfontparser-1.2.1.tgz#f4022fc8f9700c68029d542084afbaf425a3f3e3"
+ integrity sha512-6tun4LoZnj7VN6YeegOVb67KBX/7JJsqvj+pv3ZA7F878/eN33AbGa5b/S/wXxS/tcp8nc40xRUrsPlxIyNUPg==
+
cssom@^0.4.4:
version "0.4.4"
resolved "https://registry.yarnpkg.com/cssom/-/cssom-0.4.4.tgz#5a66cf93d2d0b661d80bf6a44fb65f5c2e4e0a10"
@@ -8516,6 +8521,14 @@ iterate-value@^1.0.2:
es-get-iterator "^1.0.2"
iterate-iterator "^1.0.1"
+jest-canvas-mock@^2.4.0:
+ version "2.4.0"
+ resolved "https://registry.yarnpkg.com/jest-canvas-mock/-/jest-canvas-mock-2.4.0.tgz#947b71442d7719f8e055decaecdb334809465341"
+ integrity sha512-mmMpZzpmLzn5vepIaHk5HoH3Ka4WykbSoLuG/EKoJd0x0ID/t+INo1l8ByfcUJuDM+RIsL4QDg/gDnBbrj2/IQ==
+ dependencies:
+ cssfontparser "^1.2.1"
+ moo-color "^1.0.2"
+
jest-changed-files@^27.5.1:
version "27.5.1"
resolved "https://registry.yarnpkg.com/jest-changed-files/-/jest-changed-files-27.5.1.tgz#a348aed00ec9bf671cc58a66fcbe7c3dfd6a68f5"
@@ -9799,6 +9812,13 @@ moment@2.x, moment@^2.24.0:
resolved "https://registry.yarnpkg.com/moment/-/moment-2.29.1.tgz#b2be769fa31940be9eeea6469c075e35006fa3d3"
integrity sha512-kHmoybcPV8Sqy59DwNDY3Jefr64lK/by/da0ViFcuA4DH0vQg5Q6Ze5VimxkfQNSC+Mls/Kx53s7TjP1RhFEDQ==
+moo-color@^1.0.2:
+ version "1.0.3"
+ resolved "https://registry.yarnpkg.com/moo-color/-/moo-color-1.0.3.tgz#d56435f8359c8284d83ac58016df7427febece74"
+ integrity sha512-i/+ZKXMDf6aqYtBhuOcej71YSlbjT3wCO/4H1j8rPvxDJEifdwgg5MaFyu6iYAT8GBZJg2z0dkgK4YMzvURALQ==
+ dependencies:
+ color-name "^1.1.4"
+
moo@^0.5.0:
version "0.5.1"
resolved "https://registry.yarnpkg.com/moo/-/moo-0.5.1.tgz#7aae7f384b9b09f620b6abf6f74ebbcd1b65dbc4"
diff --git a/pkg/ui/workspaces/db-console/styl/base/layout-vars.styl b/pkg/ui/workspaces/db-console/styl/base/layout-vars.styl
index 03012aeaa1eb..1149c7a58aac 100644
--- a/pkg/ui/workspaces/db-console/styl/base/layout-vars.styl
+++ b/pkg/ui/workspaces/db-console/styl/base/layout-vars.styl
@@ -20,4 +20,4 @@ $max-window-width = 1350px
$z-index-tooltip = 4
$z-index-banner = 3
$z-index-navigation = 2
-$z-index-page-config = 1
+$z-index-page-config = 2
diff --git a/pkg/ui/yarn-vendor b/pkg/ui/yarn-vendor
index 8b59cbe830a8..55001854aa05 160000
--- a/pkg/ui/yarn-vendor
+++ b/pkg/ui/yarn-vendor
@@ -1 +1 @@
-Subproject commit 8b59cbe830a81fb2a36a0659c01981c7fc017f7c
+Subproject commit 55001854aa057c94b221b34d0f0d85b069d94aaf