Génération de codes

Complete guide to Goa’s code generation - commands, process, generated code structure, and customization options.

La génération de code de Goa transforme votre conception en code prêt à la production. Plutôt qu’un simple échafaudage, Goa génère des implémentations de services complètes et exécutables qui respectent les meilleures pratiques et maintiennent la cohérence de l’ensemble de votre API.

Outils de ligne de commande

Installation

go install goa.design/goa/v3/cmd/goa@latest

Commandes

Toutes les commandes s’appuient sur les chemins d’importation des paquets Go, et non sur les chemins d’accès au système de fichiers :

# ✅ Correct: using Go package import path
goa gen goa.design/examples/calc/design

# ❌ Incorrect: using filesystem path
goa gen ./design

Générer du code (goa gen)

goa gen <design-package-import-path> [-o <output-dir>]

La commande principale pour la génération de code :

  • Traite votre paquetage de conception et génère le code d’implémentation
  • Recrée à chaque fois l’intégralité du répertoire gen/ à partir de zéro
  • Exécution après chaque modification de la conception

Créer un exemple (goa example)

goa example <design-package-import-path> [-o <output-dir>]

Commande d’échafaudage :

  • Crée une implémentation d’exemple unique
  • Génère des stubs de handler avec la logique de l’exemple
  • S’exécute une fois lors du démarrage d’un nouveau projet
  • N’écrase PAS l’implémentation personnalisée existante

Afficher la version

goa version

Workflow de développement

  1. Création de la conception initiale
  2. Exécuter goa gen pour générer le code de base
  3. Exécuter goa example pour créer des stubs d’implémentation
  4. Implémentez votre logique de service
  5. Exécutez goa gen après chaque changement de conception

Meilleure pratique: Engagez le code généré dans le contrôle de version plutôt que de le générer pendant la CI/CD. Cela garantit des constructions reproductibles et permet de suivre les changements dans le code généré.


Processus de génération

Lorsque vous exécutez goa gen, Goa suit un processus systématique :

1. Phase d’amorçage

Goa crée un main.go temporaire qui :

  • Importe les paquets Goa et votre paquetage de conception
  • Exécute l’évaluation du DSL
  • Déclenche la génération de code

2. Évaluation de la conception

  • Les fonctions DSL s’exécutent pour créer des objets d’expression
  • Les expressions se combinent en un modèle API complet
  • Les relations entre les expressions sont établies
  • Les règles et contraintes de conception sont validées

3. Génération du code

  • Les expressions validées sont transmises aux générateurs de code
  • Les modèles sont rendus pour produire des fichiers de code
  • La sortie est écrite dans le répertoire gen/

Structure du code généré

Un projet typique généré :

myservice/
├── cmd/                    # Generated example commands
│   └── calc/
│       ├── grpc.go
│       └── http.go
├── design/                 # Your design files
│   └── design.go
├── gen/                    # Generated code (don't edit)
│   ├── calc/               # Service-specific code
│   │   ├── client.go
│   │   ├── endpoints.go
│   │   └── service.go
│   ├── http/               # HTTP transport layer
│   │   ├── calc/
│   │   │   ├── client/
│   │   │   └── server/
│   │   └── openapi.json
│   └── grpc/               # gRPC transport layer
│       └── calc/
│           ├── client/
│           ├── server/
│           └── pb/
└── myservice.go            # Your service implementation

Interfaces de service

Générées dans gen/<service>/service.go :

// Service interface defines the API contract
type Service interface {
    Add(context.Context, *AddPayload) (res int, err error)
    Multiply(context.Context, *MultiplyPayload) (res int, err error)
}

// Payload types
type AddPayload struct {
    A int32
    B int32
}

// Constants for observability
const ServiceName = "calc"
var MethodNames = [2]string{"add", "multiply"}

Couche d’extrémité

Généré dans gen/<service>/endpoints.go :

// Endpoints wraps service methods in transport-agnostic endpoints
type Endpoints struct {
    Add      goa.Endpoint
    Multiply goa.Endpoint
}

// NewEndpoints creates endpoints from service implementation
func NewEndpoints(s Service) *Endpoints {
    return &Endpoints{
        Add:      NewAddEndpoint(s),
        Multiply: NewMultiplyEndpoint(s),
    }
}

// Use applies middleware to all endpoints
func (e *Endpoints) Use(m func(goa.Endpoint) goa.Endpoint) {
    e.Add = m(e.Add)
    e.Multiply = m(e.Multiply)
}

Exemple d’intergiciel de point final :

func LoggingMiddleware(next goa.Endpoint) goa.Endpoint {
    return func(ctx context.Context, req any) (res any, err error) {
        log.Printf("request: %v", req)
        res, err = next(ctx, req)
        log.Printf("response: %v", res)
        return
    }
}

endpoints.Use(LoggingMiddleware)

Code client

Généré dans gen/<service>/client.go :

