diff --git a/docs/generated/settings/settings-for-tenants.txt b/docs/generated/settings/settings-for-tenants.txt index f916e12ddff2..4443f195c9ee 100644 --- a/docs/generated/settings/settings-for-tenants.txt +++ b/docs/generated/settings/settings-for-tenants.txt @@ -283,4 +283,4 @@ trace.jaeger.agent string the address of a Jaeger agent to receive traces using trace.opentelemetry.collector string address of an OpenTelemetry trace collector to receive traces using the otel gRPC protocol, as :. If no port is specified, 4317 will be used. trace.span_registry.enabled boolean true if set, ongoing traces can be seen at https:///#/debug/tracez trace.zipkin.collector string the address of a Zipkin instance to receive traces, as :. If no port is specified, 9411 will be used. -version version 22.1-30 set the active cluster version in the format '.' +version version 22.1-32 set the active cluster version in the format '.' diff --git a/docs/generated/settings/settings.html b/docs/generated/settings/settings.html index 31db96f77148..9376a4c93c4f 100644 --- a/docs/generated/settings/settings.html +++ b/docs/generated/settings/settings.html @@ -214,6 +214,6 @@ trace.opentelemetry.collectorstringaddress of an OpenTelemetry trace collector to receive traces using the otel gRPC protocol, as :. If no port is specified, 4317 will be used. trace.span_registry.enabledbooleantrueif set, ongoing traces can be seen at https:///#/debug/tracez trace.zipkin.collectorstringthe address of a Zipkin instance to receive traces, as :. If no port is specified, 9411 will be used. -versionversion22.1-30set the active cluster version in the format '.' +versionversion22.1-32set the active cluster version in the format '.' diff --git a/pkg/ccl/logictestccl/testdata/logic_test/crdb_internal_tenant b/pkg/ccl/logictestccl/testdata/logic_test/crdb_internal_tenant index 9b80cf000c7a..c94dbcbf5523 100644 --- a/pkg/ccl/logictestccl/testdata/logic_test/crdb_internal_tenant +++ b/pkg/ccl/logictestccl/testdata/logic_test/crdb_internal_tenant @@ -195,10 +195,10 @@ SELECT * FROM crdb_internal.leases WHERE node_id < 0 ---- node_id table_id name parent_id expiration deleted -query ITTTTTIIITRRRRRRRRRRRRRRRRRRRRRRRRRRBBTTTT colnames +query ITTTTTIIITRRRRRRRRRRRRRRRRRRRRRRRRRRBBTTTTT colnames SELECT * FROM crdb_internal.node_statement_statistics WHERE node_id < 0 ---- -node_id application_name flags statement_id key anonymized count first_attempt_count max_retries last_error rows_avg rows_var parse_lat_avg parse_lat_var plan_lat_avg plan_lat_var run_lat_avg run_lat_var service_lat_avg service_lat_var overhead_lat_avg overhead_lat_var bytes_read_avg bytes_read_var rows_read_avg rows_read_var network_bytes_avg network_bytes_var network_msgs_avg network_msgs_var max_mem_usage_avg max_mem_usage_var max_disk_usage_avg max_disk_usage_var contention_time_avg contention_time_var implicit_txn full_scan sample_plan database_name exec_node_ids txn_fingerprint_id +node_id application_name flags statement_id key anonymized count first_attempt_count max_retries last_error rows_avg rows_var parse_lat_avg parse_lat_var plan_lat_avg plan_lat_var run_lat_avg run_lat_var service_lat_avg service_lat_var overhead_lat_avg overhead_lat_var bytes_read_avg bytes_read_var rows_read_avg rows_read_var network_bytes_avg network_bytes_var network_msgs_avg network_msgs_var max_mem_usage_avg max_mem_usage_var max_disk_usage_avg max_disk_usage_var contention_time_avg contention_time_var implicit_txn full_scan sample_plan database_name exec_node_ids txn_fingerprint_id index_recommendations query ITTTIIRRRRRRRRRRRRRRRRRR colnames SELECT * FROM crdb_internal.node_transaction_statistics WHERE node_id < 0 diff --git a/pkg/clusterversion/cockroach_versions.go b/pkg/clusterversion/cockroach_versions.go index d4bb66fa63ba..497cc32ac7be 100644 --- a/pkg/clusterversion/cockroach_versions.go +++ b/pkg/clusterversion/cockroach_versions.go @@ -331,6 +331,9 @@ const ( AlterSystemSQLInstancesAddLocality // SystemExternalConnectionsTable adds system.external_connections table. SystemExternalConnectionsTable + // AlterSystemStatementStatisticsAddIndexRecommendations adds an + // index_recommendations column to the system.statement_statistics table. + AlterSystemStatementStatisticsAddIndexRecommendations // ************************************************* // Step (1): Add new versions here. @@ -572,6 +575,10 @@ var versionsSingleton = keyedVersions{ Key: SystemExternalConnectionsTable, Version: roachpb.Version{Major: 22, Minor: 1, Internal: 30}, }, + { + Key: AlterSystemStatementStatisticsAddIndexRecommendations, + Version: roachpb.Version{Major: 22, Minor: 1, Internal: 32}, + }, // ************************************************* // Step (2): Add new versions here. diff --git a/pkg/clusterversion/key_string.go b/pkg/clusterversion/key_string.go index 2e253b5f9d57..73c5f7405ace 100644 --- a/pkg/clusterversion/key_string.go +++ b/pkg/clusterversion/key_string.go @@ -59,11 +59,12 @@ func _() { _ = x[EnablePredicateProjectionChangefeed-48] _ = x[AlterSystemSQLInstancesAddLocality-49] _ = x[SystemExternalConnectionsTable-50] + _ = x[AlterSystemStatementStatisticsAddIndexRecommendations-51] } -const _Key_name = "V21_2Start22_1PebbleFormatBlockPropertyCollectorProbeRequestPublicSchemasWithDescriptorsEnsureSpanConfigReconciliationEnsureSpanConfigSubscriptionEnableSpanConfigStoreSCRAMAuthenticationUnsafeLossOfQuorumRecoveryRangeLogAlterSystemProtectedTimestampAddColumnEnableProtectedTimestampsForTenantDeleteCommentsWithDroppedIndexesRemoveIncompatibleDatabasePrivilegesAddRaftAppliedIndexTermMigrationPostAddRaftAppliedIndexTermMigrationDontProposeWriteTimestampForLeaseTransfersEnablePebbleFormatVersionBlockPropertiesMVCCIndexBackfillerEnableLeaseHolderRemovalLooselyCoupledRaftLogTruncationChangefeedIdlenessBackupDoesNotOverwriteLatestAndCheckpointEnableDeclarativeSchemaChangerRowLevelTTLPebbleFormatSplitUserKeysMarkedIncrementalBackupSubdirEnableNewStoreRebalancerClusterLocksVirtualTableAutoStatsTableSettingsSuperRegionsEnableNewChangefeedOptionsSpanCountTablePreSeedSpanCountTableSeedSpanCountTableV22_1Start22_2LocalTimestampsPebbleFormatSplitUserKeysMarkedCompactedEnsurePebbleFormatVersionRangeKeysEnablePebbleFormatVersionRangeKeysTrigramInvertedIndexesRemoveGrantPrivilegeMVCCRangeTombstonesUpgradeSequenceToBeReferencedByIDSampledStmtDiagReqsAddSSTableTombstonesSystemPrivilegesTableEnablePredicateProjectionChangefeedAlterSystemSQLInstancesAddLocalitySystemExternalConnectionsTable" +const _Key_name = "V21_2Start22_1PebbleFormatBlockPropertyCollectorProbeRequestPublicSchemasWithDescriptorsEnsureSpanConfigReconciliationEnsureSpanConfigSubscriptionEnableSpanConfigStoreSCRAMAuthenticationUnsafeLossOfQuorumRecoveryRangeLogAlterSystemProtectedTimestampAddColumnEnableProtectedTimestampsForTenantDeleteCommentsWithDroppedIndexesRemoveIncompatibleDatabasePrivilegesAddRaftAppliedIndexTermMigrationPostAddRaftAppliedIndexTermMigrationDontProposeWriteTimestampForLeaseTransfersEnablePebbleFormatVersionBlockPropertiesMVCCIndexBackfillerEnableLeaseHolderRemovalLooselyCoupledRaftLogTruncationChangefeedIdlenessBackupDoesNotOverwriteLatestAndCheckpointEnableDeclarativeSchemaChangerRowLevelTTLPebbleFormatSplitUserKeysMarkedIncrementalBackupSubdirEnableNewStoreRebalancerClusterLocksVirtualTableAutoStatsTableSettingsSuperRegionsEnableNewChangefeedOptionsSpanCountTablePreSeedSpanCountTableSeedSpanCountTableV22_1Start22_2LocalTimestampsPebbleFormatSplitUserKeysMarkedCompactedEnsurePebbleFormatVersionRangeKeysEnablePebbleFormatVersionRangeKeysTrigramInvertedIndexesRemoveGrantPrivilegeMVCCRangeTombstonesUpgradeSequenceToBeReferencedByIDSampledStmtDiagReqsAddSSTableTombstonesSystemPrivilegesTableEnablePredicateProjectionChangefeedAlterSystemSQLInstancesAddLocalitySystemExternalConnectionsTableAlterSystemStatementStatisticsAddIndexRecommendations" -var _Key_index = [...]uint16{0, 5, 14, 48, 60, 88, 118, 146, 167, 186, 220, 258, 292, 324, 360, 392, 428, 470, 510, 529, 553, 584, 602, 643, 673, 684, 715, 738, 762, 786, 808, 820, 846, 860, 881, 899, 904, 913, 928, 968, 1002, 1036, 1058, 1078, 1097, 1130, 1149, 1169, 1190, 1225, 1259, 1289} +var _Key_index = [...]uint16{0, 5, 14, 48, 60, 88, 118, 146, 167, 186, 220, 258, 292, 324, 360, 392, 428, 470, 510, 529, 553, 584, 602, 643, 673, 684, 715, 738, 762, 786, 808, 820, 846, 860, 881, 899, 904, 913, 928, 968, 1002, 1036, 1058, 1078, 1097, 1130, 1149, 1169, 1190, 1225, 1259, 1289, 1342} func (i Key) String() string { if i < 0 || i >= Key(len(_Key_index)-1) { diff --git a/pkg/roachpb/app_stats.go b/pkg/roachpb/app_stats.go index fd511eb1790f..4078aa35cfb0 100644 --- a/pkg/roachpb/app_stats.go +++ b/pkg/roachpb/app_stats.go @@ -155,6 +155,7 @@ func (s *StatementStatistics) Add(other *StatementStatistics) { s.RowsWritten.Add(other.RowsWritten, s.Count, other.Count) s.Nodes = util.CombineUniqueInt64(s.Nodes, other.Nodes) s.PlanGists = util.CombineUniqueString(s.PlanGists, other.PlanGists) + s.IndexRecommendations = util.CombineUniqueString(s.IndexRecommendations, other.IndexRecommendations) s.ExecStats.Add(other.ExecStats) diff --git a/pkg/roachpb/app_stats.proto b/pkg/roachpb/app_stats.proto index 3b1b4f1653b9..a75efcf4e90e 100644 --- a/pkg/roachpb/app_stats.proto +++ b/pkg/roachpb/app_stats.proto @@ -104,12 +104,15 @@ message StatementStatistics { // Nodes is the ordered list of nodes ids on which the statement was executed. repeated int64 nodes = 24; - // plan_gists is list of a compressed version of plan that can be converted (lossily) + // plan_gists is the list of a compressed version of plan that can be converted (lossily) // back into a logical plan. // Each statement contain only one plan gist, but the same statement fingerprint id // can contain more than one value. repeated string plan_gists = 26; + // index_recommendations is the list of index recommendations generated for the statement fingerprint. + repeated string index_recommendations = 27; + // Note: be sure to update `sql/app_stats.go` when adding/removing fields here! reserved 13, 14, 17, 18, 19, 20; diff --git a/pkg/sql/catalog/systemschema/system.go b/pkg/sql/catalog/systemschema/system.go index 02f3b090d2e2..36c7761c3aea 100644 --- a/pkg/sql/catalog/systemschema/system.go +++ b/pkg/sql/catalog/systemschema/system.go @@ -481,6 +481,8 @@ CREATE TABLE system.statement_statistics ( mod(fnv32(crdb_internal.datums_to_bytes(aggregated_ts, app_name, fingerprint_id, node_id, plan_hash, transaction_fingerprint_id)), 8:::INT8) ) STORED, + index_recommendations STRING[] NOT NULL DEFAULT (array[]::STRING[]), + CONSTRAINT "primary" PRIMARY KEY (aggregated_ts, fingerprint_id, transaction_fingerprint_id, plan_hash, app_name, node_id) USING HASH WITH (bucket_count=8), INDEX "fingerprint_stats_idx" (fingerprint_id, transaction_fingerprint_id), @@ -495,7 +497,8 @@ CREATE TABLE system.statement_statistics ( agg_interval, metadata, statistics, - plan + plan, + index_recommendations ) ) ` @@ -1983,6 +1986,8 @@ var ( }, )) + defaultIndexRec = "ARRAY[]:::STRING[]" + // StatementStatisticsTable is the descriptor for the SQL statement stats table. // It stores statistics for statement fingerprints. StatementStatisticsTable = registerSystemTable( @@ -2009,6 +2014,7 @@ var ( ComputeExpr: &sqlStmtHashComputeExpr, Hidden: true, }, + {Name: "index_recommendations", ID: 12, Type: types.StringArray, Nullable: false, DefaultExpr: &defaultIndexRec}, }, []descpb.ColumnFamilyDescriptor{ { @@ -2017,9 +2023,9 @@ var ( ColumnNames: []string{ "crdb_internal_aggregated_ts_app_name_fingerprint_id_node_id_plan_hash_transaction_fingerprint_id_shard_8", "aggregated_ts", "fingerprint_id", "transaction_fingerprint_id", "plan_hash", "app_name", "node_id", - "agg_interval", "metadata", "statistics", "plan", + "agg_interval", "metadata", "statistics", "plan", "index_recommendations", }, - ColumnIDs: []descpb.ColumnID{11, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10}, + ColumnIDs: []descpb.ColumnID{11, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 12}, DefaultColumnID: 0, }, }, diff --git a/pkg/sql/catalog/systemschema_test/testdata/bootstrap b/pkg/sql/catalog/systemschema_test/testdata/bootstrap index 1da4f56312d4..1b1e73c8fb7d 100644 --- a/pkg/sql/catalog/systemschema_test/testdata/bootstrap +++ b/pkg/sql/catalog/systemschema_test/testdata/bootstrap @@ -306,6 +306,7 @@ CREATE TABLE public.statement_statistics ( statistics JSONB NOT NULL, plan JSONB NOT NULL, crdb_internal_aggregated_ts_app_name_fingerprint_id_node_id_plan_hash_transaction_fingerprint_id_shard_8 INT4 NOT VISIBLE NOT NULL AS (mod(fnv32(crdb_internal.datums_to_bytes(aggregated_ts, app_name, fingerprint_id, node_id, plan_hash, transaction_fingerprint_id)), 8:::INT8)) STORED, + index_recommendations STRING[] NOT NULL DEFAULT ARRAY[]:::STRING[], CONSTRAINT "primary" PRIMARY KEY (aggregated_ts ASC, fingerprint_id ASC, transaction_fingerprint_id ASC, plan_hash ASC, app_name ASC, node_id ASC) USING HASH WITH (bucket_count=8), INDEX fingerprint_stats_idx (fingerprint_id ASC, transaction_fingerprint_id ASC) ); diff --git a/pkg/sql/crdb_internal.go b/pkg/sql/crdb_internal.go index 5c1f935a5fd8..85ef4733421a 100644 --- a/pkg/sql/crdb_internal.go +++ b/pkg/sql/crdb_internal.go @@ -1103,7 +1103,8 @@ CREATE TABLE crdb_internal.node_statement_statistics ( sample_plan JSONB, database_name STRING NOT NULL, exec_node_ids INT[] NOT NULL, - txn_fingerprint_id STRING + txn_fingerprint_id STRING, + index_recommendations STRING[] NOT NULL )`, populate: func(ctx context.Context, p *planner, _ catalog.DatabaseDescriptor, addRow func(...tree.Datum) error) error { hasViewActivityOrViewActivityRedacted, err := p.HasViewActivityOrViewActivityRedactedRole(ctx) @@ -1156,6 +1157,13 @@ CREATE TABLE crdb_internal.node_statement_statistics ( } + indexRecommendations := tree.NewDArray(types.String) + for _, recommendation := range stats.Stats.IndexRecommendations { + if err := indexRecommendations.Append(tree.NewDString(recommendation)); err != nil { + return err + } + } + err := addRow( tree.NewDInt(tree.DInt(nodeID)), // node_id tree.NewDString(stats.Key.App), // application_name @@ -1199,6 +1207,7 @@ CREATE TABLE crdb_internal.node_statement_statistics ( tree.NewDString(stats.Key.Database), // database_name execNodeIDs, // exec_node_ids txnFingerprintID, // txn_fingerprint_id + indexRecommendations, // index_recommendations ) if err != nil { return err @@ -5371,7 +5380,8 @@ CREATE TABLE crdb_internal.cluster_statement_statistics ( metadata JSONB NOT NULL, statistics JSONB NOT NULL, sampled_plan JSONB NOT NULL, - aggregation_interval INTERVAL NOT NULL + aggregation_interval INTERVAL NOT NULL, + index_recommendations STRING[] NOT NULL );`, generator: func(ctx context.Context, p *planner, db catalog.DatabaseDescriptor, stopper *stop.Stopper) (virtualTableGenerator, cleanupFunc, error) { // TODO(azhng): we want to eventually implement memory accounting within the @@ -5402,7 +5412,7 @@ CREATE TABLE crdb_internal.cluster_statement_statistics ( curAggTs := s.ComputeAggregatedTs() aggInterval := s.GetAggregationInterval() - row := make(tree.Datums, 8 /* number of columns for this virtual table */) + row := make(tree.Datums, 9 /* number of columns for this virtual table */) worker := func(ctx context.Context, pusher rowPusher) error { return memSQLStats.IterateStatementStats(ctx, &sqlstats.IteratorOptions{ SortedAppNames: true, @@ -5437,6 +5447,13 @@ CREATE TABLE crdb_internal.cluster_statement_statistics ( duration.MakeDuration(aggInterval.Nanoseconds(), 0, 0), types.DefaultIntervalTypeMetadata) + indexRecommendations := tree.NewDArray(types.String) + for _, recommendation := range statistics.Stats.IndexRecommendations { + if err := indexRecommendations.Append(tree.NewDString(recommendation)); err != nil { + return err + } + } + row = row[:0] row = append(row, aggregatedTs, // aggregated_ts @@ -5448,6 +5465,7 @@ CREATE TABLE crdb_internal.cluster_statement_statistics ( tree.NewDJSON(statisticsJSON), // statistics tree.NewDJSON(plan), // plan aggInterval, // aggregation_interval + indexRecommendations, // index_recommendations ) return pusher.pushRow(row...) @@ -5473,7 +5491,8 @@ SELECT max(metadata) as metadata, crdb_internal.merge_statement_stats(array_agg(statistics)), max(sampled_plan), - aggregation_interval + aggregation_interval, + array_remove(array_agg(index_rec), NULL) AS index_recommendations FROM ( SELECT aggregated_ts, @@ -5484,7 +5503,8 @@ FROM ( metadata, statistics, sampled_plan, - aggregation_interval + aggregation_interval, + index_recommendations FROM crdb_internal.cluster_statement_statistics UNION ALL @@ -5497,10 +5517,12 @@ FROM ( metadata, statistics, plan, - agg_interval + agg_interval, + index_recommendations FROM system.statement_statistics ) +LEFT JOIN LATERAL unnest(index_recommendations) AS index_rec ON true GROUP BY aggregated_ts, fingerprint_id, @@ -5518,6 +5540,7 @@ GROUP BY {Name: "statistics", Typ: types.Jsonb}, {Name: "sampled_plan", Typ: types.Jsonb}, {Name: "aggregation_interval", Typ: types.Interval}, + {Name: "index_recommendations", Typ: types.StringArray}, }, } diff --git a/pkg/sql/executor_statement_metrics.go b/pkg/sql/executor_statement_metrics.go index 73751375cad1..9f7316663405 100644 --- a/pkg/sql/executor_statement_metrics.go +++ b/pkg/sql/executor_statement_metrics.go @@ -168,23 +168,24 @@ func (ex *connExecutor) recordStatementSummary( } recordedStmtStats := sqlstats.RecordedStmtStats{ - SessionID: ex.sessionID, - StatementID: planner.stmt.QueryID, - AutoRetryCount: automaticRetryCount, - RowsAffected: rowsAffected, - ParseLatency: parseLat, - PlanLatency: planLat, - RunLatency: runLat, - ServiceLatency: svcLat, - OverheadLatency: execOverhead, - BytesRead: stats.bytesRead, - RowsRead: stats.rowsRead, - RowsWritten: stats.rowsWritten, - Nodes: getNodesFromPlanner(planner), - StatementType: stmt.AST.StatementType(), - Plan: planner.instrumentation.PlanForStats(ctx), - PlanGist: planner.instrumentation.planGist.String(), - StatementError: stmtErr, + SessionID: ex.sessionID, + StatementID: planner.stmt.QueryID, + AutoRetryCount: automaticRetryCount, + RowsAffected: rowsAffected, + ParseLatency: parseLat, + PlanLatency: planLat, + RunLatency: runLat, + ServiceLatency: svcLat, + OverheadLatency: execOverhead, + BytesRead: stats.bytesRead, + RowsRead: stats.rowsRead, + RowsWritten: stats.rowsWritten, + Nodes: getNodesFromPlanner(planner), + StatementType: stmt.AST.StatementType(), + Plan: planner.instrumentation.PlanForStats(ctx), + PlanGist: planner.instrumentation.planGist.String(), + StatementError: stmtErr, + IndexRecommendations: planner.instrumentation.indexRecommendations, } stmtFingerprintID, err := diff --git a/pkg/sql/logictest/testdata/logic_test/crdb_internal b/pkg/sql/logictest/testdata/logic_test/crdb_internal index f0539bfe679a..bd9c74e097d9 100644 --- a/pkg/sql/logictest/testdata/logic_test/crdb_internal +++ b/pkg/sql/logictest/testdata/logic_test/crdb_internal @@ -315,10 +315,10 @@ SELECT * FROM crdb_internal.leases WHERE node_id < 0 ---- node_id table_id name parent_id expiration deleted -query ITTTTTIIITRRRRRRRRRRRRRRRRRRRRRRRRRRBBTTTT colnames +query ITTTTTIIITRRRRRRRRRRRRRRRRRRRRRRRRRRBBTTTTT colnames SELECT * FROM crdb_internal.node_statement_statistics WHERE node_id < 0 ---- -node_id application_name flags statement_id key anonymized count first_attempt_count max_retries last_error rows_avg rows_var parse_lat_avg parse_lat_var plan_lat_avg plan_lat_var run_lat_avg run_lat_var service_lat_avg service_lat_var overhead_lat_avg overhead_lat_var bytes_read_avg bytes_read_var rows_read_avg rows_read_var network_bytes_avg network_bytes_var network_msgs_avg network_msgs_var max_mem_usage_avg max_mem_usage_var max_disk_usage_avg max_disk_usage_var contention_time_avg contention_time_var implicit_txn full_scan sample_plan database_name exec_node_ids txn_fingerprint_id +node_id application_name flags statement_id key anonymized count first_attempt_count max_retries last_error rows_avg rows_var parse_lat_avg parse_lat_var plan_lat_avg plan_lat_var run_lat_avg run_lat_var service_lat_avg service_lat_var overhead_lat_avg overhead_lat_var bytes_read_avg bytes_read_var rows_read_avg rows_read_var network_bytes_avg network_bytes_var network_msgs_avg network_msgs_var max_mem_usage_avg max_mem_usage_var max_disk_usage_avg max_disk_usage_var contention_time_avg contention_time_var implicit_txn full_scan sample_plan database_name exec_node_ids txn_fingerprint_id index_recommendations query ITTTIIRRRRRRRRRRRRRRRRRR colnames SELECT * FROM crdb_internal.node_transaction_statistics WHERE node_id < 0 diff --git a/pkg/sql/logictest/testdata/logic_test/create_statements b/pkg/sql/logictest/testdata/logic_test/create_statements index 9e5219eb0e2f..8fb611a52738 100644 --- a/pkg/sql/logictest/testdata/logic_test/create_statements +++ b/pkg/sql/logictest/testdata/logic_test/create_statements @@ -381,7 +381,8 @@ CREATE TABLE crdb_internal.cluster_statement_statistics ( metadata JSONB NOT NULL, statistics JSONB NOT NULL, sampled_plan JSONB NOT NULL, - aggregation_interval INTERVAL NOT NULL + aggregation_interval INTERVAL NOT NULL, + index_recommendations STRING[] NOT NULL ) CREATE TABLE crdb_internal.cluster_statement_statistics ( aggregated_ts TIMESTAMPTZ NOT NULL, fingerprint_id BYTES NOT NULL, @@ -391,7 +392,8 @@ CREATE TABLE crdb_internal.cluster_statement_statistics ( metadata JSONB NOT NULL, statistics JSONB NOT NULL, sampled_plan JSONB NOT NULL, - aggregation_interval INTERVAL NOT NULL + aggregation_interval INTERVAL NOT NULL, + index_recommendations STRING[] NOT NULL ) {} {} CREATE TABLE crdb_internal.cluster_transaction_statistics ( aggregated_ts TIMESTAMPTZ NOT NULL, @@ -1041,7 +1043,8 @@ CREATE TABLE crdb_internal.node_statement_statistics ( sample_plan JSONB NULL, database_name STRING NOT NULL, exec_node_ids INT8[] NOT NULL, - txn_fingerprint_id STRING NULL + txn_fingerprint_id STRING NULL, + index_recommendations STRING[] NOT NULL ) CREATE TABLE crdb_internal.node_statement_statistics ( node_id INT8 NOT NULL, application_name STRING NOT NULL, @@ -1084,7 +1087,8 @@ CREATE TABLE crdb_internal.node_statement_statistics ( sample_plan JSONB NULL, database_name STRING NOT NULL, exec_node_ids INT8[] NOT NULL, - txn_fingerprint_id STRING NULL + txn_fingerprint_id STRING NULL, + index_recommendations STRING[] NOT NULL ) {} {} CREATE TABLE crdb_internal.node_transaction_statistics ( node_id INT8 NOT NULL, @@ -1401,7 +1405,8 @@ CREATE VIEW crdb_internal.statement_statistics ( metadata, statistics, sampled_plan, - aggregation_interval + aggregation_interval, + index_recommendations ) AS SELECT aggregated_ts, fingerprint_id, @@ -1411,7 +1416,8 @@ CREATE VIEW crdb_internal.statement_statistics ( max(metadata) AS metadata, crdb_internal.merge_statement_stats(array_agg(statistics)), max(sampled_plan), - aggregation_interval + aggregation_interval, + array_remove(array_agg(index_rec), NULL) AS index_recommendations FROM ( SELECT @@ -1423,7 +1429,8 @@ CREATE VIEW crdb_internal.statement_statistics ( metadata, statistics, sampled_plan, - aggregation_interval + aggregation_interval, + index_recommendations FROM crdb_internal.cluster_statement_statistics UNION ALL @@ -1436,10 +1443,12 @@ CREATE VIEW crdb_internal.statement_statistics ( metadata, statistics, plan, - agg_interval + agg_interval, + index_recommendations FROM system.statement_statistics ) + LEFT JOIN LATERAL unnest(index_recommendations) AS index_rec ON true GROUP BY aggregated_ts, fingerprint_id, @@ -1455,7 +1464,8 @@ CREATE VIEW crdb_internal.statement_statistics ( metadata, statistics, sampled_plan, - aggregation_interval + aggregation_interval, + index_recommendations ) AS SELECT aggregated_ts, fingerprint_id, @@ -1465,7 +1475,8 @@ CREATE VIEW crdb_internal.statement_statistics ( max(metadata) AS metadata, crdb_internal.merge_statement_stats(array_agg(statistics)), max(sampled_plan), - aggregation_interval + aggregation_interval, + array_remove(array_agg(index_rec), NULL) AS index_recommendations FROM ( SELECT @@ -1477,7 +1488,8 @@ CREATE VIEW crdb_internal.statement_statistics ( metadata, statistics, sampled_plan, - aggregation_interval + aggregation_interval, + index_recommendations FROM crdb_internal.cluster_statement_statistics UNION ALL @@ -1490,10 +1502,12 @@ CREATE VIEW crdb_internal.statement_statistics ( metadata, statistics, plan, - agg_interval + agg_interval, + index_recommendations FROM system.statement_statistics ) + LEFT JOIN LATERAL unnest(index_recommendations) AS index_rec ON true GROUP BY aggregated_ts, fingerprint_id, diff --git a/pkg/sql/logictest/testdata/logic_test/information_schema b/pkg/sql/logictest/testdata/logic_test/information_schema index adbebfb05374..515d7b5e1be0 100644 --- a/pkg/sql/logictest/testdata/logic_test/information_schema +++ b/pkg/sql/logictest/testdata/logic_test/information_schema @@ -1615,6 +1615,7 @@ system public check_sampling_probability system public primary system public statement_diagnostics_requests PRIMARY KEY NO NO system public 630200280_42_10_not_null system public statement_statistics CHECK NO NO system public 630200280_42_11_not_null system public statement_statistics CHECK NO NO +system public 630200280_42_12_not_null system public statement_statistics CHECK NO NO system public 630200280_42_1_not_null system public statement_statistics CHECK NO NO system public 630200280_42_2_not_null system public statement_statistics CHECK NO NO system public 630200280_42_3_not_null system public statement_statistics CHECK NO NO @@ -1801,6 +1802,7 @@ system public 630200280_41_1_not_null system public 630200280_41_2_not_null secret IS NOT NULL system public 630200280_41_3_not_null expiration IS NOT NULL system public 630200280_42_10_not_null plan IS NOT NULL +system public 630200280_42_12_not_null index_recommendations IS NOT NULL system public 630200280_42_1_not_null aggregated_ts IS NOT NULL system public 630200280_42_2_not_null fingerprint_id IS NOT NULL system public 630200280_42_3_not_null transaction_fingerprint_id IS NOT NULL @@ -2196,6 +2198,7 @@ system public statement_statistics aggregated_ts system public statement_statistics app_name 5 system public statement_statistics crdb_internal_aggregated_ts_app_name_fingerprint_id_node_id_plan_hash_transaction_fingerprint_id_shard_8 11 system public statement_statistics fingerprint_id 2 +system public statement_statistics index_recommendations 12 system public statement_statistics metadata 8 system public statement_statistics node_id 6 system public statement_statistics plan 10 diff --git a/pkg/sql/sqlstats/persistedsqlstats/BUILD.bazel b/pkg/sql/sqlstats/persistedsqlstats/BUILD.bazel index 816f6bc5206f..8238609271e7 100644 --- a/pkg/sql/sqlstats/persistedsqlstats/BUILD.bazel +++ b/pkg/sql/sqlstats/persistedsqlstats/BUILD.bazel @@ -41,6 +41,8 @@ go_library( "//pkg/sql/sqlstats/sslocal", "//pkg/sql/sqlstats/ssmemstorage", "//pkg/sql/sqlutil", + "//pkg/sql/types", + "//pkg/util", "//pkg/util/log", "//pkg/util/metric", "//pkg/util/mon", diff --git a/pkg/sql/sqlstats/persistedsqlstats/flush.go b/pkg/sql/sqlstats/persistedsqlstats/flush.go index 34e9de79f80a..64d0045c4f53 100644 --- a/pkg/sql/sqlstats/persistedsqlstats/flush.go +++ b/pkg/sql/sqlstats/persistedsqlstats/flush.go @@ -21,6 +21,7 @@ import ( "github.com/cockroachdb/cockroach/pkg/sql/sessiondata" "github.com/cockroachdb/cockroach/pkg/sql/sqlstats" "github.com/cockroachdb/cockroach/pkg/sql/sqlstats/persistedsqlstats/sqlstatsutil" + "github.com/cockroachdb/cockroach/pkg/sql/types" "github.com/cockroachdb/cockroach/pkg/util/log" "github.com/cockroachdb/cockroach/pkg/util/timeutil" "github.com/cockroachdb/errors" @@ -403,13 +404,14 @@ func (s *PersistedSQLStats) updateStatementStats( ) error { updateStmt := ` UPDATE system.statement_statistics -SET statistics = $1 -WHERE fingerprint_id = $2 - AND transaction_fingerprint_id = $3 - AND aggregated_ts = $4 - AND app_name = $5 - AND plan_hash = $6 - AND node_id = $7 +SET statistics = $1, +index_recommendations = $2 +WHERE fingerprint_id = $3 + AND transaction_fingerprint_id = $4 + AND aggregated_ts = $5 + AND app_name = $6 + AND plan_hash = $7 + AND node_id = $8 ` statisticsJSON, err := sqlstatsutil.BuildStmtStatisticsJSON(&stats.Stats) @@ -417,6 +419,12 @@ WHERE fingerprint_id = $2 return err } statistics := tree.NewDJSON(statisticsJSON) + indexRecommendations := tree.NewDArray(types.String) + for _, recommendation := range stats.Stats.IndexRecommendations { + if err := indexRecommendations.Append(tree.NewDString(recommendation)); err != nil { + return err + } + } rowsAffected, err := s.cfg.InternalExecutor.ExecEx( ctx, @@ -427,6 +435,7 @@ WHERE fingerprint_id = $2 }, updateStmt, statistics, // statistics + indexRecommendations, // index_recommendations serializedFingerprintID, // fingerprint_id serializedTransactionFingerprintID, // transaction_fingerprint_id aggregatedTs, // aggregated_ts @@ -465,7 +474,7 @@ func (s *PersistedSQLStats) insertStatementStats( ) (rowsAffected int, err error) { insertStmt := ` INSERT INTO system.statement_statistics -VALUES ($1, $2, $3, $4, $5, $6, $7, $8, $9, $10) +VALUES ($1, $2, $3, $4, $5, $6, $7, $8, $9, $10, $11) ON CONFLICT (crdb_internal_aggregated_ts_app_name_fingerprint_id_node_id_plan_hash_transaction_fingerprint_id_shard_8, aggregated_ts, fingerprint_id, transaction_fingerprint_id, app_name, plan_hash, node_id) DO NOTHING @@ -486,6 +495,12 @@ DO NOTHING statistics := tree.NewDJSON(statisticsJSON) plan := tree.NewDJSON(sqlstatsutil.ExplainTreePlanNodeToJSON(&stats.Stats.SensitiveInfo.MostRecentPlanDescription)) + indexRecommendations := tree.NewDArray(types.String) + for _, recommendation := range stats.Stats.IndexRecommendations { + if err := indexRecommendations.Append(tree.NewDString(recommendation)); err != nil { + return 0, err + } + } rowsAffected, err = s.cfg.InternalExecutor.ExecEx( ctx, @@ -505,6 +520,7 @@ DO NOTHING metadata, // metadata statistics, // statistics plan, // plan + indexRecommendations, // index_recommendations ) return rowsAffected, err diff --git a/pkg/sql/sqlstats/persistedsqlstats/sqlstatsutil/json_encoding_test.go b/pkg/sql/sqlstats/persistedsqlstats/sqlstatsutil/json_encoding_test.go index 04b23341bf5c..cb9e98321245 100644 --- a/pkg/sql/sqlstats/persistedsqlstats/sqlstatsutil/json_encoding_test.go +++ b/pkg/sql/sqlstats/persistedsqlstats/sqlstatsutil/json_encoding_test.go @@ -121,7 +121,8 @@ func TestSQLStatsJsonEncoding(t *testing.T) { "mean": {{.Float}}, "sqDiff": {{.Float}} } - } + }, + "index_recommendations": [{{joinStrings .StringArray}}] } ` @@ -239,7 +240,8 @@ func TestSQLStatsJsonEncoding(t *testing.T) { "mean": {{.Float}}, "sqDiff": {{.Float}} } - } + }, + "index_recommendations": [{{joinStrings .StringArray}}] } ` diff --git a/pkg/sql/sqlstats/persistedsqlstats/sqlstatsutil/json_impl.go b/pkg/sql/sqlstats/persistedsqlstats/sqlstatsutil/json_impl.go index 240dee25204e..19c0b41d61c1 100644 --- a/pkg/sql/sqlstats/persistedsqlstats/sqlstatsutil/json_impl.go +++ b/pkg/sql/sqlstats/persistedsqlstats/sqlstatsutil/json_impl.go @@ -75,6 +75,7 @@ func (s *stmtStats) jsonFields() jsonFields { return jsonFields{ {"statistics", (*innerStmtStats)(s)}, {"execution_statistics", (*execStats)(&s.ExecStats)}, + {"index_recommendations", (*stringArray)(&s.IndexRecommendations)}, } } diff --git a/pkg/sql/sqlstats/persistedsqlstats/stmt_reader.go b/pkg/sql/sqlstats/persistedsqlstats/stmt_reader.go index b34d60684a12..fce7e1051a6e 100644 --- a/pkg/sql/sqlstats/persistedsqlstats/stmt_reader.go +++ b/pkg/sql/sqlstats/persistedsqlstats/stmt_reader.go @@ -23,6 +23,7 @@ import ( "github.com/cockroachdb/cockroach/pkg/sql/sqlstats" "github.com/cockroachdb/cockroach/pkg/sql/sqlstats/persistedsqlstats/sqlstatsutil" "github.com/cockroachdb/cockroach/pkg/sql/sqlutil" + "github.com/cockroachdb/cockroach/pkg/util" "github.com/cockroachdb/errors" ) @@ -111,6 +112,7 @@ func (s *PersistedSQLStats) getFetchQueryForStmtStatsTable( "statistics", "plan", "agg_interval", + "index_recommendations", } // [1]: selection columns @@ -188,5 +190,12 @@ func rowToStmtStats(row tree.Datums) (*roachpb.CollectedStatementStatistics, err aggInterval := tree.MustBeDInterval(row[8]).Duration stats.AggregationInterval = time.Duration(aggInterval.Nanos()) + recommendations := tree.MustBeDArray(row[9]) + var indexRecommendations []string + for _, s := range recommendations.Array { + indexRecommendations = util.CombineUniqueString(indexRecommendations, []string{string(tree.MustBeDString(s))}) + } + stats.Stats.IndexRecommendations = indexRecommendations + return &stats, nil } diff --git a/pkg/sql/sqlstats/ssmemstorage/ss_mem_writer.go b/pkg/sql/sqlstats/ssmemstorage/ss_mem_writer.go index 631ced0b3088..322c53733058 100644 --- a/pkg/sql/sqlstats/ssmemstorage/ss_mem_writer.go +++ b/pkg/sql/sqlstats/ssmemstorage/ss_mem_writer.go @@ -126,6 +126,8 @@ func (s *Container) RecordStatement( stats.mu.data.LastExecTimestamp = s.getTimeNow() stats.mu.data.Nodes = util.CombineUniqueInt64(stats.mu.data.Nodes, value.Nodes) stats.mu.data.PlanGists = util.CombineUniqueString(stats.mu.data.PlanGists, []string{value.PlanGist}) + stats.mu.data.IndexRecommendations = util.CombineUniqueString(stats.mu.data.IndexRecommendations, value.IndexRecommendations) + // Note that some fields derived from tracing statements (such as // BytesSentOverNetwork) are not updated here because they are collected // on-demand. diff --git a/pkg/sql/sqlstats/ssprovider.go b/pkg/sql/sqlstats/ssprovider.go index 33a1617ae929..1dc98687b69a 100644 --- a/pkg/sql/sqlstats/ssprovider.go +++ b/pkg/sql/sqlstats/ssprovider.go @@ -191,23 +191,24 @@ type Provider interface { // RecordedStmtStats stores the statistics of a statement to be recorded. type RecordedStmtStats struct { - SessionID clusterunique.ID - StatementID clusterunique.ID - AutoRetryCount int - RowsAffected int - ParseLatency float64 - PlanLatency float64 - RunLatency float64 - ServiceLatency float64 - OverheadLatency float64 - BytesRead int64 - RowsRead int64 - RowsWritten int64 - Nodes []int64 - StatementType tree.StatementType - Plan *roachpb.ExplainTreePlanNode - PlanGist string - StatementError error + SessionID clusterunique.ID + StatementID clusterunique.ID + AutoRetryCount int + RowsAffected int + ParseLatency float64 + PlanLatency float64 + RunLatency float64 + ServiceLatency float64 + OverheadLatency float64 + BytesRead int64 + RowsRead int64 + RowsWritten int64 + Nodes []int64 + StatementType tree.StatementType + Plan *roachpb.ExplainTreePlanNode + PlanGist string + StatementError error + IndexRecommendations []string } // RecordedTxnStats stores the statistics of a transaction to be recorded. diff --git a/pkg/ui/workspaces/cluster-ui/src/statementsPage/statementsPage.fixture.ts b/pkg/ui/workspaces/cluster-ui/src/statementsPage/statementsPage.fixture.ts index b1e1d3084d90..47124953d7e0 100644 --- a/pkg/ui/workspaces/cluster-ui/src/statementsPage/statementsPage.fixture.ts +++ b/pkg/ui/workspaces/cluster-ui/src/statementsPage/statementsPage.fixture.ts @@ -94,6 +94,7 @@ const statementStats: Required = { squared_diffs: 0.005, }, plan_gists: ["Ais="], + index_recommendations: [""], exec_stats: execStats, last_exec_timestamp: { seconds: Long.fromInt(1599670292), diff --git a/pkg/ui/workspaces/cluster-ui/src/util/appStats/appStats.spec.ts b/pkg/ui/workspaces/cluster-ui/src/util/appStats/appStats.spec.ts index fc1306dc57a7..f35b12c121ce 100644 --- a/pkg/ui/workspaces/cluster-ui/src/util/appStats/appStats.spec.ts +++ b/pkg/ui/workspaces/cluster-ui/src/util/appStats/appStats.spec.ts @@ -275,6 +275,7 @@ function randomStats( }, nodes: [Long.fromInt(1), Long.fromInt(3), Long.fromInt(4)], plan_gists: ["Ais="], + index_recommendations: [""], }; } diff --git a/pkg/ui/workspaces/cluster-ui/src/util/appStats/appStats.ts b/pkg/ui/workspaces/cluster-ui/src/util/appStats/appStats.ts index e047e04f4474..f579fb161829 100644 --- a/pkg/ui/workspaces/cluster-ui/src/util/appStats/appStats.ts +++ b/pkg/ui/workspaces/cluster-ui/src/util/appStats/appStats.ts @@ -139,6 +139,15 @@ export function addStatementStats( planGists = b.plan_gists; } + let indexRec: string[] = []; + if (a.index_recommendations && b.index_recommendations) { + indexRec = unique(a.index_recommendations.concat(b.index_recommendations)); + } else if (a.index_recommendations) { + indexRec = a.index_recommendations; + } else if (b.index_recommendations) { + indexRec = b.index_recommendations; + } + return { count: a.count.add(b.count), first_attempt_count: a.first_attempt_count.add(b.first_attempt_count), @@ -187,6 +196,7 @@ export function addStatementStats( : b.last_exec_timestamp, nodes: uniqueLong([...a.nodes, ...b.nodes]), plan_gists: planGists, + index_recommendations: indexRec, }; } diff --git a/pkg/ui/workspaces/db-console/src/views/statements/statements.spec.tsx b/pkg/ui/workspaces/db-console/src/views/statements/statements.spec.tsx index b6f463a87769..bebb26596b16 100644 --- a/pkg/ui/workspaces/db-console/src/views/statements/statements.spec.tsx +++ b/pkg/ui/workspaces/db-console/src/views/statements/statements.spec.tsx @@ -527,6 +527,7 @@ function makeStats(): Required { }, nodes: [Long.fromInt(1), Long.fromInt(2), Long.fromInt(3)], plan_gists: ["Ais="], + index_recommendations: [], }; } diff --git a/pkg/upgrade/upgrades/BUILD.bazel b/pkg/upgrade/upgrades/BUILD.bazel index a53bf7c8c4cf..8dbbb9d487e8 100644 --- a/pkg/upgrade/upgrades/BUILD.bazel +++ b/pkg/upgrade/upgrades/BUILD.bazel @@ -5,6 +5,7 @@ go_library( name = "upgrades", srcs = [ "alter_sql_instances_locality.go", + "alter_statement_statistics_index_recommendations.go", "alter_table_protected_timestamp_records.go", "comment_on_index_migration.go", "descriptor_utils.go", @@ -67,6 +68,7 @@ go_test( size = "large", srcs = [ "alter_sql_instances_locality_test.go", + "alter_statement_statistics_index_recommendations_test.go", "alter_table_protected_timestamp_records_test.go", "builtins_test.go", "comment_on_index_migration_external_test.go", diff --git a/pkg/upgrade/upgrades/alter_statement_statistics_index_recommendations.go b/pkg/upgrade/upgrades/alter_statement_statistics_index_recommendations.go new file mode 100644 index 000000000000..6bbcb5e73902 --- /dev/null +++ b/pkg/upgrade/upgrades/alter_statement_statistics_index_recommendations.go @@ -0,0 +1,42 @@ +// 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. + +package upgrades + +import ( + "context" + + "github.com/cockroachdb/cockroach/pkg/clusterversion" + "github.com/cockroachdb/cockroach/pkg/jobs" + "github.com/cockroachdb/cockroach/pkg/keys" + "github.com/cockroachdb/cockroach/pkg/sql/catalog/systemschema" + "github.com/cockroachdb/cockroach/pkg/upgrade" +) + +const addIndexRecommendationsCol = ` +ALTER TABLE system.statement_statistics +ADD COLUMN IF NOT EXISTS "index_recommendations" STRING[] NOT NULL DEFAULT (array[]::STRING[]) +FAMILY "primary" +` + +func alterSystemStatementStatisticsAddIndexRecommendations( + ctx context.Context, cs clusterversion.ClusterVersion, d upgrade.TenantDeps, _ *jobs.Job, +) error { + op := operation{ + name: "add-statement-statistics-index-recommendations-col", + schemaList: []string{"index_rec"}, + query: addIndexRecommendationsCol, + schemaExistsFn: hasColumn, + } + if err := migrateTable(ctx, cs, d, op, keys.StatementStatisticsTableID, systemschema.StatementStatisticsTable); err != nil { + return err + } + return nil +} diff --git a/pkg/upgrade/upgrades/alter_statement_statistics_index_recommendations_test.go b/pkg/upgrade/upgrades/alter_statement_statistics_index_recommendations_test.go new file mode 100644 index 000000000000..893bf951faab --- /dev/null +++ b/pkg/upgrade/upgrades/alter_statement_statistics_index_recommendations_test.go @@ -0,0 +1,193 @@ +// 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. + +package upgrades_test + +import ( + "context" + "testing" + + "github.com/cockroachdb/cockroach/pkg/base" + "github.com/cockroachdb/cockroach/pkg/clusterversion" + "github.com/cockroachdb/cockroach/pkg/keys" + "github.com/cockroachdb/cockroach/pkg/security/username" + "github.com/cockroachdb/cockroach/pkg/server" + "github.com/cockroachdb/cockroach/pkg/sql/catalog/catpb" + "github.com/cockroachdb/cockroach/pkg/sql/catalog/descpb" + "github.com/cockroachdb/cockroach/pkg/sql/catalog/systemschema" + "github.com/cockroachdb/cockroach/pkg/sql/catalog/tabledesc" + "github.com/cockroachdb/cockroach/pkg/sql/privilege" + "github.com/cockroachdb/cockroach/pkg/sql/sem/catconstants" + "github.com/cockroachdb/cockroach/pkg/sql/types" + "github.com/cockroachdb/cockroach/pkg/testutils/testcluster" + "github.com/cockroachdb/cockroach/pkg/upgrade/upgrades" + "github.com/cockroachdb/cockroach/pkg/util/leaktest" + "github.com/cockroachdb/cockroach/pkg/util/log" +) + +func TestAlterSystemStatementStatisticsTable(t *testing.T) { + defer leaktest.AfterTest(t)() + defer log.Scope(t).Close(t) + + clusterArgs := base.TestClusterArgs{ + ServerArgs: base.TestServerArgs{ + Knobs: base.TestingKnobs{ + Server: &server.TestingKnobs{ + DisableAutomaticVersionUpgrade: make(chan struct{}), + BinaryVersionOverride: clusterversion.ByKey( + clusterversion.AlterSystemStatementStatisticsAddIndexRecommendations - 1), + }, + }, + }, + } + + var ( + ctx = context.Background() + + tc = testcluster.StartTestCluster(t, 1, clusterArgs) + s = tc.Server(0) + sqlDB = tc.ServerConn(0) + ) + defer tc.Stopper().Stop(ctx) + + var ( + validationSchemas = []upgrades.Schema{ + {Name: "index_recommendations", ValidationFn: upgrades.HasColumn}, + {Name: "primary", ValidationFn: upgrades.HasColumnFamily}, + } + ) + + // Inject the old copy of the descriptor. + upgrades.InjectLegacyTable(ctx, t, s, systemschema.StatementStatisticsTable, getDeprecatedStatementStatisticsDescriptor) + // Validate that the table statement_statistics has the old schema. + upgrades.ValidateSchemaExists( + ctx, + t, + s, + sqlDB, + keys.StatementStatisticsTableID, + systemschema.StatementStatisticsTable, + []string{}, + validationSchemas, + false, /* expectExists */ + ) + // Run the upgrade. + upgrades.Upgrade( + t, + sqlDB, + clusterversion.AlterSystemStatementStatisticsAddIndexRecommendations, + nil, /* done */ + false, /* expectError */ + ) + // Validate that the table has new schema. + upgrades.ValidateSchemaExists( + ctx, + t, + s, + sqlDB, + keys.StatementStatisticsTableID, + systemschema.StatementStatisticsTable, + []string{}, + validationSchemas, + true, /* expectExists */ + ) +} + +// getDeprecatedSqlInstancesDescriptor returns the system.sql_instances +// table descriptor that was being used before adding a new column in the +// current version. +func getDeprecatedStatementStatisticsDescriptor() *descpb.TableDescriptor { + sqlStmtHashComputeExpr := `mod(fnv32(crdb_internal.datums_to_bytes(aggregated_ts, app_name, fingerprint_id, node_id, plan_hash, transaction_fingerprint_id)), 8:::INT8)` + + return &descpb.TableDescriptor{ + Name: string(catconstants.StatementStatisticsTableName), + ID: keys.StatementStatisticsTableID, + ParentID: keys.SystemDatabaseID, + UnexposedParentSchemaID: keys.PublicSchemaID, + Version: 1, + Columns: []descpb.ColumnDescriptor{ + {Name: "aggregated_ts", ID: 1, Type: types.TimestampTZ, Nullable: false}, + {Name: "fingerprint_id", ID: 2, Type: types.Bytes, Nullable: false}, + {Name: "transaction_fingerprint_id", ID: 3, Type: types.Bytes, Nullable: false}, + {Name: "plan_hash", ID: 4, Type: types.Bytes, Nullable: false}, + {Name: "app_name", ID: 5, Type: types.String, Nullable: false}, + {Name: "node_id", ID: 6, Type: types.Int, Nullable: false}, + {Name: "agg_interval", ID: 7, Type: types.Interval, Nullable: false}, + {Name: "metadata", ID: 8, Type: types.Jsonb, Nullable: false}, + {Name: "statistics", ID: 9, Type: types.Jsonb, Nullable: false}, + {Name: "plan", ID: 10, Type: types.Jsonb, Nullable: false}, + { + Name: "crdb_internal_aggregated_ts_app_name_fingerprint_id_node_id_plan_hash_transaction_fingerprint_id_shard_8", + ID: 11, + Type: types.Int4, + Nullable: false, + ComputeExpr: &sqlStmtHashComputeExpr, + Hidden: true, + }, + }, + NextColumnID: 12, + Families: []descpb.ColumnFamilyDescriptor{ + { + Name: "primary", + ID: 0, + ColumnNames: []string{ + "crdb_internal_aggregated_ts_app_name_fingerprint_id_node_id_plan_hash_transaction_fingerprint_id_shard_8", + "aggregated_ts", "fingerprint_id", "transaction_fingerprint_id", "plan_hash", "app_name", "node_id", + "agg_interval", "metadata", "statistics", "plan", + }, + ColumnIDs: []descpb.ColumnID{11, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10}, + DefaultColumnID: 0, + }, + }, + NextFamilyID: 1, + PrimaryIndex: descpb.IndexDescriptor{ + Name: tabledesc.LegacyPrimaryKeyIndexName, + ID: 1, + Unique: true, + KeyColumnNames: []string{ + "crdb_internal_aggregated_ts_app_name_fingerprint_id_node_id_plan_hash_transaction_fingerprint_id_shard_8", + "aggregated_ts", + "fingerprint_id", + "transaction_fingerprint_id", + "plan_hash", + "app_name", + "node_id", + }, + KeyColumnDirections: []catpb.IndexColumn_Direction{ + catpb.IndexColumn_ASC, + catpb.IndexColumn_ASC, + catpb.IndexColumn_ASC, + catpb.IndexColumn_ASC, + catpb.IndexColumn_ASC, + catpb.IndexColumn_ASC, + catpb.IndexColumn_ASC, + }, + KeyColumnIDs: []descpb.ColumnID{11, 1, 2, 3, 4, 5, 6}, + Version: descpb.StrictIndexColumnIDGuaranteesVersion, + Sharded: catpb.ShardedDescriptor{ + IsSharded: true, + Name: "crdb_internal_aggregated_ts_app_name_fingerprint_id_node_id_plan_hash_transaction_fingerprint_id_shard_8", + ShardBuckets: 8, + ColumnNames: []string{ + "aggregated_ts", + "app_name", + "fingerprint_id", + "node_id", + "plan_hash", + "transaction_fingerprint_id", + }, + }, + }, + NextIndexID: 3, + Privileges: catpb.NewCustomSuperuserPrivilegeDescriptor(privilege.ReadWriteData, username.NodeUserName()), + NextMutationID: 1, + FormatVersion: 3, + } +} diff --git a/pkg/upgrade/upgrades/upgrades.go b/pkg/upgrade/upgrades/upgrades.go index 627a9f1cdcae..38db772ea6f2 100644 --- a/pkg/upgrade/upgrades/upgrades.go +++ b/pkg/upgrade/upgrades/upgrades.go @@ -140,6 +140,12 @@ var upgrades = []upgrade.Upgrade{ NoPrecondition, systemExternalConnectionsTableMigration, ), + upgrade.NewTenantUpgrade( + "add column index_recommendations to table system.statement_statistics", + toCV(clusterversion.AlterSystemStatementStatisticsAddIndexRecommendations), + NoPrecondition, + alterSystemStatementStatisticsAddIndexRecommendations, + ), } func init() {