Skip to content

Commit

Permalink
ui: show data when max size reached
Browse files Browse the repository at this point in the history
Previously, when the sql api returned a max size reached
error, we were just showing the error, but not the data
that was also being returned.

This PR creates a new function to format the return of
the api calls, so when is a max size error it doesn't
throw an error, but still pass that info so we can display
a warning on the pages.

This commit updates the Insights Workload > Statement page
with the new behaviour.
Following PRs will update other usages of the sql api.

Part Of: cockroachdb#96184
Release note (ui change): Still show data on the console
(with a warning) for Statement Insights when we reach a
"max size exceed" error from the sql api.
  • Loading branch information
maryliag committed Feb 16, 2023
1 parent 8ebb9f7 commit a6fd886
Show file tree
Hide file tree
Showing 13 changed files with 94 additions and 26 deletions.
30 changes: 30 additions & 0 deletions pkg/ui/workspaces/cluster-ui/src/api/sqlApi.ts
Original file line number Diff line number Diff line change
Expand Up @@ -60,6 +60,11 @@ export type SqlExecutionErrorMessage = {
source: { file: string; line: number; function: "string" };
};

export type ApiResponse<ResultType> = {
maxSizeReached: boolean;
results: Array<ResultType>;
};

export const SQL_API_PATH = "/api/v2/sql/";

