Skip to content

gRPC

The pkg/grpc package provides a standard gRPC server implementation that integrates with the controls package for lifecycle management and observability.

Features

  • Standard Observability: Implements the standard gRPC Health Checking Protocol.
  • Named Probes: Supports liveness and readiness service names for orchestrator integration.
  • Reflection: Built-in support for gRPC reflection (enabled by default).

Functions

  • NewServer(cfg config.Containable, opts ...any) (*grpc.Server, error): Returns a new *grpc.Server with reflection registered. The variadic accepts both ServerOption values (e.g. WithConfigPrefix, which selects the config block the reflection flag is read from) and grpc.ServerOption values.
  • Start(cfg config.Containable, logger logger.Logger, srv *grpc.Server, opts ...ServerOption) controls.StartFunc: Returns a controller start function. Pass WithConfigPrefix/WithPort to target a custom server; it logs the bound listener address (so an ephemeral :0 port surfaces resolved).
  • RegisterHealthService(srv *grpc.Server, controller controls.Controllable): Wires the gRPC health service to the controller status.
  • Register(ctx context.Context, id string, controller controls.Controllable, cfg config.Containable, logger logger.Logger, opts ...any) (*grpc.Server, error): Creates a server, registers the health service, adds it to the controller, and returns the server instance. Accepts ServerOption, RegisterOption and grpc.ServerOption values.

Server Options

ServerOption values select the config block (and port) a gRPC server uses, so you can run more than one gRPC server in a process. They are accepted by NewServer, Start, DialLocal and Register:

  • WithConfigPrefix(prefix string) ServerOption: Config prefix for port, reflection and TLS (default server.grpc). The keys become <prefix>.port, <prefix>.reflection, <prefix>.tls.*.
  • WithPort(port int) ServerOption: Explicit listen/dial port, bypassing config lookup (overrides <prefix>.port and the server.port fallback).
// A second gRPC server on its own config block (server.internal.*):
srv, _ := gtbgrpc.Register(ctx, "internal", controller, props.Config, props.Logger,
    gtbgrpc.WithConfigPrefix("server.internal"))

// ...and a gateway/in-process client dialling that same server:
conn, _ := gtbgrpc.DialLocal(props.Config, gtbgrpc.WithConfigPrefix("server.internal"))

Interceptor Chaining

Interceptors are gRPC's expression of the same chain pattern used by HTTP middleware. For the unified transport story — server/client × HTTP/gRPC, the resilience composition rules, and the config-prefix convention — see the Transport Middleware & Resilience concept.

The package provides an interceptor chaining API for composing gRPC unary and stream interceptors.

  • NewInterceptorChain(interceptors ...Interceptor) InterceptorChain: Creates a chain from paired unary/stream interceptors.
  • (c InterceptorChain) Append(interceptors ...Interceptor) InterceptorChain: Returns a new chain with additional interceptors (immutable).
  • (c InterceptorChain) ServerOptions() []grpc.ServerOption: Returns grpc.ChainUnaryInterceptor and grpc.ChainStreamInterceptor options.
  • WithInterceptors(chain InterceptorChain) RegisterOption: Applies an interceptor chain when using Register.

Built-in Logging Interceptor

LoggingInterceptor logs each completed RPC with structured fields (method, status code, latency, RPC type).

  • LoggingInterceptor(logger logger.Logger, opts ...GRPCLoggingOption) Interceptor

Options: WithGRPCLogLevel, WithoutGRPCLatency, WithGRPCPathFilter.

Built-in Rate-Limit Interceptor

RateLimitInterceptor protects a gRPC server from overload, mirroring the HTTP server limiter. It admits RPCs under a token-bucket limiter and rejects excess with codes.ResourceExhausted. It is an Interceptor (unary + stream), so it composes into any InterceptorChain.

  • RateLimitInterceptor(log logger.Logger, cfg RateLimitConfig) Interceptor
  • DefaultRateLimitConfig() RateLimitConfig — 50 rps, burst 100, single global bucket
  • PeerKey(ctx, fullMethod) string — a ready-made per-peer KeyFunc
  • RateLimitConfigFromConfig(cfg config.Containable, prefix string) RateLimitConfig

Admission is non-blocking (Allow, not Wait). Per-method or per-client scoping is achieved with KeyFunc (key on fullMethod, or use PeerKey); the per-key bucket store is bounded and LRU-evicting (MaxTrackedKeys, default 8192). Config keys live under server.grpc.ratelimit.* (requests_per_second, burst, max_tracked_keys).

Built-in Circuit Breaker (client)

CircuitBreakerInterceptor and CircuitBreakerStreamInterceptor are client interceptors that fail fast while a downstream is consistently failing, sharing the same Closed/Open/HalfOpen core as the HTTP breaker. While open they reject calls with codes.Unavailable — indistinguishable on the wire from a genuine outage.

  • CircuitBreakerInterceptor(log logger.Logger, cfg CircuitBreakerConfig) grpc.UnaryClientInterceptor — install via grpc.WithChainUnaryInterceptor
  • CircuitBreakerStreamInterceptor(log logger.Logger, cfg CircuitBreakerConfig) grpc.StreamClientInterceptor
  • DefaultCircuitBreakerConfig() — threshold 5, cooldown 30s, half-open trial 1
  • CircuitBreakerConfigFromConfig(cfg config.Containable, prefix string)

