diff --git a/pkg/sql/instrumentation.go b/pkg/sql/instrumentation.go index 05b57b3edf76..1805f1e9cff3 100644 --- a/pkg/sql/instrumentation.go +++ b/pkg/sql/instrumentation.go @@ -179,7 +179,7 @@ func (ih *instrumentationHelper) Finish( } if ih.traceMetadata != nil && ih.explainPlan != nil { - ih.traceMetadata.addSpans(trace) + ih.traceMetadata.addSpans(trace, cfg.TestingKnobs.DeterministicExplainAnalyze) ih.traceMetadata.annotateExplain(ih.explainPlan) } @@ -386,7 +386,7 @@ func (m execNodeTraceMetadata) associateNodeWithProcessors( // addSpans populates the processorTraceMetadata.fields with the statistics // recorded in a trace. -func (m execNodeTraceMetadata) addSpans(spans []tracingpb.RecordedSpan) { +func (m execNodeTraceMetadata) addSpans(spans []tracingpb.RecordedSpan, makeDeterministic bool) { // Build a map from pair (encoded as a string) // to the corresponding processorTraceMetadata entry. processorKeyToMetadata := make(map[string]*processorTraceMetadata) @@ -422,6 +422,9 @@ func (m execNodeTraceMetadata) addSpans(spans []tracingpb.RecordedSpan) { if err := types.UnmarshalAny(span.Stats, &stats); err != nil { continue } + if makeDeterministic { + stats.MakeDeterministic() + } procMetadata.stats = &stats } } @@ -433,7 +436,8 @@ func (m execNodeTraceMetadata) annotateExplain(plan *explain.Plan) { walk = func(n *explain.Node) { wrapped := n.WrappedNode() if meta, ok := m[wrapped]; ok { - var rowCount uint64 + var nodeStats exec.ExecutionStats + incomplete := false for i := range meta.processors { stats := meta.processors[i].stats @@ -441,15 +445,15 @@ func (m execNodeTraceMetadata) annotateExplain(plan *explain.Plan) { incomplete = true break } - rowCount += stats.Output.NumTuples.Value() + nodeStats.RowCount.MaybeAdd(stats.Output.NumTuples) + nodeStats.KVBytesRead.MaybeAdd(stats.KV.BytesRead) + nodeStats.KVRowsRead.MaybeAdd(stats.KV.TuplesRead) } // If we didn't get statistics for all processors, we don't show the // incomplete results. In the future, we may consider an incomplete flag // if we want to show them with a warning. if !incomplete { - n.Annotate(exec.ExecutionStatsID, &exec.ExecutionStats{ - RowCount: rowCount, - }) + n.Annotate(exec.ExecutionStatsID, &nodeStats) } } diff --git a/pkg/sql/logictest/testdata/logic_test/explain_analyze b/pkg/sql/logictest/testdata/logic_test/explain_analyze index 66457753be57..56780cecb778 100644 --- a/pkg/sql/logictest/testdata/logic_test/explain_analyze +++ b/pkg/sql/logictest/testdata/logic_test/explain_analyze @@ -16,6 +16,8 @@ vectorized: · • scan actual row count: 0 + KV rows read: 0 + KV bytes read: 0 B missing stats table: kv@primary spans: [/2 - ] @@ -35,6 +37,8 @@ vectorized: · • scan actual row count: 3 + KV rows read: 3 + KV bytes read: 24 B missing stats table: kv@primary spans: [/2 - ] @@ -63,6 +67,8 @@ vectorized: ├── • scan │ columns: (k, v) │ actual row count: 4 +│ KV rows read: 4 +│ KV bytes read: 32 B │ estimated row count: 1000 (missing stats) │ table: kv@primary │ spans: FULL SCAN @@ -70,6 +76,8 @@ vectorized: └── • scan columns: (a, b) actual row count: 3 + KV rows read: 3 + KV bytes read: 24 B estimated row count: 1000 (missing stats) table: ab@primary spans: FULL SCAN diff --git a/pkg/sql/opt/exec/BUILD.bazel b/pkg/sql/opt/exec/BUILD.bazel index f7a8aa6c2042..bf12621d71c8 100644 --- a/pkg/sql/opt/exec/BUILD.bazel +++ b/pkg/sql/opt/exec/BUILD.bazel @@ -19,6 +19,7 @@ go_library( "//pkg/sql/sem/tree", "//pkg/sql/types", "//pkg/util", + "//pkg/util/optional", ], ) diff --git a/pkg/sql/opt/exec/explain/BUILD.bazel b/pkg/sql/opt/exec/explain/BUILD.bazel index 933d94168de1..72ce4abf0e91 100644 --- a/pkg/sql/opt/exec/explain/BUILD.bazel +++ b/pkg/sql/opt/exec/explain/BUILD.bazel @@ -28,6 +28,7 @@ go_library( "//pkg/util/errorutil", "//pkg/util/treeprinter", "//vendor/github.com/cockroachdb/errors", + "//vendor/github.com/dustin/go-humanize", ], ) diff --git a/pkg/sql/opt/exec/explain/emit.go b/pkg/sql/opt/exec/explain/emit.go index 616670b0f5f9..cedfef5d2f31 100644 --- a/pkg/sql/opt/exec/explain/emit.go +++ b/pkg/sql/opt/exec/explain/emit.go @@ -24,6 +24,7 @@ import ( "github.com/cockroachdb/cockroach/pkg/sql/sem/tree" "github.com/cockroachdb/cockroach/pkg/util" "github.com/cockroachdb/errors" + "github.com/dustin/go-humanize" ) // Emit produces the EXPLAIN output against the given OutputBuilder. The @@ -328,7 +329,15 @@ func (e *emitter) joinNodeName(algo string, joinType descpb.JoinType) string { func (e *emitter) emitNodeAttributes(n *Node) error { if stats, ok := n.annotations[exec.ExecutionStatsID]; ok { s := stats.(*exec.ExecutionStats) - e.ob.Attr("actual row count", s.RowCount) + if s.RowCount.HasValue() { + e.ob.AddField("actual row count", s.RowCount.String()) + } + if s.KVRowsRead.HasValue() { + e.ob.AddField("KV rows read", s.KVRowsRead.String()) + } + if s.KVBytesRead.HasValue() { + e.ob.AddField("KV bytes read", humanize.IBytes(s.KVBytesRead.Value())) + } } if stats, ok := n.annotations[exec.EstimatedStatsID]; ok { diff --git a/pkg/sql/opt/exec/factory.go b/pkg/sql/opt/exec/factory.go index cf63aed438e6..7d2ad3afda88 100644 --- a/pkg/sql/opt/exec/factory.go +++ b/pkg/sql/opt/exec/factory.go @@ -20,6 +20,7 @@ import ( "github.com/cockroachdb/cockroach/pkg/sql/sem/tree" "github.com/cockroachdb/cockroach/pkg/sql/types" "github.com/cockroachdb/cockroach/pkg/util" + "github.com/cockroachdb/cockroach/pkg/util/optional" ) // Node represents a node in the execution tree @@ -296,7 +297,10 @@ type EstimatedStats struct { // TODO(radu): can/should we just use execinfrapb.ComponentStats instead? type ExecutionStats struct { // RowCount is the number of rows produced by the operator. - RowCount uint64 + RowCount optional.Uint + + KVBytesRead optional.Uint + KVRowsRead optional.Uint } // BuildPlanForExplainFn builds an execution plan against the given diff --git a/pkg/util/optional/uint.go b/pkg/util/optional/uint.go index 289b0a4a7b86..152b420fb86c 100644 --- a/pkg/util/optional/uint.go +++ b/pkg/util/optional/uint.go @@ -63,3 +63,11 @@ func (i *Uint) Set(value uint64) { func (i *Uint) Add(delta int64) { *i = MakeUint(uint64(int64(i.Value()) + delta)) } + +// MaybeAdd adds the given value, if it is set. Does nothing if other is not +// set. +func (i *Uint) MaybeAdd(other Uint) { + if other.HasValue() { + *i = MakeUint(i.Value() + other.Value()) + } +} diff --git a/pkg/util/optional/uint_test.go b/pkg/util/optional/uint_test.go index 8ce6fbcb591c..3461ecd01ea9 100644 --- a/pkg/util/optional/uint_test.go +++ b/pkg/util/optional/uint_test.go @@ -43,7 +43,15 @@ func TestUint(t *testing.T) { require.Equal(t, uint64(0), v.Value()) require.Equal(t, v.String(), "") - v.Add(100) + var other optional.Uint + + v.MaybeAdd(other) + require.False(t, v.HasValue()) + require.Equal(t, uint64(0), v.Value()) + require.Equal(t, v.String(), "") + + other.Set(100) + v.MaybeAdd(other) require.True(t, v.HasValue()) require.Equal(t, uint64(100), v.Value()) require.Equal(t, v.String(), "100")