// Client provides typed methods for service calls
type Client struct {
    AddEndpoint      goa.Endpoint
    MultiplyEndpoint goa.Endpoint
}

func NewClient(add, multiply goa.Endpoint) *Client {
    return &Client{
        AddEndpoint:      add,
        MultiplyEndpoint: multiply,
    }
}

func (c *Client) Add(ctx context.Context, p *AddPayload) (res int, err error) {
    ires, err := c.AddEndpoint(ctx, p)
    if err != nil {
        return
    }
    return ires.(int), nil
}

Génération de code HTTP

Mise en œuvre du serveur

Généré dans gen/http/<service>/server/server.go :

func New(
    e *calc.Endpoints,
    mux goahttp.Muxer,
    decoder func(*http.Request) goahttp.Decoder,
    encoder func(context.Context, http.ResponseWriter) goahttp.Encoder,
    errhandler func(context.Context, http.ResponseWriter, error),
    formatter func(ctx context.Context, err error) goahttp.Statuser,
) *Server

// Server exposes handlers for modification
type Server struct {
    Mounts   []*MountPoint
    Add      http.Handler
    Multiply http.Handler
}

// Use applies HTTP middleware to all handlers
func (s *Server) Use(m func(http.Handler) http.Handler)

Configuration complète du serveur :

func main() {
    svc := calc.New()
    endpoints := gencalc.NewEndpoints(svc)
    mux := goahttp.NewMuxer()
    server := genhttp.New(
        endpoints,
        mux,
        goahttp.RequestDecoder,
        goahttp.ResponseEncoder,
        nil, nil)
    genhttp.Mount(mux, server)
    http.ListenAndServe(":8080", mux)
}

Mise en œuvre du client

Généré dans gen/http/<service>/client/client.go :

func NewClient(
    scheme string,
    host string,
    doer goahttp.Doer,
    enc func(*http.Request) goahttp.Encoder,
    dec func(*http.Response) goahttp.Decoder,
    restoreBody bool,
) *Client

Configuration complète du client :

func main() {
    httpClient := genclient.NewClient(
        "http",
        "localhost:8080",
        http.DefaultClient,
        goahttp.RequestEncoder,
        goahttp.ResponseDecoder,
        false,
    )

    client := gencalc.NewClient(
        httpClient.Add(),
        httpClient.Multiply(),
    )

    result, err := client.Add(context.Background(), &gencalc.AddPayload{A: 1, B: 2})
}

Génération de code gRPC

Définition de Protobuf

Généré dans gen/grpc/<service>/pb/ :

syntax = "proto3";
package calc;

service Calc {
    rpc Add (AddRequest) returns (AddResponse);
    rpc Multiply (MultiplyRequest) returns (MultiplyResponse);
}

message AddRequest {
    int64 a = 1;
    int64 b = 2;
}

Implémentation du serveur

func main() {
    svc := calc.New()
    endpoints := gencalc.NewEndpoints(svc)
    svr := grpc.NewServer()
    gensvr := gengrpc.New(endpoints, nil)
    genpb.RegisterCalcServer(svr, gensvr)
    lis, _ := net.Listen("tcp", ":8080")
    svr.Serve(lis)
}

Implémentation du client

func main() {
    conn, _ := grpc.Dial("localhost:8080",
        grpc.WithTransportCredentials(insecure.NewCredentials()))
    defer conn.Close()

    grpcClient := genclient.NewClient(conn)
    client := gencalc.NewClient(
        grpcClient.Add(),
        grpcClient.Multiply(),
    )

    result, _ := client.Add(context.Background(), &gencalc.AddPayload{A: 1, B: 2})
}

Personnalisation

Contrôle de la génération des types

Force la génération de types qui ne sont pas directement référencés par les méthodes :

var MyType = Type("MyType", func() {
    // Force generation in specific services
    Meta("type:generate:force", "service1", "service2")
    
    // Or force generation in all services
    Meta("type:generate:force")
    
    Attribute("name", String)
})

Organisation du paquet

Générer des types dans un paquetage partagé :

var CommonType = Type("CommonType", func() {
    Meta("struct:pkg:path", "types")
    Attribute("id", String)
})

Crée :

gen/
└── types/
    └── common_type.go

Personnalisation des champs

var Message = Type("Message", func() {
    Attribute("id", String, func() {
        // Override field name
        Meta("struct:field:name", "ID")
        
        // Add custom struct tags
        Meta("struct:tag:json", "id,omitempty")
        Meta("struct:tag:msgpack", "id,omitempty")
        
        // Override type
        Meta("struct:field:type", "bson.ObjectId", "github.com/globalsign/mgo/bson", "bson")
    })
})

Personnalisation de la mémoire tampon du protocole

