Using Context for Value Propagation
Learn how to use Go contexts to propagate values within your application
The Go context
package is a powerful tool used to control cancellations and timeouts, but it can also be utilized to propagate request-scoped values across API boundaries and between processes. This snippet demonstrates how to propagate values using contexts.
Basic Context Value Propagation
Here's how you can pass values using Go's context:
package main
import (
"context"
"fmt"
)
func main() {
ctx := context.Background()
ctx = context.WithValue(ctx, "sessionID", 42)
processRequest(ctx)
}
func processRequest(ctx context.Context) {
sessionID, ok := ctx.Value("sessionID").(int)
if !ok {
fmt.Println("sessionID not found in context")
return
}
fmt.Printf("Processing request for session ID: %d\n", sessionID)
}
Custom Key Types for Context
Instead of using primitive types or strings for context keys, it's better practice to define a custom key type to avoid potential collisions:
package main
import (
"context"
"fmt"
)
type contextKey string
const sessionKey contextKey = "sessionID"
func main() {
ctx := context.Background()
ctx = context.WithValue(ctx, sessionKey, 42)
processRequest(ctx)
}
func processRequest(ctx context.Context) {
sessionID, ok := ctx.Value(sessionKey).(int)
if !ok {
fmt.Println("sessionID not found in context")
return
}
fmt.Printf("Processing request for session ID: %d\n", sessionID)
}
Nested Context Values
You can nest context values to pass multiple pieces of information:
package main
import (
"context"
"fmt"
)
type contextKey string
const (
userIDKey contextKey = "userID"
userRoleKey contextKey = "userRole"
)
func main() {
ctx := context.Background()
ctx = context.WithValue(ctx, userIDKey, 42)
ctx = context.WithValue(ctx, userRoleKey, "admin")
processRequest(ctx)
}
func processRequest(ctx context.Context) {
userID, ok := ctx.Value(userIDKey).(int)
if !ok {
fmt.Println("userID not found in context")
return
}
userRole, _ := ctx.Value(userRoleKey).(string)
fmt.Printf("Processing request for user ID: %d with role: %s\n", userID, userRole)
}
Best Practices
- Custom Key Types: Always define custom key types for storing data in context to avoid collisions.
- Immutable Design: Treat contexts as immutable; always create a new context using
context.WithValue
rather than modifying an existing one. - Conciseness: Store only necessary values within the context to keep it lightweight.
- Avoid Abuse: Do not use context for passing optional parameters; reserve it for highly relevant information that travels across API/infrastructure boundaries.
Common Pitfalls
- Key Collisions: Using string or primitive types as keys which can lead to unintended key collisions.
- Unintended Mutability: Assuming you can mutate context values; they're designed to be immutable.
- Overusing Context: Overloading context with too many values, which can make the application difficult to maintain.
Performance Tips
- Use Custom Types for Keys: Improves type safety and performance by avoiding key collisions and reducing type assertion costs.
- Minimal Data in Context: Keep values passed through context minimal to reduce memory overhead.
- Avoid Large Values: Large context values can significantly increase memory usage and slow down context propagation.