Avez-vous déjà souhaité que l'orchestration de conteneurs soit plus flexible que les chaînes de dépendances statiques mais plus simple que Kubernetes ? Découvrez PnR (Prompt and Response) - une approche basée sur la configuration qui exploite les puissantes capacités d'abstraction de la plate-forme Go pour orchestrer les conteneurs en fonction des états de préparation réels plutôt que de simples dépendances.
Avant de plonger dans PnR, comprenons pourquoi Go est particulièrement bien adapté à l'orchestration de conteneurs multiplateforme :
Interface API Docker unifiée : la bibliothèque client Docker de Go fournit une interface cohérente sur Windows, Linux et macOS via des connexions socket spécifiques à la plate-forme :
Prise en charge native de la concurrence : les goroutines et les canaux de Go permettent une surveillance efficace des conteneurs :
Gestion du réseau multiplateforme : le package net de Go résume les détails du réseau spécifiques à la plate-forme :
PnR orchestre les conteneurs à travers trois composants clés :
Voyons cela en action avec une pile Web typique : MongoDB, serveur API et client Web.
{ "name": "dev_stack", "cpuxs": { "stack_startup": { "design_chunks": [ { "name": "mongodb", "gatekeeper": { "system_ready": { "prompt": "Is system ready?", "response": ["yes"], "tv": "Y" } }, "flowout": { "mongodb_ready": { "prompt": "Is MongoDB ready?", "response": ["yes"], "tv": "Y" } }, "health_check": { "type": "tcp", "port_key": "27017", "timeout_seconds": 2, "status_mapping": { "success": { "key": "mongodb_ready", "response": ["yes"], "tv": "Y" }, "failure": { "key": "mongodb_ready", "response": ["no"], "tv": "N" } } }, "container": { "name": "pnr_mongodb", "image": "mongo:latest", "ports": { "27017": "27017" } } } ] } } }
Le cœur de PnR est sa gestion des conteneurs indépendante de la plate-forme. Voici comment cela fonctionne :
func (il *ContainerIntentionLoop) Execute() error { // Create platform-specific network _, err := il.dockerClient.NetworkCreate(il.ctx, "pnr_network", types.NetworkCreate{}) if err != nil { return fmt.Errorf("failed to create network: %v", err) } for { // Update runtime state if err := il.updateRTStateFromRuntime(); err != nil { return err } allCompleted := true anyExecuting := false // Process each container for i := range il.cpux.DesignChunks { chunk := &il.cpux.DesignChunks[i] // Container state machine switch chunk.Status { case "completed": continue case "executing": anyExecuting = true allCompleted = false if il.checkChunkCompletion(chunk) { chunk.Status = "completed" } case "", "ready": allCompleted = false if il.checkGatekeeper(chunk) { if err := il.startContainer(chunk); err != nil { return err } chunk.Status = "executing" anyExecuting = true } } } // Check termination conditions if allCompleted { return nil } if !anyExecuting && !allCompleted { return fmt.Errorf("no progress possible - execution stalled") } time.Sleep(5 * time.Second) } }
PnR implémente des contrôles de santé indépendants de la plate-forme à l'aide des bibliothèques standard de Go :
{ "name": "dev_stack", "cpuxs": { "stack_startup": { "design_chunks": [ { "name": "mongodb", "gatekeeper": { "system_ready": { "prompt": "Is system ready?", "response": ["yes"], "tv": "Y" } }, "flowout": { "mongodb_ready": { "prompt": "Is MongoDB ready?", "response": ["yes"], "tv": "Y" } }, "health_check": { "type": "tcp", "port_key": "27017", "timeout_seconds": 2, "status_mapping": { "success": { "key": "mongodb_ready", "response": ["yes"], "tv": "Y" }, "failure": { "key": "mongodb_ready", "response": ["no"], "tv": "N" } } }, "container": { "name": "pnr_mongodb", "image": "mongo:latest", "ports": { "27017": "27017" } } } ] } } }
Installer Go (1.19 ou version ultérieure) :
Installer Docker
func (il *ContainerIntentionLoop) Execute() error { // Create platform-specific network _, err := il.dockerClient.NetworkCreate(il.ctx, "pnr_network", types.NetworkCreate{}) if err != nil { return fmt.Errorf("failed to create network: %v", err) } for { // Update runtime state if err := il.updateRTStateFromRuntime(); err != nil { return err } allCompleted := true anyExecuting := false // Process each container for i := range il.cpux.DesignChunks { chunk := &il.cpux.DesignChunks[i] // Container state machine switch chunk.Status { case "completed": continue case "executing": anyExecuting = true allCompleted = false if il.checkChunkCompletion(chunk) { chunk.Status = "completed" } case "", "ready": allCompleted = false if il.checkGatekeeper(chunk) { if err := il.startContainer(chunk); err != nil { return err } chunk.Status = "executing" anyExecuting = true } } } // Check termination conditions if allCompleted { return nil } if !anyExecuting && !allCompleted { return fmt.Errorf("no progress possible - execution stalled") } time.Sleep(5 * time.Second) } }
func (il *ContainerIntentionLoop) checkChunkCompletion(chunk *DesignChunk) bool { // Platform-agnostic container status check isRunning, err := il.isContainerRunning(chunk.Container.Name) if !isRunning { il.updateChunkStatus(chunk, false) return false } // Health check based on configuration status := false switch chunk.HealthCheck.Type { case "tcp": addr := fmt.Sprintf("localhost:%s", chunk.Container.Ports[chunk.HealthCheck.PortKey]) conn, err := net.DialTimeout("tcp", addr, timeout) if err == nil { conn.Close() status = true } case "http": url := fmt.Sprintf("http://localhost:%s%s", chunk.Container.Ports[chunk.HealthCheck.PortKey], chunk.HealthCheck.Path) resp, err := client.Get(url) if err == nil { status = (resp.StatusCode == chunk.HealthCheck.ExpectedCode) } } il.updateChunkStatus(chunk, status) return status }
pnr-orchestrator/ ├── main.go ├── containers.go ├── config/ │ └── domain.json └── runtime/ # Created automatically
Composition Docker traditionnelle :
# Create project directory mkdir pnr-orchestrator cd pnr-orchestrator # Initialize Go module go mod init pnr-orchestrator # Install dependencies go get github.com/docker/docker/client go get github.com/docker/docker/api/types go get github.com/docker/go-connections/nat
L'orchestration intelligente de PnR :
# Option 1: Direct run go run main.go containers.go # Option 2: Build and run separately go build ./pnr-orchestrator # Unix/Linux/Mac pnr-orchestrator.exe # Windows
La principale différence ? PnR garantit la disponibilité réelle du service sur n'importe quelle plate-forme, pas seulement le démarrage du conteneur.
PnR démontre comment les fortes capacités d'abstraction de la plateforme Go peuvent créer des outils d'orchestration de conteneurs multiplateformes robustes sans sacrifier la simplicité ou la puissance.
Faites-moi savoir dans les commentaires si vous souhaitez voir plus d'exemples ou si vous avez des questions sur les implémentations spécifiques à la plateforme !
Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!