GinSeniorExperience

Какие production-риски есть у Gin (Go framework): blocking code, connection pooling, config, auth, observability, deploy или graceful shutdown?

Основные production-риски Gin: blocking IO в хендлерах снижает throughput, отсутствие graceful shutdown обрывает in-flight запросы, панике без recover middleware кладёт весь процесс, и слабая observability без явной интеграции OpenTelemetry.

Blocking code

Gin работает на стандартном net/http — каждый запрос занимает одну горутину. Блокирующие операции (медленный SQL, внешние HTTP без таймаута, time.Sleep) держат горутину и снижают concurrency. Всегда передавайте контекст с таймаутом:

func getUser(c *gin.Context) {
	ctx, cancel := context.WithTimeout(c.Request.Context(), 3*time.Second)
	defer cancel()

	user, err := db.GetUser(ctx, c.Param("id"))
	if err != nil {
		c.JSON(500, gin.H{"error": err.Error()})
		return
	}
	c.JSON(200, user)
}

Connection pooling

Gin не управляет пулом соединений. Для database/sql обязательно настройте лимиты:

db, err := sql.Open("postgres", dsn)
if err != nil {
	log.Fatal(err)
}
db.SetMaxOpenConns(25)
db.SetMaxIdleConns(10)
db.SetConnMaxLifetime(5 * time.Minute)
db.SetConnMaxIdleTime(1 * time.Minute)

Graceful shutdown

package main

import (
	"context"
	"net/http"
	"os"
	"os/signal"
	"syscall"
	"time"

	"github.com/gin-gonic/gin"
)

func main() {
	r := gin.Default()
	r.GET("/ping", func(c *gin.Context) {
		c.JSON(200, gin.H{"message": "pong"})
	})

	srv := &http.Server{
		Addr:         ":8080",
		Handler:      r,
		ReadTimeout:  5 * time.Second,
		WriteTimeout: 10 * time.Second,
		IdleTimeout:  30 * time.Second,
	}

	go func() {
		if err := srv.ListenAndServe(); err != nil && err != http.ErrServerClosed {
			log.Fatalf("listen: %v", err)
		}
	}()

	quit := make(chan os.Signal, 1)
	signal.Notify(quit, syscall.SIGINT, syscall.SIGTERM)
	<-quit

	ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second)
	defer cancel()
	if err := srv.Shutdown(ctx); err != nil {
		log.Fatal("Server forced to shutdown:", err)
	}
}

Auth и конфигурация

Читайте секреты из переменных окружения и валидируйте при старте. Для JWT-аутентификации используйте middleware с явной проверкой алгоритма:

func AuthMiddleware(secret []byte) gin.HandlerFunc {
	return func(c *gin.Context) {
		tokenStr := strings.TrimPrefix(
			c.GetHeader("Authorization"), "Bearer ",
		)
		token, err := jwt.Parse(tokenStr,
			func(t *jwt.Token) (interface{}, error) {
				if _, ok := t.Method.(*jwt.SigningMethodHMAC); !ok {
					return nil, fmt.Errorf("unexpected alg: %v", t.Header["alg"])
				}
				return secret, nil
			},
		)
		if err != nil || !token.Valid {
			c.AbortWithStatusJSON(401, gin.H{"error": "unauthorized"})
			return
		}
		c.Next()
	}
}

Observability

Подключите OpenTelemetry и Prometheus до первого деплоя:

import (
	"github.com/penglongli/gin-metrics/ginmetrics"
	"go.opentelemetry.io/contrib/instrumentation/github.com/gin-gonic/gin/otelgin"
)

r.Use(otelgin.Middleware("my-service"))

m := ginmetrics.GetMonitor()
m.SetMetricPath("/metrics")
m.Use(r)

Подводные камни

  • Нет middleware recovergin.Default() включает recover, но gin.New() — нет; паника положит процесс.
  • ReadTimeout не выставлен — slow-client держит соединение бесконечно; всегда задавайте http.Server.ReadTimeout.
  • Передача c.Request.Context() в горутину — после завершения хендлера контекст запроса отменяется; создавайте независимый контекст для фоновых задач.
  • Хардкод портов и секретов — используйте os.Getenv + fail fast при пустом значении.
  • Отсутствие rate limiting — без ограничений один клиент может исчерпать пул горутин; используйте golang.org/x/time/rate или nginx.
  • GOMAXPROCS в Docker — без uber-go/automaxprocs Go видит хостовые CPU, а не ограничения cgroup.
  • Игнорирование ошибки Shutdown — если Shutdown не завершился за таймаут, процесс завершается с брошенными соединениями.

What hurts your answer

  • Говорить только о запуске Gin (Go framework), но не об эксплуатации
  • Не упоминать observability, обновления, безопасность и rollback
  • Описывать риски абстрактно, без способов их снижать

What they're listening for

  • Видит production-риски Gin (Go framework)
  • Говорит про monitoring, rollout, rollback и безопасность
  • Умеет ранжировать риски по вероятности и влиянию

Related topics