var MyType = Type("MyType", func() {
    // Override protobuf message name
    Meta("struct:name:proto", "CustomProtoType")
    
    Field(1, "status", Int32, func() {
        // Override protobuf field type
        Meta("struct:field:proto", "int32")
    })

    // Use Google's timestamp type
    Field(2, "created_at", String, func() {
        Meta("struct:field:proto", 
            "google.protobuf.Timestamp",
            "google/protobuf/timestamp.proto",
            "Timestamp",
            "google.golang.org/protobuf/types/known/timestamppb")
    })
})

// Specify protoc include paths
var _ = API("calc", func() {
    Meta("protoc:include", "/usr/include", "/usr/local/include")
})

Personnalisation de l’OpenAPI

var _ = API("MyAPI", func() {
    // Control generation
    Meta("openapi:generate", "false")
    
    // Format JSON output
    Meta("openapi:json:prefix", "  ")
    Meta("openapi:json:indent", "  ")
    
    // Disable example generation
    Meta("openapi:example", "false")
})

var _ = Service("UserService", func() {
    // Add tags
    HTTP(func() {
        Meta("openapi:tag:Users")
        Meta("openapi:tag:Backend:desc", "Backend API Operations")
    })
    
    Method("CreateUser", func() {
        // Custom operation ID
        Meta("openapi:operationId", "{service}.{method}")
        
        // Custom summary
        Meta("openapi:summary", "Create a new user")
        
        HTTP(func() {
            // Add extensions
            Meta("openapi:extension:x-rate-limit", `{"rate": 100}`)
            POST("/users")
        })
    })
})

var User = Type("User", func() {
    // Override type name in OpenAPI spec
    Meta("openapi:typename", "CustomUser")
})

Types et validation

Application de la validation

Goa valide les données aux limites du système :

  • Côté serveur : Validation des requêtes entrantes
  • Côté client : Valide les réponses entrantes
  • Code interne : Confiance dans le maintien des invariants

Règles de pointeur pour les champs de structure

Propriétés de la structure : charge utile/résultat, corps de la demande (serveur), corps de la réponse (serveur), etc
Requise OU par défautDirecte (-)Pointeur (*)Directe (-)
Pointeur (*)Pointeur (*)Pointeur (*)Pointeur (*)

Types spéciaux :

  • Objets (structs) : Toujours utiliser des pointeurs
  • Tableaux et cartes : N’utilisent jamais de pointeurs (ce sont déjà des types de référence)

Exemple :

type Person struct {
    Name     string             // required, direct value
    Age      *int               // optional, pointer
    Hobbies  []string           // array, no pointer
    Metadata map[string]string  // map, no pointer
}

Traitement des valeurs par défaut

  • Marshaling : Les valeurs par défaut initialisent les tableaux/maps nuls
  • **Les valeurs par défaut s’appliquent aux champs optionnels manquants (et non aux champs obligatoires manquants) : Les valeurs par défaut s’appliquent aux champs optionnels manquants (et non aux champs obligatoires manquants)

Vues et types de résultats

Les vues contrôlent la manière dont les types de résultats sont affichés dans les réponses.

Fonctionnement des vues

  1. La méthode de service comprend un paramètre de vue
  2. Un paquet de vues est généré au niveau du service
  3. La validation spécifique à la vue est automatiquement générée

Réponse côté serveur

  1. Le type de résultat visualisé est marshallé
  2. Les attributs Nil sont omis
  3. Le nom de la vue est transmis dans l’en-tête “Goa-View”

Réponse côté client

  1. La réponse est unmarshalled
  2. Transformée en type de résultat visualisé
  3. Nom de la vue extrait de l’en-tête “Goa-View”
  4. Validation spécifique à la vue effectuée
  5. Reconverti en type de résultat de service

Vue par défaut

Si aucune vue n’est définie, Goa ajoute une vue “par défaut” qui comprend tous les champs de base.


Système de plugin

Le système de plugins de Goa étend la génération de code. Les plugins peuvent :

  1. Ajouter de nouveaux DSL - Constructions de langage de conception supplémentaires
  2. Modifier le code généré - Inspecter et modifier les fichiers, ajouter de nouveaux fichiers

Exemple utilisant le plugin CORS :

import (
    . "goa.design/goa/v3/dsl"
    cors "goa.design/plugins/v3/cors/dsl"
)

var _ = Service("calc", func() {
    cors.Origin("/.*localhost.*/", func() {
        cors.Headers("X-Shared-Secret")
        cors.Methods("GET", "POST")
    })
})

Cas d’utilisation courants des plugins :

  • Prise en charge des protocoles (CORS, etc.)
  • Formats de documentation supplémentaires
  • Règles de validation personnalisées
  • Questions transversales (journalisation, métriques)
  • Génération de fichiers de configuration

Voir aussi

  • DSL Reference - Référence DSL complète pour les fichiers de conception
  • Guide HTTP - Fonctionnalités de transport HTTP et personnalisation
  • Guide gRPC - Fonctionnalités de transport gRPC et tampons de protocole
  • Quickstart - Démarrage avec la génération de code