Skip to content

Commit

Permalink
platform: implement the kola elements for Hetzner
Browse files Browse the repository at this point in the history
Signed-off-by: Mathieu Tortuyaux <mtortuyaux@microsoft.com>
  • Loading branch information
tormath1 committed Jun 10, 2024
1 parent 507332b commit e27a6b3
Show file tree
Hide file tree
Showing 5 changed files with 269 additions and 1 deletion.
9 changes: 8 additions & 1 deletion cmd/kola/options.go
Original file line number Diff line number Diff line change
Expand Up @@ -40,7 +40,7 @@ var (
kolaOffering string
defaultTargetBoard = sdk.DefaultBoard()
kolaArchitectures = []string{"amd64"}
kolaPlatforms = []string{"aws", "azure", "brightbox", "do", "esx", "external", "gce", "openstack", "equinixmetal", "qemu", "qemu-unpriv", "scaleway"}
kolaPlatforms = []string{"aws", "azure", "brightbox", "do", "esx", "external", "gce", "hetzner", "openstack", "equinixmetal", "qemu", "qemu-unpriv", "scaleway"}
kolaDistros = []string{"cl", "fcos", "rhcos"}
kolaChannels = []string{"alpha", "beta", "stable", "edge", "lts"}
kolaOfferings = []string{"basic", "pro"}
Expand Down Expand Up @@ -244,6 +244,12 @@ func init() {
sv(&kola.ScalewayOptions.SecretKey, "scaleway-secret-key", "", "Scaleway credentials secret key")
sv(&kola.ScalewayOptions.Image, "scaleway-image", "", "Scaleway image ID")
sv(&kola.ScalewayOptions.InstanceType, "scaleway-instance-type", "DEV1-S", "Scaleway instance type")

// Hetzner specific options
sv(&kola.HetznerOptions.Token, "hetzner-token", "", "Hetzner token for client authentication")
sv(&kola.HetznerOptions.Location, "hetzner-location", "fsn1-dc8", "Hetzner location name")
sv(&kola.HetznerOptions.Image, "hetzner-image", "", "Hetzner image ID")
sv(&kola.HetznerOptions.InstanceType, "hetzner-instance-type", "cx11", "Hetzner instance type")
}

// Sync up the command line options if there is dependency
Expand All @@ -264,6 +270,7 @@ func syncOptions() error {
kola.EquinixMetalOptions.GSOptions = &kola.GCEOptions
kola.BrightboxOptions.Board = board
kola.ScalewayOptions.Board = board
kola.HetznerOptions.Board = board

validateOption := func(name, item string, valid []string) error {
for _, v := range valid {
Expand Down
5 changes: 5 additions & 0 deletions kola/harness.go
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,7 @@ import (
equinixmetalapi "github.com/flatcar/mantle/platform/api/equinixmetal"
esxapi "github.com/flatcar/mantle/platform/api/esx"
gcloudapi "github.com/flatcar/mantle/platform/api/gcloud"
hetznerapi "github.com/flatcar/mantle/platform/api/hetzner"
openstackapi "github.com/flatcar/mantle/platform/api/openstack"
scalewayapi "github.com/flatcar/mantle/platform/api/scaleway"
"github.com/flatcar/mantle/platform/conf"
Expand All @@ -56,6 +57,7 @@ import (
"github.com/flatcar/mantle/platform/machine/esx"
"github.com/flatcar/mantle/platform/machine/external"
"github.com/flatcar/mantle/platform/machine/gcloud"
"github.com/flatcar/mantle/platform/machine/hetzner"
"github.com/flatcar/mantle/platform/machine/openstack"
"github.com/flatcar/mantle/platform/machine/qemu"
"github.com/flatcar/mantle/platform/machine/scaleway"
Expand All @@ -78,6 +80,7 @@ var (
EquinixMetalOptions = equinixmetalapi.Options{Options: &Options} // glue to set platform options from main
QEMUOptions = qemu.Options{Options: &Options} // glue to set platform options from main
ScalewayOptions = scalewayapi.Options{Options: &Options} // glue to set platform options from main
HetznerOptions = hetznerapi.Options{Options: &Options} // glue to set platform options from main

TestParallelism int //glue var to set test parallelism from main
TAPFile string // if not "", write TAP results here
Expand Down Expand Up @@ -241,6 +244,8 @@ func NewFlight(pltfrm string) (flight platform.Flight, err error) {
flight, err = external.NewFlight(&ExternalOptions)
case "gce":
flight, err = gcloud.NewFlight(&GCEOptions)
case "hetzner":
flight, err = hetzner.NewFlight(&HetznerOptions)
case "openstack":
flight, err = openstack.NewFlight(&OpenStackOptions)
case "equinixmetal":
Expand Down
91 changes: 91 additions & 0 deletions platform/machine/hetzner/cluster.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,91 @@
// Copyright The Mantle Authors.
// SPDX-License-Identifier: Apache-2.0

package hetzner

import (
"context"
"crypto/rand"
"fmt"
"os"
"path/filepath"

"github.com/flatcar/mantle/platform"
"github.com/flatcar/mantle/platform/conf"
)

type cluster struct {
*platform.BaseCluster
flight *flight
}

func (bc *cluster) NewMachine(userdata *conf.UserData) (platform.Machine, error) {
conf, err := bc.RenderUserData(userdata, map[string]string{
"$public_ipv4": "${COREOS_CUSTOM_PUBLIC_IPV4}",
"$private_ipv4": "${COREOS_CUSTOM_PRIVATE_IPV4}",
})
if err != nil {
return nil, err
}

// Hack to workaround CT inheritance.
// Can be dropped once we remove CT dependency.
// https://github.com/flatcar/Flatcar/issues/1386
conf.AddSystemdUnitDropin("coreos-metadata.service", "00-custom-metadata.conf", `[Service]
ExecStartPost=/usr/bin/sed -i "s/HETZNER/CUSTOM/" /run/metadata/flatcar
ExecStartPost=/usr/bin/sed -i "s/IPV4_PRIVATE/PRIVATE_IPV4/" /run/metadata/flatcar
ExecStartPost=/usr/bin/sed -i "s/IPV4_PUBLIC/PUBLIC_IPV4/" /run/metadata/flatcar
`)

instance, err := bc.flight.api.CreateServer(context.TODO(), bc.vmname(), conf.String())
if err != nil {
return nil, err
}

mach := &machine{
cluster: bc,
mach: instance,
}

// machine to destroy
m := mach
defer func() {
if m != nil {
m.Destroy()
}
}()

mach.dir = filepath.Join(bc.RuntimeConf().OutputDir, mach.ID())
if err := os.Mkdir(mach.dir, 0777); err != nil {
return nil, err
}

confPath := filepath.Join(mach.dir, "ignition.json")
if err := conf.WriteFile(confPath); err != nil {
return nil, err
}

if mach.journal, err = platform.NewJournal(mach.dir); err != nil {
return nil, err
}

if err := platform.StartMachine(mach, mach.journal); err != nil {
return nil, err
}

m = nil
bc.AddMach(mach)

return mach, nil
}

func (bc *cluster) vmname() string {
b := make([]byte, 5)
rand.Read(b)
return fmt.Sprintf("%s-%x", bc.Name()[0:13], b)
}

func (bc *cluster) Destroy() {
bc.BaseCluster.Destroy()
bc.flight.DelCluster(bc)
}
69 changes: 69 additions & 0 deletions platform/machine/hetzner/flight.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,69 @@
// Copyright The Mantle Authors.
// SPDX-License-Identifier: Apache-2.0

package hetzner

import (
"fmt"

"github.com/coreos/pkg/capnslog"
ctplatform "github.com/flatcar/container-linux-config-transpiler/config/platform"

"github.com/flatcar/mantle/platform"
"github.com/flatcar/mantle/platform/api/hetzner"
)

const (
Platform platform.Name = "hetzner"
)

var (
plog = capnslog.NewPackageLogger("github.com/flatcar/mantle", "platform/machine/hetzner")
)

type flight struct {
*platform.BaseFlight
api *hetzner.API
}

func NewFlight(opts *hetzner.Options) (platform.Flight, error) {
api, err := hetzner.New(opts)
if err != nil {
return nil, fmt.Errorf("creating hetzner API client: %w", err)
}

// TODO: Rework the Base Flight to remove the CT dependency.
base, err := platform.NewBaseFlight(opts.Options, Platform, ctplatform.Custom)
if err != nil {
return nil, fmt.Errorf("creating base flight: %w", err)
}

bf := &flight{
BaseFlight: base,
api: api,
}

return bf, nil
}

// NewCluster creates an instance of a Cluster suitable for spawning
// instances on the Hetzner platform.
func (bf *flight) NewCluster(rconf *platform.RuntimeConfig) (platform.Cluster, error) {
bc, err := platform.NewBaseCluster(bf.BaseFlight, rconf)
if err != nil {
return nil, fmt.Errorf("creating hetzner base cluster: %w", err)
}

c := &cluster{
BaseCluster: bc,
flight: bf,
}

bf.AddCluster(c)

return c, nil
}

func (bf *flight) Destroy() {
bf.BaseFlight.Destroy()
}
96 changes: 96 additions & 0 deletions platform/machine/hetzner/machine.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,96 @@
// Copyright The Mantle Authors.
// SPDX-License-Identifier: Apache-2.0

package hetzner

import (
"context"
"strconv"

"golang.org/x/crypto/ssh"

"github.com/flatcar/mantle/platform"
"github.com/flatcar/mantle/platform/api/hetzner"
)

type machine struct {
cluster *cluster
mach *hetzner.Server
dir string
journal *platform.Journal
console string
}

// ID returns the ID of the machine.
func (bm *machine) ID() string {
return strconv.FormatInt(bm.mach.Server.ID, 10)
}

// IP returns the IP of the machine.
func (bm *machine) IP() string {
return bm.mach.Server.PublicNet.IPv4.IP.String()
}

// IP returns the private IP of the machine.
func (bm *machine) PrivateIP() string {
if len(bm.mach.Server.PrivateNet) > 0 {
return bm.mach.Server.PrivateNet[0].IP.String()
}

return ""
}

// RuntimeConf returns the runtime configuration of the cluster.
func (bm *machine) RuntimeConf() *platform.RuntimeConfig {
return bm.cluster.RuntimeConf()
}

func (bm *machine) SSHClient() (*ssh.Client, error) {
return bm.cluster.SSHClient(bm.IP())
}

func (bm *machine) PasswordSSHClient(user string, password string) (*ssh.Client, error) {
return bm.cluster.PasswordSSHClient(bm.IP(), user, password)
}

func (bm *machine) SSH(cmd string) ([]byte, []byte, error) {
return bm.cluster.SSH(bm, cmd)
}

func (bm *machine) Reboot() error {
return platform.RebootMachine(bm, bm.journal)
}

func (bm *machine) Destroy() {
// TODO: Add "saveConsole" logic here when Hetzner API will support fetching the console output.

if err := bm.cluster.flight.api.DeleteServer(context.TODO(), bm.ID()); err != nil {
plog.Errorf("deleting server %v: %v", bm.ID(), err)
}

if bm.journal != nil {
bm.journal.Destroy()
}

bm.cluster.DelMach(bm)
}

func (bm *machine) ConsoleOutput() string {
return bm.console
}

func (bm *machine) JournalOutput() string {
if bm.journal == nil {
return ""
}

data, err := bm.journal.Read()
if err != nil {
plog.Errorf("Reading journal for instance %v: %v", bm.ID(), err)
}
return string(data)
}

func (bm *machine) Board() string {
return bm.cluster.flight.Options().Board
}

0 comments on commit e27a6b3

Please sign in to comment.