GORM’s context support, enabled by the WithContext method, is a powerful feature that enhances the flexibility and control of database operations in Go applications. It allows for context management across different operational modes, timeout settings, and even integration into hooks/callbacks and middlewares. Let’s delve into these various aspects:

Single Session Mode

Single session mode is appropriate for executing individual operations. It ensures that the specific operation is executed within the context’s scope, allowing for better control and monitoring.

  1. db.WithContext(ctx).Find(&users)

Continuous Session Mode

Continuous session mode is ideal for performing a series of related operations. It maintains the context across these operations, which is particularly useful in scenarios like transactions.

  1. tx := db.WithContext(ctx)
  2. tx.First(&user, 1)
  3. tx.Model(&user).Update("role", "admin")

Context Timeout

Setting a timeout on the context passed to db.WithContext can control the duration of long-running queries. This is crucial for maintaining performance and avoiding resource lock-ups in database interactions.

  1. ctx, cancel := context.WithTimeout(context.Background(), 2*time.Second)
  2. defer cancel()
  3. db.WithContext(ctx).Find(&users)

Context in Hooks/Callbacks

The context can also be accessed within GORM’s hooks/callbacks. This enables contextual information to be used during these lifecycle events.

  1. func (u *User) BeforeCreate(tx *gorm.DB) (err error) {
  2. ctx := tx.Statement.Context
  3. // ... use context
  4. return
  5. }

Integration with Chi Middleware

GORM’s context support extends to web server middlewares, such as those in the Chi router. This allows setting a context with a timeout for all database operations within the scope of a web request.

  1. func SetDBMiddleware(next http.Handler) http.Handler {
  2. return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
  3. timeoutContext, _ := context.WithTimeout(context.Background(), time.Second)
  4. ctx := context.WithValue(r.Context(), "DB", db.WithContext(timeoutContext))
  5. next.ServeHTTP(w, r.WithContext(ctx))
  6. })
  7. }
  8. // Router setup
  9. r := chi.NewRouter()
  10. r.Use(SetDBMiddleware)
  11. // Route handlers
  12. r.Get("/", func(w http.ResponseWriter, r *http.Request) {
  13. db, ok := r.Context().Value("DB").(*gorm.DB)
  14. // ... db operations
  15. })
  16. r.Get("/user", func(w http.ResponseWriter, r *http.Request) {
  17. db, ok := r.Context().Value("DB").(*gorm.DB)
  18. // ... db operations
  19. })

Note: Setting the Context with WithContext is goroutine-safe. This ensures that database operations are safely managed across multiple goroutines. For more details, refer to the Session documentation in GORM.

Logger Integration

GORM’s logger also accepts Context, which can be used for log tracking and integrating with existing logging infrastructures.

Refer to Logger documentation for more details.