Benchmarking¶
Blink includes benchmarks to measure the performance of critical functions. This page describes how to run the benchmarks and how to interpret the results.
Running Benchmarks¶
You can run the benchmarks using the Makefile:
Or you can use the Go test command directly:
# Run benchmarks
go test -v -bench=. -benchmem ./...
# Run benchmarks for a specific package
go test -v -bench=. -benchmem ./pkg/blink
# Run a specific benchmark
go test -v -bench=BenchmarkShouldIgnoreFile -benchmem ./pkg/blink
Benchmark Results¶
The benchmark results include:
- Operations per Second: How many operations the function can perform per second
- Time per Operation: How long each operation takes
- Memory Allocation: How much memory is allocated per operation
- Allocations per Operation: How many memory allocations are made per operation
Example benchmark results:
BenchmarkShouldIgnoreFile-10 39295886 30.57 ns/op 0 B/op 0 allocs/op
BenchmarkSubfolders-10 144 7940372 ns/op 708393 B/op 4614 allocs/op
BenchmarkRemoveOldEvents-10 10000 121779 ns/op 298686 B/op 20 allocs/op
In this example:
BenchmarkShouldIgnoreFile-10
: The benchmark for theShouldIgnoreFile
function, running on 10 CPUs39295886
: The number of iterations the benchmark ran30.57 ns/op
: Each operation took 30.57 nanoseconds0 B/op
: No memory was allocated per operation0 allocs/op
: No memory allocations were made per operation
Benchmark Functions¶
Blink includes the following benchmark functions:
BenchmarkShouldIgnoreFile¶
Benchmarks the ShouldIgnoreFile
function, which determines if a file should be ignored based on its name.
func BenchmarkShouldIgnoreFile(b *testing.B) {
filenames := []string{
"file.txt",
".hidden",
"_temp",
"dir",
".git",
"_build",
}
b.ResetTimer()
for i := 0; i < b.N; i++ {
for _, filename := range filenames {
ShouldIgnoreFile(filename)
}
}
}
BenchmarkSubfolders¶
Benchmarks the Subfolders
function, which returns a list of all subdirectories in a given path.
func BenchmarkSubfolders(b *testing.B) {
// Create a temporary directory structure for benchmarking
tempDir, err := ioutil.TempDir("", "blink-bench-")
if err != nil {
b.Fatalf("Failed to create temp dir: %v", err)
}
defer os.RemoveAll(tempDir)
// Create a directory structure with 5 levels and 3 directories per level
createDirTree(b, tempDir, 5, 3)
b.ResetTimer()
for i := 0; i < b.N; i++ {
Subfolders(tempDir)
}
}
BenchmarkRemoveOldEvents¶
Benchmarks the RemoveOldEvents
function, which removes old events from a TimeEventMap
.
func BenchmarkRemoveOldEvents(b *testing.B) {
// Create a TimeEventMap with a large number of events
events := make(TimeEventMap)
now := time.Now()
// Add 1000 events at different times
for i := 0; i < 1000; i++ {
events[now.Add(-time.Duration(i)*time.Minute)] = Event{}
}
// Benchmark removing events older than 500 minutes
maxAge := 500 * time.Minute
b.ResetTimer()
for i := 0; i < b.N; i++ {
// Create a copy of the events map for each iteration
eventsCopy := make(TimeEventMap)
for k, v := range events {
eventsCopy[k] = v
}
RemoveOldEvents(&eventsCopy, maxAge)
}
}
Writing Benchmarks¶
When writing benchmarks for Blink, follow these guidelines:
- Reset Timer: Use
b.ResetTimer()
before the benchmark loop to exclude setup time - Benchmark Memory: Use the
-benchmem
flag to measure memory usage - Benchmark Realistic Scenarios: Benchmark realistic scenarios with realistic data
- Benchmark Critical Functions: Focus on performance-critical functions
- Benchmark Edge Cases: Benchmark edge cases and worst-case scenarios
Example Benchmark¶
Here's an example of a good benchmark:
func BenchmarkEventFilter(b *testing.B) {
// Create a filter
filter := NewEventFilter()
filter.SetIncludePatterns("*.js,*.css,*.html")
filter.SetExcludePatterns("node_modules,*.tmp")
filter.SetIncludeEvents("write,create")
filter.SetIgnoreEvents("chmod")
// Create events
events := []fsnotify.Event{
{Name: "file.js", Op: fsnotify.Write},
{Name: "file.css", Op: fsnotify.Create},
{Name: "file.html", Op: fsnotify.Chmod},
{Name: "file.tmp", Op: fsnotify.Write},
{Name: "node_modules/file.js", Op: fsnotify.Write},
}
b.ResetTimer()
for i := 0; i < b.N; i++ {
for _, event := range events {
filter.ShouldInclude(event)
}
}
}
Performance Profiling¶
In addition to benchmarks, you can use Go's profiling tools to analyze performance:
# CPU profiling
go test -cpuprofile=cpu.prof -bench=. ./pkg/blink
# Memory profiling
go test -memprofile=mem.prof -bench=. ./pkg/blink
# Block profiling
go test -blockprofile=block.prof -bench=. ./pkg/blink
You can then analyze the profiles with:
Performance Goals¶
Blink aims to be high-performance, with the following goals:
- Low Latency: Events should be delivered with low latency
- High Throughput: Blink should handle a high rate of events
- Low Memory Usage: Blink should use memory efficiently
- Low CPU Usage: Blink should use CPU efficiently
The benchmarks help ensure that these goals are met and that performance regressions are caught early.