Skip to content

Commit

Permalink
storage: make pebbleLogger.Eventf also log
Browse files Browse the repository at this point in the history
This is useful when tracing is not enabled, but verbosity of logging can
be increased. Also, we currently have gaps in context propagation in
tracing which this can help mitigate for scenarios where slowness is
reproducible.

Relates to cockroachdb/pebble#3728

Epic: none

Release note: None
  • Loading branch information
sumeerbhola committed Jul 13, 2024
1 parent 0775a62 commit 0ff86f8
Show file tree
Hide file tree
Showing 5 changed files with 136 additions and 28 deletions.
1 change: 1 addition & 0 deletions pkg/storage/BUILD.bazel
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,7 @@ go_library(
"pebble.go",
"pebble_batch.go",
"pebble_iterator.go",
"pebble_logger_and_tracer.go",
"pebble_merge.go",
"pebble_mvcc_scanner.go",
"read_as_of_iterator.go",
Expand Down
27 changes: 0 additions & 27 deletions pkg/storage/pebble.go
Original file line number Diff line number Diff line change
Expand Up @@ -887,33 +887,6 @@ func shortAttributeExtractorForValues(
return 0, nil
}

type pebbleLogger struct {
ctx context.Context
depth int
}

var _ pebble.LoggerAndTracer = pebbleLogger{}

func (l pebbleLogger) Infof(format string, args ...interface{}) {
log.Storage.InfofDepth(l.ctx, l.depth, format, args...)
}

func (l pebbleLogger) Fatalf(format string, args ...interface{}) {
log.Storage.FatalfDepth(l.ctx, l.depth, format, args...)
}

func (l pebbleLogger) Eventf(ctx context.Context, format string, args ...interface{}) {
log.Eventf(ctx, format, args...)
}

func (l pebbleLogger) IsTracingEnabled(ctx context.Context) bool {
return log.HasSpan(ctx)
}

func (l pebbleLogger) Errorf(format string, args ...interface{}) {
log.Storage.ErrorfDepth(l.ctx, l.depth, format, args...)
}

// engineConfig holds all configuration parameters and knobs used in setting up
// a new storage engine.
type engineConfig struct {
Expand Down
54 changes: 54 additions & 0 deletions pkg/storage/pebble_logger_and_tracer.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,54 @@
// Copyright 2024 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 storage

import (
"context"

"github.com/cockroachdb/cockroach/pkg/util/log"
"github.com/cockroachdb/pebble"
)

type pebbleLogger struct {
ctx context.Context
depth int
}

var _ pebble.LoggerAndTracer = pebbleLogger{}

func (l pebbleLogger) Infof(format string, args ...interface{}) {
log.Storage.InfofDepth(l.ctx, l.depth, format, args...)
}

func (l pebbleLogger) Fatalf(format string, args ...interface{}) {
log.Storage.FatalfDepth(l.ctx, l.depth, format, args...)
}

func (l pebbleLogger) Errorf(format string, args ...interface{}) {
log.Storage.ErrorfDepth(l.ctx, l.depth, format, args...)
}

// pebble.LoggerAndTracer does not expose verbosity levels in its logging
// interface, and Pebble logs go to a separate STORAGE channel.
//
// The tracing part of the interface is meant for user-facing activities, so
// in addition to outputting the event when tracing is enabled, we also log.
// The eventAlsoLogVerbosityLevel of 2 is chosen semi-arbitrarily since this
// is the only verbosity level in this file.
const eventAlsoLogVerbosityLevel = 2

func (l pebbleLogger) Eventf(ctx context.Context, format string, args ...interface{}) {
log.VEventf(ctx, eventAlsoLogVerbosityLevel, format, args...)
}

func (l pebbleLogger) IsTracingEnabled(ctx context.Context) bool {
return log.HasSpan(ctx) || log.ExpensiveLogEnabled(ctx, eventAlsoLogVerbosityLevel)
}
80 changes: 80 additions & 0 deletions pkg/storage/pebble_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -13,8 +13,10 @@ package storage
import (
"context"
"fmt"
"math"
"math/rand"
"path/filepath"
"regexp"
"runtime"
"strings"
"sync/atomic"
Expand Down Expand Up @@ -1508,3 +1510,81 @@ func TestMinimumSupportedFormatVersion(t *testing.T) {
require.Equal(t, pebbleFormatVersionMap[clusterversion.MinSupported], MinimumSupportedFormatVersion,
"MinimumSupportedFormatVersion must match the format version for %s", clusterversion.MinSupported)
}

// delayFS injects a delay on each read.
type delayFS struct {
vfs.FS
}

func (fs delayFS) Open(name string, opts ...vfs.OpenOption) (vfs.File, error) {
f, err := fs.FS.Open(name, opts...)
if err != nil {
return nil, err
}
return delayFile{File: f}, nil
}

type delayFile struct {
vfs.File
}

func (f delayFile) ReadAt(p []byte, off int64) (n int, err error) {
time.Sleep(10 * time.Millisecond)
return f.File.ReadAt(p, off)
}

func TestPebbleLoggingSlowReads(t *testing.T) {
defer leaktest.AfterTest(t)()

s := log.ScopeWithoutShowLogs(t)
prevVModule := log.GetVModule()
_ = log.SetVModule("pebble_logger_and_tracer=2")
defer func() { _ = log.SetVModule(prevVModule) }()
defer s.Close(t)

ctx := context.Background()
testStartTs := timeutil.Now()

memFS := vfs.NewMem()
dFS := delayFS{FS: memFS}
e, err := fs.InitEnv(context.Background(), dFS, "" /* dir */, fs.EnvConfig{})
require.NoError(t, err)
// No block cache, so all reads go to FS.
db, err := Open(ctx, e, cluster.MakeClusterSettings(), func(cfg *engineConfig) error {
cfg.cacheSize = nil
return nil
})
require.NoError(t, err)
defer db.Close()
// Write some data and flush to disk.
ts1 := hlc.Timestamp{WallTime: 1}
k1 := MVCCKey{Key: []byte("a"), Timestamp: ts1}
v1 := MVCCValue{Value: roachpb.MakeValueFromString("a1")}
require.NoError(t, db.PutMVCC(k1, v1))
require.NoError(t, db.Flush())
// Read the data.
require.NoError(t, db.MVCCIterate(ctx, roachpb.Key("a"), roachpb.Key("b"), MVCCKeyIterKind, IterKeyTypePointsOnly,
UnknownReadCategory, func(MVCCKeyValue, MVCCRangeKeyStack) error {
return nil
}))

// Grab the logs and count the slow read entries.
log.FlushFiles()
entries, err := log.FetchEntriesFromFiles(testStartTs.UnixNano(),
math.MaxInt64, 2000,
regexp.MustCompile(`pebble_logger_and_tracer\.go`),
log.WithMarkedSensitiveData)
require.NoError(t, err)

// There should be some entries like the following:
// I240708 14:47:54.610060 12 storage/pebble_logger_and_tracer.go:49 [-] 15 reading 32 bytes took 11.246041ms
slowReadRegexp, err := regexp.Compile("reading .* bytes took .*")
require.NoError(t, err)
slowCount := 0
for i := range entries {
if slowReadRegexp.MatchString(entries[i].Message) {
slowCount++
}
}
require.Less(t, 0, slowCount)
}
2 changes: 1 addition & 1 deletion pkg/storage/pebbleiter/crdb_test_on.go
Original file line number Diff line number Diff line change
Expand Up @@ -143,7 +143,7 @@ func (i *assertionIter) RangeBounds() ([]byte, []byte) {
panic(errors.AssertionFailedf("RangeBounds() called on !Valid() pebble.Iterator"))
}
if _, hasRange := i.Iterator.HasPointAndRange(); !hasRange {
panic(errors.AssertionFailedf("RangeBounds() called on pebble.Iterator without range keys"))
return nil, nil
}
// See maybeSaveAndMangleRangeKeyBufs for where these are saved.
j := i.rangeKeyBufs.idx
Expand Down

0 comments on commit 0ff86f8

Please sign in to comment.