By default only Unavailable and DeadlineExceeded count as failures. ResourceExhausted does not trip the breaker — like an HTTP 429 it is a "slow down" signal (retry's domain), not a downstream-health signal, so a server's own rate limiter cannot trip its callers' breakers. The stream breaker inspects per-message errors (a RecvMsg/SendMsg returning a classified failure), not just stream establishment; a clean io.EOF closes the stream as a success. Config keys live under server.grpc.circuitbreaker.*.

Built-in Authentication Interceptor

AuthInterceptor authenticates (and optionally authorizes) each RPC from an pkg/authn verifier — an API key or JWT/OIDC bearer token from metadata, or an mTLS client certificate from the peer — storing the verified identity in the RPC context. It is a paired unary + stream Interceptor; the stream check runs once at stream open.

  • AuthInterceptor(opts ...GRPCAuthOption) (Interceptor, error) — fail-closed: no verifier is a construction error
  • IdentityFromContext(ctx context.Context) (*authn.Identity, bool) — read the identity in a handler (same key as the HTTP middleware)
authIC, _ := gtbgrpc.AuthInterceptor(gtbgrpc.WithGRPCBearerVerifier(jwtVerifier))
chain := gtbgrpc.NewInterceptorChain(gtbgrpc.LoggingInterceptor(log), authIC)
gtbgrpc.Register(ctx, "api", controller, cfg, log, gtbgrpc.WithInterceptors(chain))

Options: WithGRPCBearerVerifier, WithGRPCAPIKeyMetadata, WithGRPCMTLSVerifier, WithGRPCAuthorize, WithGRPCAuthLogger, WithGRPCMethodSkipper. Failures yield a generic codes.Unauthenticated / codes.PermissionDenied with the cause logged redacted. The standard health (/grpc.health.v1.Health/*) and reflection (/grpc.reflection.v1*) services are auto-skipped so probes keep working; a custom skipper adds to that set. See Authentication & Authorization for the full reference.

TLS

The gRPC server supports TLS using the shared hardened configuration from pkg/tls (TLS 1.2 minimum, curated AEAD cipher suites, X25519 curve preference). The TLS listener advertises HTTP/2 via ALPN (h2); without it, grpc-go 1.67+ clients — including the gateway — refuse the connection with "missing selected ALPN property". The Register/Start path sets this for you.

Configuration

TLS configuration cascades — transport-specific keys override the shared defaults:

Key Shared Default gRPC Override
Enabled server.tls.enabled server.grpc.tls.enabled
Certificate server.tls.cert server.grpc.tls.cert
Private key server.tls.key server.grpc.tls.key

To use the same certificate for both HTTP and gRPC, configure the shared keys only:

server:
  tls:
    enabled: true
    cert: /etc/certs/server.crt
    key: /etc/certs/server.key

To use different certificates per transport:

server:
  tls:
    enabled: true
    cert: /etc/certs/http.crt
    key: /etc/certs/http.key
  grpc:
    tls:
      cert: /etc/certs/grpc.crt
      key: /etc/certs/grpc.key

Direct Credential Construction

For cases where you need to pass TLS credentials directly to grpc.NewServer (e.g. when not using the Register helper):

creds, err := gtbgrpc.TLSServerCredentials("/path/to/cert.pem", "/path/to/key.pem")
if err != nil {
    return err
}

srv := grpc.NewServer(grpc.Creds(creds))

This uses the same shared hardened TLS config from pkg/tls as the automatic setup. (credentials.NewTLS advertises h2 itself, so no explicit ALPN is needed on this path.)

Client Credentials and Local Dialling

The package also provides the client side, used for example by the gateway when it dials the gRPC server over a self-signed or private-CA certificate:

  • TLSClientCredentials(caFiles ...string) (credentials.TransportCredentials, error): client transport credentials trusting the given CA/cert files — the mirror of TLSServerCredentials. With no files it trusts the system roots.
  • DialLocal(cfg config.Containable, opts ...any) (*grpc.ClientConn, error): dials the gRPC server described by cfg over the loopback interface, with transport security that matches the server's own config (<prefix>.tls cascading to server.tls). The variadic accepts both ServerOption values (e.g. WithConfigPrefix to dial a non-default server) and grpc.DialOption values. Intended for in-process callers such as the gateway, so they connect without re-deriving the endpoint or credentials by hand.
// Connect to the local gRPC server with matching transport security in one call.
conn, err := gtbgrpc.DialLocal(props.Config)
if err != nil {
    return err
}

Config Keys

DefaultConfigPrefix (server.grpc) is the default config block; WithConfigPrefix derives <prefix>.port, <prefix>.reflection and <prefix>.tls.* from it. The shared ConfigKeySharedPort (server.port) is the fallback when the per-server port is unset.

Constant Key Notes
DefaultConfigPrefix server.grpc Default prefix; override with WithConfigPrefix.
ConfigKeySharedPort server.port Fallback when the per-server port is unset.
ConfigKeyPort server.grpc.port Deprecated — prefer WithConfigPrefix / <prefix>.port.
ConfigKeyReflection server.grpc.reflection Deprecated — prefer WithConfigPrefix.
ConfigTLSPrefix server.grpc.tls Deprecated — the TLS prefix is <prefix>.tls.

Usage Example

// Build an interceptor chain with logging
chain := gtbgrpc.NewInterceptorChain(
    gtbgrpc.LoggingInterceptor(props.Logger,
        gtbgrpc.WithGRPCPathFilter("/grpc.health.v1.Health/Check"),
    ),
)

// Register with interceptors
srv, err := gtbgrpc.Register(ctx, "grpc-api", controller, props.Config, props.Logger,
    gtbgrpc.WithInterceptors(chain),
)
if err != nil {
    return err
}

// Register your custom services
pb.RegisterMyServiceServer(srv, &myServiceImpl{})