/**
Expand Down Expand Up @@ -159,3 +164,28 @@ export function sqlApiErrorMessage(message: string): string {

return message;
}

function isMaxSizeError(message: string): boolean {
return !!message?.includes("max result size exceeded");
}

export function formatApiResult(
results: Array<any>,
error: SqlExecutionErrorMessage,
errorMessageContext: string,
): ApiResponse<any> {
const maxSizeError = isMaxSizeError(error?.message);

if (error && !maxSizeError) {
throw new Error(
`Error while ${errorMessageContext}: ${sqlApiErrorMessage(
error?.message,
)}`,
);
}

return {
maxSizeReached: maxSizeError,
results: results,
};
}
21 changes: 9 additions & 12 deletions pkg/ui/workspaces/cluster-ui/src/api/stmtInsightsApi.ts
Original file line number Diff line number Diff line change
Expand Up @@ -9,10 +9,11 @@
// licenses/APL.txt.

import {
ApiResponse,
executeInternalSql,
formatApiResult,
LARGE_RESULT_SIZE,
LONG_TIMEOUT,
sqlApiErrorMessage,
SqlExecutionRequest,
sqlResultsAreEmpty,
SqlTxnResult,
Expand Down Expand Up @@ -137,7 +138,7 @@ export const stmtInsightsByTxnExecutionQuery = (id: string): string => `

export async function getStmtInsightsApi(
req?: StmtInsightsReq,
): Promise<StmtInsightEvent[]> {
): Promise<ApiResponse<StmtInsightEvent>> {
const request: SqlExecutionRequest = {
statements: [
{
Expand All @@ -150,21 +151,17 @@ export async function getStmtInsightsApi(
};

const result = await executeInternalSql<StmtInsightsResponseRow>(request);
if (result.error) {
throw new Error(
`Error while retrieving insights information: ${sqlApiErrorMessage(
result.error.message,
)}`,
);
}

if (sqlResultsAreEmpty(result)) {
return [];
return formatApiResult([], result.error, "retrieving insights information");
}

const stmtInsightEvent = formatStmtInsights(result.execution?.txn_results[0]);
await addStmtContentionInfoApi(stmtInsightEvent);
return stmtInsightEvent;
return formatApiResult(
stmtInsightEvent,
result.error,
"retrieving insights information",
);
}

async function addStmtContentionInfoApi(
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -124,7 +124,7 @@ export const StatementInsightDetails: React.FC<
getStmtInsightsApi({ stmtExecutionID: executionID, start, end })
.then(res => {
setInsightDetails({
details: res?.length ? res[0] : null,
details: res?.results?.length ? res.results[0] : null,
loaded: true,
});
})
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -56,6 +56,9 @@ import styles from "src/statementsPage/statementsPage.module.scss";
import sortableTableStyles from "src/sortedtable/sortedtable.module.scss";
import { commonStyles } from "../../../common";
import { useFetchDataWithPolling } from "src/util/hooks";
import { InlineAlert } from "@cockroachlabs/ui-components";
import { insights } from "src/util";
import { Anchor } from "src/anchor";

const cx = classNames.bind(styles);
const sortableTableCx = classNames.bind(sortableTableStyles);
Expand All @@ -72,6 +75,7 @@ export type StatementInsightsViewStateProps = {
isLoading?: boolean;
dropDownSelect?: React.ReactElement;
timeScale?: TimeScale;
maxSizeApiReached?: boolean;
};

export type StatementInsightsViewDispatchProps = {
Expand Down Expand Up @@ -105,6 +109,7 @@ export const StatementInsightsView: React.FC<StatementInsightsViewProps> = ({
setTimeScale,
selectedColumnNames,
dropDownSelect,
maxSizeApiReached,
}: StatementInsightsViewProps) => {
const [pagination, setPagination] = useState<ISortedTablePagination>({
current: 1,
Expand Down Expand Up @@ -317,6 +322,20 @@ export const StatementInsightsView: React.FC<StatementInsightsViewProps> = ({
total={filteredStatements?.length}
onChange={onChangePage}
/>
{maxSizeApiReached && (
<InlineAlert
intent="info"
title={
<>
Not all insights are displayed because the maximum number of
insights was reached in the console.&nbsp;
<Anchor href={insights} target="_blank">
Learn more
</Anchor>
</>
}
/>
)}
</div>
</Loading>
</div>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,7 @@ import {
selectStmtInsights,
selectStmtInsightsError,
selectStmtInsightsLoading,
selectStmtInsightsMaxApiReached,
} from "src/store/insights/statementInsights";
import {
actions as txnInsights,
Expand Down Expand Up @@ -78,6 +79,7 @@ const statementMapStateToProps = (
selectedColumnNames: selectColumns(state),
timeScale: selectTimeScale(state),
isLoading: selectStmtInsightsLoading(state),
maxSizeApiReached: selectStmtInsightsMaxApiReached(state),
});

const TransactionDispatchProps = (
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -11,11 +11,11 @@
import { createSlice, PayloadAction } from "@reduxjs/toolkit";
import { DOMAIN_NAME } from "../../utils";
import moment, { Moment } from "moment";
import { ErrorWithKey, StmtInsightsReq } from "src/api";
import { ApiResponse, ErrorWithKey, StmtInsightsReq } from "src/api";
import { StmtInsightEvent } from "../../../insights";

export type StatementFingerprintInsightsState = {
data: StmtInsightEvent[] | null;
data: ApiResponse<StmtInsightEvent> | null;
lastUpdated: Moment | null;
lastError: Error;
valid: boolean;
Expand All @@ -26,7 +26,7 @@ export type StatementFingerprintInsightsCachedState = {
};

export type FingerprintInsightResponseWithKey = {
response: StmtInsightEvent[];
response: ApiResponse<StmtInsightEvent>;
key: string;
};

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,6 @@ export const selectStatementFingerprintInsights = createSelector(
if (!cachedFingerprintInsights) {
return null;
}
return cachedFingerprintInsights[fingerprintID]?.data;
return cachedFingerprintInsights[fingerprintID]?.data?.results;
},
);
Original file line number Diff line number Diff line change
Expand Up @@ -11,11 +11,11 @@
import { createSlice, PayloadAction } from "@reduxjs/toolkit";
import { DOMAIN_NAME } from "../../utils";
import { StmtInsightEvent } from "src/insights";
import { StmtInsightsReq } from "src/api";
import { ApiResponse, StmtInsightsReq } from "src/api";
import moment from "moment";

export type StmtInsightsState = {
data: StmtInsightEvent[];
data: ApiResponse<StmtInsightEvent>;
lastError: Error;
valid: boolean;
inFlight: boolean;
Expand All @@ -34,7 +34,7 @@ const statementInsightsSlice = createSlice({
name: `${DOMAIN_NAME}/statementInsightsSlice`,
initialState,
reducers: {
received: (state, action: PayloadAction<StmtInsightEvent[]>) => {
received: (state, action: PayloadAction<ApiResponse<StmtInsightEvent>>) => {
state.data = action.payload;
state.valid = true;
state.lastError = null;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -20,11 +20,14 @@ import { selectStatementFingerprintID, selectID } from "src/selectors/common";
import { InsightEnumToLabel, StmtInsightEvent } from "src/insights";

export const selectStmtInsights = (state: AppState): StmtInsightEvent[] =>
state.adminUI.stmtInsights?.data;
state.adminUI.stmtInsights?.data?.results;

export const selectStmtInsightsError = (state: AppState): Error | null =>
state.adminUI.stmtInsights?.lastError;

export const selectStmtInsightsMaxApiReached = (state: AppState): boolean =>
state.adminUI.stmtInsights?.data?.maxSizeReached;

export const selectStmtInsightDetails = createSelector(
selectStmtInsights,
selectID,
Expand Down
8 changes: 6 additions & 2 deletions pkg/ui/workspaces/db-console/src/redux/apiReducers.ts
Original file line number Diff line number Diff line change
Expand Up @@ -557,11 +557,15 @@ export interface APIReducersState {
userSQLRoles: CachedDataReducerState<api.UserSQLRolesResponseMessage>;
hotRanges: PaginatedCachedDataReducerState<api.HotRangesV2ResponseMessage>;
clusterLocks: CachedDataReducerState<clusterUiApi.ClusterLocksResponse>;
stmtInsights: CachedDataReducerState<StmtInsightEvent[]>;
stmtInsights: CachedDataReducerState<
clusterUiApi.ApiResponse<StmtInsightEvent>
>;
txnInsightDetails: KeyedCachedDataReducerState<clusterUiApi.TxnInsightDetailsResponse>;
txnInsights: CachedDataReducerState<TxnInsightEvent[]>;
schemaInsights: CachedDataReducerState<clusterUiApi.InsightRecommendation[]>;
statementFingerprintInsights: KeyedCachedDataReducerState<StmtInsightEvent[]>;
statementFingerprintInsights: KeyedCachedDataReducerState<
clusterUiApi.ApiResponse<StmtInsightEvent>
>;
schedules: KeyedCachedDataReducerState<clusterUiApi.Schedules>;
schedule: KeyedCachedDataReducerState<clusterUiApi.Schedule>;
snapshots: KeyedCachedDataReducerState<clusterUiApi.ListTracingSnapshotsResponse>;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -70,8 +70,19 @@ const selectTxnInsight = createSelector(
},
);

export const selectStmtInsights = (state: AdminUIState) =>
state.cachedData.stmtInsights?.data;
export const selectStmtInsights = (state: AdminUIState) => {
return state.cachedData.stmtInsights?.data
? state.cachedData.stmtInsights.data["results"]
: null;
};

export const selectStmtInsightsMaxApiReached = (
state: AdminUIState,
): boolean => {
return state.cachedData.stmtInsights?.data
? state.cachedData.stmtInsights?.data["maxSizeReached"]
: false;
};

export const selectTxnInsightDetails = createSelector(
selectTxnInsight,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -30,6 +30,7 @@ import {
selectStmtInsightsLoading,
selectTransactionInsightsLoading,
selectInsightTypes,
selectStmtInsightsMaxApiReached,
} from "src/views/insights/insightsSelectors";
import { bindActionCreators } from "redux";
import { LocalSetting } from "src/redux/localsettings";
Expand Down Expand Up @@ -75,6 +76,7 @@ const statementMapStateToProps = (
insightStatementColumnsLocalSetting.selectorToArray(state),
timeScale: selectTimeScale(state),
isLoading: selectStmtInsightsLoading(state),
maxSizeApiReached: selectStmtInsightsMaxApiReached(state),
});

const TransactionDispatchProps = {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -108,7 +108,7 @@ const selectStatementFingerprintInsights = createSelector(
(_state: AdminUIState, props: RouteComponentProps): string =>
getMatchParamByName(props.match, statementAttr),
(cachedFingerprintInsights, fingerprintID) => {
return cachedFingerprintInsights[fingerprintID]?.data;
return cachedFingerprintInsights[fingerprintID]?.data?.results;
},
);

Expand Down

0 comments on commit a6fd886

Please sign in to comment.