Skip to content

Commit

Permalink
ui: update sort setting on search criteria
Browse files Browse the repository at this point in the history
Fixes cockroachdb#99397
When a new search criteria is applying, the table on Statement
and Transactions are sorted to match the value selected by the
search criteria.
When a new column is selected on the table, a warning is displayed
to let the users know they're looking into a subset od the data.
If the new column selected is one of the options on the search
criteria, we give a suggestion to update the search criteria with
that value instead.

This commit adds the counter per page on the Statement
and Transaction tables.

This commit also adds analytics to the apply button,
sending the information about each criteria.

Release note (ui change): Add a warning when a user select a sorting
column on Statement and Transaction tables that were not the original
selected sorting on the search criteria.
  • Loading branch information
maryliag committed Mar 27, 2023
1 parent fadd137 commit f2e448e
Show file tree
Hide file tree
Showing 12 changed files with 194 additions and 6 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -136,7 +136,7 @@ export const SchemaInsightsView: React.FC<SchemaInsightsViewProps> = ({
// redux changes and syncs the URL params with redux.
syncHistory(
{
ascending: sortSetting.ascending.toString(),
ascending: sortSetting.ascending?.toString(),
columnTitle: sortSetting.columnTitle,
...getFullFiltersAsStringRecord(filters),
[SCHEMA_INSIGHT_SEARCH_PARAM]: search,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -621,6 +621,7 @@ const statementsPagePropsFixture: StatementsPageProps = {
onFilterChange: noop,
onChangeLimit: noop,
onChangeReqSort: noop,
onApplySearchCriteria: noop,
};

export const statementsPagePropsWithRequestError: StatementsPageProps = {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -56,6 +56,10 @@ cl-table-container {
margin-bottom: 0px;
}

.margin-bottom {
margin-bottom: 10px;
}

.root {
flex-grow: 0;
width: 100%;
Expand Down
56 changes: 53 additions & 3 deletions pkg/ui/workspaces/cluster-ui/src/statementsPage/statementsPage.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -21,7 +21,7 @@ import {
updateSortSettingQueryParamsOnTab,
} from "src/sortedtable";
import { Search } from "src/search";
import { Pagination } from "src/pagination";
import { Pagination, ResultsPerPageLabel } from "src/pagination";
import {
calculateActiveFilters,
defaultFilters,
Expand Down Expand Up @@ -59,6 +59,7 @@ import {
SqlStatsSortType,
StatementsRequest,
createCombinedStmtsRequest,
SqlStatsSortOptions,
} from "src/api/statementsApi";
import ClearStats from "../sqlActivity/clearStats";
import LoadingError from "../sqlActivity/errorComponent";
Expand All @@ -83,8 +84,8 @@ import {
STATS_LONG_LOADING_DURATION,
stmtRequestSortOptions,
getSortLabel,
getSortColumn,
} from "src/util/sqlActivityConstants";
import { Button } from "src/button";
import { SearchCriteria } from "src/searchCriteria/searchCriteria";
import timeScaleStyles from "../timeScaleDropdown/timeScale.module.scss";

Expand Down Expand Up @@ -125,6 +126,7 @@ export interface StatementsPageDispatchProps {
onTimeScaleChange: (ts: TimeScale) => void;
onChangeLimit: (limit: number) => void;
onChangeReqSort: (sort: SqlStatsSortType) => void;
onApplySearchCriteria: (ts: TimeScale, limit: number, sort: string) => void;
}
export interface StatementsPageStateProps {
statements: AggregateStatistics[];
Expand Down Expand Up @@ -275,6 +277,12 @@ export class StatementsPage extends React.Component<
}
};

isSortSettingSameAsReqSort = (): boolean => {
return;
getSortColumn(this.state.reqSortSetting) ==
this.props.sortSetting.columnTitle;
};

changeTimeScale = (ts: TimeScale): void => {
this.setState(prevState => ({
...prevState,
Expand All @@ -294,8 +302,19 @@ export class StatementsPage extends React.Component<
if (this.props.timeScale !== this.state.timeScale) {
this.props.onTimeScaleChange(this.state.timeScale);
}

if (this.props.onApplySearchCriteria) {
this.props.onApplySearchCriteria(
this.state.timeScale,
this.state.limit,
getSortLabel(this.state.reqSortSetting),
);
}
this.refreshStatements();
const ss: SortSetting = {
ascending: false,
columnTitle: getSortColumn(this.state.reqSortSetting),
};
this.changeSortSetting(ss);
};

resetPagination = (): void => {
Expand Down Expand Up @@ -487,6 +506,16 @@ export class StatementsPage extends React.Component<
this.setState(prevState => ({ ...prevState, reqSortSetting: newSort }));
};

hasReqSortOption = (): boolean => {
let found = false;
Object.values(SqlStatsSortOptions).forEach((option: SqlStatsSortType) => {
if (getSortColumn(option) == this.props.sortSetting.columnTitle) {
found = true;
}
});
return found;
};

renderStatements = (): React.ReactElement => {
const { pagination, filters, activeFilters } = this.state;
const {
Expand Down Expand Up @@ -564,6 +593,13 @@ export class StatementsPage extends React.Component<

const period = timeScaleToString(this.props.timeScale);
const sortSettingLabel = getSortLabel(this.props.reqSortSetting);
const warningSuggestion = this.hasReqSortOption()
? `Update Search Criteria to see the statement fingerprints
sorted on ${getLabel(
this.props.sortSetting.columnTitle as StatisticTableColumnKeys,
"statement",
)}.`
: "";

return (
<>
Expand Down Expand Up @@ -607,6 +643,12 @@ export class StatementsPage extends React.Component<
<p className={timeScaleStylesCx("time-label")}>
Showing aggregated stats from{" "}
<span className={timeScaleStylesCx("bold")}>{period}</span>
{", "}
<ResultsPerPageLabel
pagination={{ ...pagination, total: data.length }}
pageName={"Statements"}
search={search}
/>
</p>
</PageConfigItem>
{hasAdminRole && (
Expand All @@ -629,6 +671,14 @@ export class StatementsPage extends React.Component<
onRemoveFilter={this.onSubmitFilters}
onClearFilters={this.onClearFilters}
/>
{!this.isSortSettingSameAsReqSort() && (
<InlineAlert
intent="warning"
title={`You are viewing a subset (Top ${this.props.limit}) of fingerprints by ${sortSettingLabel}.
${warningSuggestion}`}
className={cx("margin-bottom")}
/>
)}
<StatementsSortedTable
className="statements-table"
data={data}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -258,6 +258,16 @@ export const ConnectedStatementsPage = withRouter(
dispatch(updateStmtsPageLimitAction(limit)),
onChangeReqSort: (sort: SqlStatsSortType) =>
dispatch(updateStmsPageReqSortAction(sort)),
onApplySearchCriteria: (ts: TimeScale, limit: number, sort: string) =>
dispatch(
analyticsActions.track({
name: "Apply Search Criteria",
page: "Statements",
tsValue: ts.key,
limitValue: limit,
sortValue: sort,
}),
),
},
activePageProps: mapDispatchToRecentStatementsPageProps(dispatch),
}),
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,14 @@ type Page =
| "Workload Insights - Statement"
| "Workload Insights - Transaction";

type ApplySearchCriteriaEvent = {
name: "Apply Search Criteria";
page: Page;
tsValue: string;
limitValue: number;
sortValue: string;
};

type BackButtonClick = {
name: "Back Clicked";
page: Page;
Expand Down Expand Up @@ -102,6 +110,7 @@ type TimeScaleChangeEvent = {
};

type AnalyticsEvent =
| ApplySearchCriteriaEvent
| BackButtonClick
| ColumnsChangeEvent
| FilterEvent
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,7 @@ import {
SortSetting,
updateSortSettingQueryParamsOnTab,
} from "../sortedtable";
import { Pagination } from "../pagination";
import { Pagination, ResultsPerPageLabel } from "../pagination";
import { statisticsClasses } from "./transactionsPageClasses";
import {
aggregateAcrossNodeIDs,
Expand Down Expand Up @@ -54,6 +54,7 @@ import {
SqlStatsSortType,
createCombinedStmtsRequest,
StatementsRequest,
SqlStatsSortOptions,
} from "src/api/statementsApi";
import ColumnsSelector from "../columnsSelector/columnsSelector";
import { SelectOption } from "../multiSelectCheckbox/multiSelectCheckbox";
Expand All @@ -79,8 +80,8 @@ import {
STATS_LONG_LOADING_DURATION,
txnRequestSortOptions,
getSortLabel,
getSortColumn,
} from "src/util/sqlActivityConstants";
import { Button } from "src/button";
import { SearchCriteria } from "src/searchCriteria/searchCriteria";
import timeScaleStyles from "../timeScaleDropdown/timeScale.module.scss";

Expand Down Expand Up @@ -131,6 +132,7 @@ export interface TransactionsPageDispatchProps {
columnTitle: string,
ascending: boolean,
) => void;
onApplySearchCriteria: (ts: TimeScale, limit: number, sort: string) => void;
}

export type TransactionsPageProps = TransactionsPageStateProps &
Expand Down Expand Up @@ -289,6 +291,13 @@ export class TransactionsPage extends React.Component<
}
};

isSortSettingSameAsReqSort = (): boolean => {
return (
getSortColumn(this.state.reqSortSetting) ==
this.props.sortSetting.columnTitle
);
};

onChangePage = (current: number): void => {
const { pagination } = this.state;
this.setState({ pagination: { ...pagination, current } });
Expand Down Expand Up @@ -403,7 +412,30 @@ export class TransactionsPage extends React.Component<
this.props.onTimeScaleChange(this.state.timeScale);
}

if (this.props.onApplySearchCriteria) {
this.props.onApplySearchCriteria(
this.state.timeScale,
this.state.limit,
getSortLabel(this.state.reqSortSetting),
);
}
this.refreshData();
const ss: SortSetting = {
ascending: false,
columnTitle: getSortColumn(this.state.reqSortSetting),
};
this.onChangeSortSetting(ss);
};

hasReqSortOption = (): boolean => {
let found = false;
Object.values(SqlStatsSortOptions).forEach((option: SqlStatsSortType) => {
console.log(option);
if (getSortColumn(option) == this.props.sortSetting.columnTitle) {
found = true;
}
});
return found;
};

renderTransactions(): React.ReactElement {
Expand Down Expand Up @@ -497,6 +529,14 @@ export class TransactionsPage extends React.Component<

const period = timeScaleToString(this.props.timeScale);
const sortSettingLabel = getSortLabel(this.props.reqSortSetting);
const warningSuggestion = this.hasReqSortOption()
? `Update Search Criteria to see the statement fingerprints
sorted on ${getLabel(
this.props.sortSetting.columnTitle as StatisticTableColumnKeys,
"transaction",
)}.`
: "";

return (
<>
<h5 className={`${commonStyles("base-heading")} ${cx("margin-top")}`}>
Expand Down Expand Up @@ -536,6 +576,15 @@ export class TransactionsPage extends React.Component<
<p className={timeScaleStylesCx("time-label")}>
Showing aggregated stats from{" "}
<span className={timeScaleStylesCx("bold")}>{period}</span>
{", "}
<ResultsPerPageLabel
pagination={{
...pagination,
total: transactionsToDisplay.length,
}}
pageName={"Statements"}
search={search}
/>
</p>
</PageConfigItem>
{hasAdminRole && (
Expand All @@ -558,6 +607,14 @@ export class TransactionsPage extends React.Component<
onRemoveFilter={this.onSubmitFilters}
onClearFilters={this.onClearFilters}
/>
{!this.isSortSettingSameAsReqSort() && (
<InlineAlert
intent="warning"
title={`You are viewing a subset (Top ${this.props.limit}) of fingerprints by ${sortSettingLabel}.
${warningSuggestion}`}
className={cx("margin-bottom")}
/>
)}
<TransactionsTable
columns={displayColumns}
transactions={transactionsToDisplay}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -171,6 +171,16 @@ export const TransactionsPageConnected = withRouter(
dispatch(updateTxnsPageLimitAction(limit)),
onChangeReqSort: (sort: SqlStatsSortType) =>
dispatch(updateTxnsPageReqSortAction(sort)),
onApplySearchCriteria: (ts: TimeScale, limit: number, sort: string) =>
dispatch(
analyticsActions.track({
name: "Apply Search Criteria",
page: "Transactions",
tsValue: ts.key,
limitValue: limit,
sortValue: sort,
}),
),
},
activePageProps: mapDispatchToRecentTransactionsPageProps(dispatch),
}),
Expand Down
19 changes: 19 additions & 0 deletions pkg/ui/workspaces/cluster-ui/src/util/sqlActivityConstants.ts
Original file line number Diff line number Diff line change
Expand Up @@ -37,6 +37,25 @@ export function getSortLabel(sort: SqlStatsSortType): string {
}
}

export function getSortColumn(sort: SqlStatsSortType): string {
switch (sort) {
case SqlStatsSortOptions.SERVICE_LAT:
return "time";
case SqlStatsSortOptions.EXECUTION_COUNT:
return "executionCount";
case SqlStatsSortOptions.CPU_TIME:
return "cpu";
case SqlStatsSortOptions.P99_STMTS_ONLY:
return "latencyP99";
case SqlStatsSortOptions.CONTENTION_TIME:
return "contention";
case SqlStatsSortOptions.PCT_RUNTIME:
return "workloadPct";
default:
return "";
}
}

export const stmtRequestSortOptions = Object.values(SqlStatsSortOptions)
.map(sortVal => ({
value: sortVal as SqlStatsSortType,
Expand Down
Loading

0 comments on commit f2e448e

Please sign in to comment.