Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

sql: add statement fingerprint ID to sampled query #84182

Merged
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions docs/generated/eventlog.md
Original file line number Diff line number Diff line change
Expand Up @@ -2464,6 +2464,7 @@ contains common SQL event/execution details.
| `StatementID` | Statement ID of the query. | no |
| `TransactionID` | Transaction ID of the query. | no |
| `DatabaseID` | Database ID of the query. | no |
| `StatementFingerprintID` | Statement fingerprint ID of the query. | no |


#### Common fields
Expand Down
10 changes: 0 additions & 10 deletions pkg/roachpb/app_stats.go
Original file line number Diff line number Diff line change
Expand Up @@ -19,16 +19,6 @@ import (
// StmtFingerprintID is the type of a Statement's fingerprint ID.
type StmtFingerprintID uint64

// FingerprintID returns the FingerprintID of the StatementStatisticsKey.
func (m *StatementStatisticsKey) FingerprintID() StmtFingerprintID {
return ConstructStatementFingerprintID(
m.Query,
m.Failed,
m.ImplicitTxn,
m.Database,
)
}

// ConstructStatementFingerprintID constructs an ID by hashing an anonymized query, its database
// and failure status, and if it was part of an implicit txn. At the time of writing,
// these are the axis' we use to bucket queries for stats collection
Expand Down
1 change: 0 additions & 1 deletion pkg/sql/conn_executor.go
Original file line number Diff line number Diff line change
Expand Up @@ -1365,7 +1365,6 @@ type connExecutor struct {
savepoints savepointStack
sessionDataStack *sessiondata.Stack
}

// transactionStatementFingerprintIDs tracks all statement IDs that make up the current
// transaction. It's length is bound by the TxnStatsNumStmtFingerprintIDsToRecord
// cluster setting.
Expand Down
4 changes: 3 additions & 1 deletion pkg/sql/conn_executor_exec.go
Original file line number Diff line number Diff line change
Expand Up @@ -1061,6 +1061,7 @@ func (ex *connExecutor) dispatchToExecutionEngine(
res.DisableBuffering()
}

var stmtFingerprintID roachpb.StmtFingerprintID
defer func() {
planner.maybeLogStatement(
ctx,
Expand All @@ -1072,6 +1073,7 @@ func (ex *connExecutor) dispatchToExecutionEngine(
ex.statsCollector.PhaseTimes().GetSessionPhaseTime(sessionphase.SessionQueryReceived),
&ex.extraTxnState.hasAdminRoleCache,
ex.server.TelemetryLoggingMetrics,
stmtFingerprintID,
)
}()

Expand Down Expand Up @@ -1185,7 +1187,7 @@ func (ex *connExecutor) dispatchToExecutionEngine(

// Record the statement summary. This also closes the plan if the
// plan has not been closed earlier.
ex.recordStatementSummary(
stmtFingerprintID = ex.recordStatementSummary(
ctx, planner,
int(ex.state.mu.autoRetryCounter), res.RowsAffected(), res.Err(), stats,
)
Expand Down
24 changes: 14 additions & 10 deletions pkg/sql/exec_log.go
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@ import (
"fmt"
"time"

"github.com/cockroachdb/cockroach/pkg/roachpb"
"github.com/cockroachdb/cockroach/pkg/settings"
"github.com/cockroachdb/cockroach/pkg/sql/catalog"
"github.com/cockroachdb/cockroach/pkg/sql/catalog/descpb"
Expand Down Expand Up @@ -155,8 +156,9 @@ func (p *planner) maybeLogStatement(
queryReceived time.Time,
hasAdminRoleCache *HasAdminRoleCache,
telemetryLoggingMetrics *TelemetryLoggingMetrics,
stmtFingerprintID roachpb.StmtFingerprintID,
) {
p.maybeLogStatementInternal(ctx, execType, numRetries, txnCounter, rows, err, queryReceived, hasAdminRoleCache, telemetryLoggingMetrics)
p.maybeLogStatementInternal(ctx, execType, numRetries, txnCounter, rows, err, queryReceived, hasAdminRoleCache, telemetryLoggingMetrics, stmtFingerprintID)
}

func (p *planner) maybeLogStatementInternal(
Expand All @@ -167,6 +169,7 @@ func (p *planner) maybeLogStatementInternal(
startTime time.Time,
hasAdminRoleCache *HasAdminRoleCache,
telemetryMetrics *TelemetryLoggingMetrics,
stmtFingerprintID roachpb.StmtFingerprintID,
) {
// Note: if you find the code below crashing because p.execCfg == nil,
// do not add a test "if p.execCfg == nil { do nothing }" !
Expand Down Expand Up @@ -386,15 +389,16 @@ func (p *planner) maybeLogStatementInternal(
skippedQueries := telemetryMetrics.resetSkippedQueryCount()
databaseName := p.CurrentDatabase()
sampledQuery := eventpb.SampledQuery{
CommonSQLExecDetails: execDetails,
SkippedQueries: skippedQueries,
CostEstimate: p.curPlan.instrumentation.costEstimate,
Distribution: p.curPlan.instrumentation.distribution.String(),
PlanGist: p.curPlan.instrumentation.planGist.String(),
SessionID: p.extendedEvalCtx.SessionID.String(),
Database: p.CurrentDatabase(),
StatementID: p.stmt.QueryID.String(),
TransactionID: p.txn.ID().String(),
CommonSQLExecDetails: execDetails,
SkippedQueries: skippedQueries,
CostEstimate: p.curPlan.instrumentation.costEstimate,
Distribution: p.curPlan.instrumentation.distribution.String(),
PlanGist: p.curPlan.instrumentation.planGist.String(),
SessionID: p.extendedEvalCtx.SessionID.String(),
Database: databaseName,
StatementID: p.stmt.QueryID.String(),
TransactionID: p.txn.ID().String(),
StatementFingerprintID: uint64(stmtFingerprintID),
}
db, _ := p.Descriptors().GetImmutableDatabaseByName(ctx, p.txn, databaseName, tree.DatabaseLookupFlags{Required: true})
if db != nil {
Expand Down
3 changes: 2 additions & 1 deletion pkg/sql/executor_statement_metrics.go
Original file line number Diff line number Diff line change
Expand Up @@ -116,7 +116,7 @@ func (ex *connExecutor) recordStatementSummary(
rowsAffected int,
stmtErr error,
stats topLevelQueryStats,
) {
) roachpb.StmtFingerprintID {
phaseTimes := ex.statsCollector.PhaseTimes()

// Collect the statistics.
Expand Down Expand Up @@ -235,6 +235,7 @@ func (ex *connExecutor) recordStatementSummary(
sessionAge,
)
}
return stmtFingerprintID
}

func (ex *connExecutor) updateOptCounters(planFlags planFlags) {
Expand Down
10 changes: 10 additions & 0 deletions pkg/sql/telemetry_logging_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,7 @@ import (
"time"

"github.com/cockroachdb/cockroach/pkg/base"
"github.com/cockroachdb/cockroach/pkg/roachpb"
"github.com/cockroachdb/cockroach/pkg/testutils/serverutils"
"github.com/cockroachdb/cockroach/pkg/testutils/sqlutils"
"github.com/cockroachdb/cockroach/pkg/util/leaktest"
Expand Down Expand Up @@ -119,6 +120,7 @@ func TestTelemetryLogging(t *testing.T) {
testData := []struct {
name string
query string
queryNoConstants string
execTimestampsSeconds []float64 // Execute the query with the following timestamps.
expectedLogStatement string
stubMaxEventFrequency int64
Expand All @@ -133,6 +135,7 @@ func TestTelemetryLogging(t *testing.T) {
// logged since we log all statements that are not of type DML.
"truncate-table-query",
"TRUNCATE t;",
"TRUNCATE TABLE t",
[]float64{1, 1.1, 1.2, 2},
`TRUNCATE TABLE`,
1,
Expand All @@ -145,6 +148,7 @@ func TestTelemetryLogging(t *testing.T) {
// The first statement should be logged.
"select-*-limit-1-query",
"SELECT * FROM t LIMIT 1;",
"SELECT * FROM t LIMIT _",
[]float64{3},
`SELECT * FROM \"\".\"\".t LIMIT ‹1›`,
1,
Expand All @@ -158,6 +162,7 @@ func TestTelemetryLogging(t *testing.T) {
// thus 2 log statements are expected, with 2 skipped queries.
"select-*-limit-2-query",
"SELECT * FROM t LIMIT 2;",
"SELECT * FROM t LIMIT _",
[]float64{4, 4.1, 4.2, 5},
`SELECT * FROM \"\".\"\".t LIMIT ‹2›`,
1,
Expand All @@ -170,6 +175,7 @@ func TestTelemetryLogging(t *testing.T) {
// Once required time has elapsed, the next statement should be logged.
"select-*-limit-3-query",
"SELECT * FROM t LIMIT 3;",
"SELECT * FROM t LIMIT _",
[]float64{6, 6.01, 6.05, 6.06, 6.1, 6.2},
`SELECT * FROM \"\".\"\".t LIMIT ‹3›`,
10,
Expand Down Expand Up @@ -277,6 +283,10 @@ func TestTelemetryLogging(t *testing.T) {
if !strings.Contains(e.Message, "\"DatabaseID\":"+strconv.Itoa(int(dbID))) {
t.Errorf("expected to find DatabaseID: %v", dbID)
}
stmtFingerprintID := roachpb.ConstructStatementFingerprintID(tc.queryNoConstants, false, true, databaseName)
if !strings.Contains(e.Message, "\"StatementFingerprintID\":"+strconv.FormatUint(uint64(stmtFingerprintID), 10)) {
t.Errorf("expected to find StatementFingerprintID: %v", stmtFingerprintID)
}
}
}
if logCount != expectedLogCount {
Expand Down
9 changes: 9 additions & 0 deletions pkg/util/log/eventpb/json_encode_generated.go

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

3 changes: 3 additions & 0 deletions pkg/util/log/eventpb/telemetry.proto
Original file line number Diff line number Diff line change
Expand Up @@ -61,6 +61,9 @@ message SampledQuery {

// Database ID of the query.
uint32 database_id = 12 [(gogoproto.customname) = "DatabaseID", (gogoproto.jsontag) = ",omitempty"];

// Statement fingerprint ID of the query.
uint64 statement_fingerprint_id = 13 [(gogoproto.customname) = "StatementFingerprintID", (gogoproto.jsontag) = ',omitempty'];
}

// CapturedIndexUsageStats
Expand Down