mirror of
				https://github.com/siderolabs/talos.git
				synced 2025-10-26 14:01:39 +01:00 
			
		
		
		
	This removes use of private fork with custom `ip=` kernel argument handling and switches fully to upstream version of it. Firecracker Go SDK version is `master` + following PRs: * https://github.com/firecracker-microvm/firecracker-go-sdk/pull/167 * https://github.com/firecracker-microvm/firecracker-go-sdk/pull/177 * https://github.com/firecracker-microvm/firecracker-go-sdk/pull/178 MTU handling support was implemented as well. Changes: * hostname to each node is passed via `talos.hostname=` kernel arg * IP configuration is generated by SDK from CNI result * fixed bugs with wrong netmask * nameservers & MTU is passed via Talos config Signed-off-by: Andrey Smirnov <smirnov.andrey@gmail.com>
		
			
				
	
	
		
			137 lines
		
	
	
		
			3.6 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			137 lines
		
	
	
		
			3.6 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // This Source Code Form is subject to the terms of the Mozilla Public
 | |
| // License, v. 2.0. If a copy of the MPL was not distributed with this
 | |
| // file, You can obtain one at http://mozilla.org/MPL/2.0/.
 | |
| 
 | |
| package firecracker
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"encoding/json"
 | |
| 	"fmt"
 | |
| 	"os"
 | |
| 	"os/signal"
 | |
| 	"strings"
 | |
| 	"syscall"
 | |
| 	"time"
 | |
| 
 | |
| 	"github.com/firecracker-microvm/firecracker-go-sdk"
 | |
| 
 | |
| 	"github.com/talos-systems/talos/internal/pkg/provision/providers/firecracker/inmemhttp"
 | |
| )
 | |
| 
 | |
| // LaunchConfig is passed in to the Launch function over stdin.
 | |
| type LaunchConfig struct {
 | |
| 	GatewayAddr       string
 | |
| 	Config            string
 | |
| 	FirecrackerConfig firecracker.Config
 | |
| }
 | |
| 
 | |
| // Launch a control process around firecracker VM manager.
 | |
| //
 | |
| // This function is invoked from 'osctl firecracker-launch' hidden command
 | |
| // and wraps starting, controlling and restarting 'firecracker' VM process.
 | |
| //
 | |
| // Launch restarts VM forever until control process is stopped itself with a signal.
 | |
| //
 | |
| // Process is expected to receive configuration on stdin. Current working directory
 | |
| // should be cluster state directory, process output should be redirected to the
 | |
| // logfile in state directory.
 | |
| //
 | |
| // When signals SIGINT, SIGTERM are received, control process stops firecracker and exits.
 | |
| //
 | |
| //nolint: gocyclo
 | |
| func Launch() error {
 | |
| 	var config LaunchConfig
 | |
| 
 | |
| 	d := json.NewDecoder(os.Stdin)
 | |
| 
 | |
| 	if err := d.Decode(&config); err != nil {
 | |
| 		return fmt.Errorf("error decoding config from stdin: %w", err)
 | |
| 	}
 | |
| 
 | |
| 	if d.More() {
 | |
| 		return fmt.Errorf("extra unexpected input on stdin")
 | |
| 	}
 | |
| 
 | |
| 	if err := os.Stdin.Close(); err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	signal.Ignore(syscall.SIGHUP)
 | |
| 
 | |
| 	c := make(chan os.Signal, 1)
 | |
| 	signal.Notify(c, syscall.SIGTERM, syscall.SIGINT)
 | |
| 
 | |
| 	ctx := context.Background()
 | |
| 
 | |
| 	httpServer, err := inmemhttp.NewServer(fmt.Sprintf("%s:0", config.GatewayAddr))
 | |
| 	if err != nil {
 | |
| 		return fmt.Errorf("error launching in-memory HTTP server: %w", err)
 | |
| 	}
 | |
| 
 | |
| 	if err = httpServer.AddFile("config.yaml", []byte(config.Config)); err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	// patch kernel args
 | |
| 	config.FirecrackerConfig.KernelArgs = strings.ReplaceAll(config.FirecrackerConfig.KernelArgs, "{TALOS_CONFIG_URL}", fmt.Sprintf("http://%s/config.yaml", httpServer.GetAddr()))
 | |
| 
 | |
| 	httpServer.Serve()
 | |
| 	defer httpServer.Shutdown(ctx) //nolint: errcheck
 | |
| 
 | |
| 	for {
 | |
| 		cmd := firecracker.VMCommandBuilder{}.
 | |
| 			WithBin("firecracker").
 | |
| 			WithSocketPath(config.FirecrackerConfig.SocketPath).
 | |
| 			WithStdin(os.Stdin).
 | |
| 			WithStdout(os.Stdout).
 | |
| 			WithStderr(os.Stderr).
 | |
| 			Build(ctx)
 | |
| 
 | |
| 		// reset static configuration, as it gets set each time CNI runs
 | |
| 		config.FirecrackerConfig.NetworkInterfaces[0].StaticConfiguration = nil
 | |
| 
 | |
| 		m, err := firecracker.NewMachine(ctx, config.FirecrackerConfig, firecracker.WithProcessRunner(cmd))
 | |
| 		if err != nil {
 | |
| 			return fmt.Errorf("failed to create new machine: %w", err)
 | |
| 		}
 | |
| 
 | |
| 		if err := m.Start(ctx); err != nil {
 | |
| 			return fmt.Errorf("failed to initialize machine: %w", err)
 | |
| 		}
 | |
| 
 | |
| 		waitCh := make(chan error)
 | |
| 
 | |
| 		go func() {
 | |
| 			waitCh <- m.Wait(ctx)
 | |
| 		}()
 | |
| 
 | |
| 		select {
 | |
| 		case err := <-waitCh:
 | |
| 			if err != nil {
 | |
| 				return fmt.Errorf("failed running VM: %w", err)
 | |
| 			}
 | |
| 
 | |
| 			select {
 | |
| 			case sig := <-c:
 | |
| 				fmt.Fprintf(os.Stderr, "exiting VM as signal %s was received\n", sig)
 | |
| 
 | |
| 				return nil
 | |
| 
 | |
| 			case <-time.After(500 * time.Millisecond): // wait a bit to prevent crash loop
 | |
| 			}
 | |
| 		case sig := <-c:
 | |
| 			fmt.Fprintf(os.Stderr, "stopping VM as signal %s was received\n", sig)
 | |
| 
 | |
| 			m.StopVMM() //nolint: errcheck
 | |
| 
 | |
| 			<-waitCh // wait for process to exit
 | |
| 
 | |
| 			return nil
 | |
| 		}
 | |
| 
 | |
| 		// restart the vm by proceeding with the for loop
 | |
| 		os.Remove(config.FirecrackerConfig.SocketPath) //nolint: errcheck
 | |
| 	}
 | |
| }
 |