From ca9787d4ddee70e20cbda3afb3570f9dfe816297 Mon Sep 17 00:00:00 2001 From: Eric Harmeling Date: Fri, 3 Jun 2022 19:22:00 -0400 Subject: [PATCH] server, ui: handle null plan gist in getStatementDetailsPerPlanHash This commit adds a new function that handles null plan gists in `getStatementDetailsPerPlanHash`. The commit also adds some logic to hide null plan gists in the Statement Details page. Fixes #82095. For more details, see https://github.com/cockroachdb/cockroach/issues/82095#issuecomment-1146266191. Release note: None --- pkg/server/combined_statement_stats.go | 7 +++++-- pkg/sql/sem/tree/datum.go | 10 ++++++++++ .../src/statementDetails/planDetails/planDetails.tsx | 4 +++- 3 files changed, 18 insertions(+), 3 deletions(-) diff --git a/pkg/server/combined_statement_stats.go b/pkg/server/combined_statement_stats.go index 91871b4cc7b6..fb8bec52ba65 100644 --- a/pkg/server/combined_statement_stats.go +++ b/pkg/server/combined_statement_stats.go @@ -729,8 +729,11 @@ func getStatementDetailsPerPlanHash( if planHash, err = sqlstatsutil.DatumToUint64(row[0]); err != nil { return nil, serverError(ctx, err) } - planGist := string(tree.MustBeDString(row[1])) - explainPlan := getExplainPlanFromGist(ctx, ie, planGist) + planGist := string(tree.MustBeDStringOrDNull(row[1])) + var explainPlan string + if planGist != "" { + explainPlan = getExplainPlanFromGist(ctx, ie, planGist) + } var metadata roachpb.CollectedStatementStatistics var aggregatedMetadata roachpb.AggregatedStatementMetadata diff --git a/pkg/sql/sem/tree/datum.go b/pkg/sql/sem/tree/datum.go index c26f62156090..d717b31554fa 100644 --- a/pkg/sql/sem/tree/datum.go +++ b/pkg/sql/sem/tree/datum.go @@ -1220,6 +1220,16 @@ func MustBeDString(e Expr) DString { return i } +// MustBeDStringOrDNull attempts to retrieve a DString or DNull from an Expr, panicking if the +// assertion fails. +func MustBeDStringOrDNull(e Expr) DString { + i, ok := AsDString(e) + if !ok && e != DNull { + panic(errors.AssertionFailedf("expected *DString or DNull, found %T", e)) + } + return i +} + // ResolvedType implements the TypedExpr interface. func (*DString) ResolvedType() *types.T { return types.String diff --git a/pkg/ui/workspaces/cluster-ui/src/statementDetails/planDetails/planDetails.tsx b/pkg/ui/workspaces/cluster-ui/src/statementDetails/planDetails/planDetails.tsx index 42693aa2a223..e7e04f8aa05b 100644 --- a/pkg/ui/workspaces/cluster-ui/src/statementDetails/planDetails/planDetails.tsx +++ b/pkg/ui/workspaces/cluster-ui/src/statementDetails/planDetails/planDetails.tsx @@ -57,6 +57,8 @@ function renderExplainPlan( plan: PlanHashStats, backToPlanTable: () => void, ): React.ReactElement { + const explainPlan = + plan.explain_plan === "" ? "unavailable" : plan.explain_plan; return (
@@ -70,7 +72,7 @@ function renderExplainPlan( > All Plans - +
); }