Skip to content

Commit

Permalink
feat: add ability to find disk by disk properties
Browse files Browse the repository at this point in the history
Fixes: siderolabs#3323

Not exactly matching with udevd generated `by-<id>` symlinks, but should
provide sufficient amount of property selectors to be able to pick
specific disks for any kind of disk: sd card, hdd, ssd, nvme.

Signed-off-by: Artem Chernyshev <artem.0xD2@gmail.com>
  • Loading branch information
Unix4ever committed Mar 23, 2021
1 parent ac87647 commit 5329346
Show file tree
Hide file tree
Showing 23 changed files with 1,028 additions and 92 deletions.
19 changes: 19 additions & 0 deletions api/storage/storage.proto
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,25 @@ message Disk {
string model = 2;
// DeviceName indicates the disk name (e.g. `sda`).
string device_name = 3;
// Name as in `/sys/block/<dev>/device/name`.
string name = 4;
// Serial as in `/sys/block/<dev>/device/serial`.
string serial = 5;
// Modalias as in `/sys/block/<dev>/device/modalias`.
string modalias = 6;
// Uuid as in `/sys/block/<dev>/device/uuid`.
string uuid = 7;
// Wwid as in `/sys/block/<dev>/device/wwid`.
string wwid = 8;
enum DiskType {
UNKNOWN = 0;
SSD = 1;
HDD = 2;
NVME = 3;
SD = 4;
}
// Type is a type of the disk: nvme, ssd, hdd, sd card.
DiskType type = 9;
}

// DisksResponse represents the response of the `Disks` RPC.
Expand Down
5 changes: 3 additions & 2 deletions cmd/talosctl/cmd/mgmt/validate.go
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@ import (

"github.com/talos-systems/talos/internal/app/machined/pkg/runtime"
"github.com/talos-systems/talos/pkg/cli"
"github.com/talos-systems/talos/pkg/machinery/config"
"github.com/talos-systems/talos/pkg/machinery/config/configloader"
)

Expand All @@ -26,7 +27,7 @@ var validateCmd = &cobra.Command{
Long: ``,
Args: cobra.NoArgs,
RunE: func(cmd *cobra.Command, args []string) error {
config, err := configloader.NewFromFile(validateConfigArg)
cfg, err := configloader.NewFromFile(validateConfigArg)
if err != nil {
return err
}
Expand All @@ -35,7 +36,7 @@ var validateCmd = &cobra.Command{
if err != nil {
return err
}
if err := config.Validate(mode); err != nil {
if err := cfg.Validate(mode, config.WithLocal()); err != nil {
return err
}

Expand Down
118 changes: 118 additions & 0 deletions cmd/talosctl/cmd/talos/disks.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,118 @@
// 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 talos

import (
"context"
"crypto/tls"
"fmt"
"os"
"strings"
"text/tabwriter"

humanize "github.com/dustin/go-humanize"
"github.com/spf13/cobra"

"github.com/talos-systems/talos/pkg/cli"
"github.com/talos-systems/talos/pkg/machinery/client"
)

var disksCmd = &cobra.Command{
Use: "disks",
Short: "Get the list of disks from /sys/block on the machine",
Long: ``,
RunE: func(cmd *cobra.Command, args []string) error {
ctx := context.Background()

c, err := client.New(ctx, client.WithTLSConfig(&tls.Config{
InsecureSkipVerify: true,
}), client.WithEndpoints(Nodes...))
if err != nil {
return err
}

response, err := c.Disks(ctx)
if err != nil {
if response == nil {
return fmt.Errorf("error getting disks: %w", err)
}
cli.Warning("%s", err)
}

w := tabwriter.NewWriter(os.Stdout, 0, 0, 3, ' ', 0)
node := ""

labels := strings.Join(
[]string{
"DEV",
"MODEL",
"SERIAL",
"TYPE",
"UUID",
"WWID",
"MODALIAS",
"NAME",
"SIZE",
}, "\t")

getWithPlaceholder := func(in string) string {
if in == "" {
return "-"
}

return in
}

for i, message := range response.Messages {
if message.Metadata != nil && message.Metadata.Hostname != "" {
node = message.Metadata.Hostname
}

if len(message.Disks) == 0 {
continue
}

for j, disk := range message.Disks {

if i == 0 && j == 0 {
if node != "" {
fmt.Fprintln(w, "NODE\t"+labels)
} else {
fmt.Fprintln(w, labels)
}
}

args := []interface{}{}

if node != "" {
args = append(args, node)
}

args = append(args, []interface{}{
getWithPlaceholder(disk.DeviceName),
getWithPlaceholder(disk.Model),
getWithPlaceholder(disk.Serial),
disk.Type.String(),
getWithPlaceholder(disk.Uuid),
getWithPlaceholder(disk.Wwid),
getWithPlaceholder(disk.Modalias),
getWithPlaceholder(disk.Name),
humanize.Bytes(disk.Size),
}...)

pattern := strings.Repeat("%s\t", len(args))
pattern = strings.TrimSpace(pattern) + "\n"

fmt.Fprintf(w, pattern, args...)
}
}

return w.Flush()
},
}

func init() {
addCommand(disksCmd)
}
2 changes: 1 addition & 1 deletion go.mod
Original file line number Diff line number Diff line change
Expand Up @@ -68,7 +68,7 @@ require (
github.com/stretchr/testify v1.7.0
github.com/syndtr/gocapability v0.0.0-20180916011248-d98352740cb2 // can't be bumped, requires new runc
github.com/talos-systems/crypto v0.2.1-0.20210202170911-39584f1b6e54
github.com/talos-systems/go-blockdevice v0.2.1-0.20210305142622-bb3ad73f6983
github.com/talos-systems/go-blockdevice v0.2.1-0.20210322192639-776b37d31de0
github.com/talos-systems/go-cmd v0.0.0-20210216164758-68eb0067e0f0
github.com/talos-systems/go-loadbalancer v0.1.0
github.com/talos-systems/go-procfs v0.0.0-20210108152626-8cbc42d3dc24
Expand Down
6 changes: 4 additions & 2 deletions go.sum
Original file line number Diff line number Diff line change
Expand Up @@ -893,6 +893,8 @@ github.com/russross/blackfriday/v2 v2.0.1/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQD
github.com/ryanuber/columnize v0.0.0-20160712163229-9b3edd62028f/go.mod h1:sm1tb6uqfes/u+d4ooFouqFdy9/2g9QGwK3SQygK0Ts=
github.com/ryanuber/columnize v2.1.2+incompatible h1:C89EOx/XBWwIXl8wm8OPJBd7kPF25UfsK2X7Ph/zCAk=
github.com/ryanuber/columnize v2.1.2+incompatible/go.mod h1:sm1tb6uqfes/u+d4ooFouqFdy9/2g9QGwK3SQygK0Ts=
github.com/ryanuber/go-glob v1.0.0 h1:iQh3xXAumdQ+4Ufa5b25cRpC5TYKlno6hsv6Cb3pkBk=
github.com/ryanuber/go-glob v1.0.0/go.mod h1:807d1WSdnB0XRJzKNil9Om6lcp/3a0v4qIHxIXzX/Yc=
github.com/safchain/ethtool v0.0.0-20190326074333-42ed695e3de8/go.mod h1:Z0q5wiBQGYcxhMZ6gUqHn6pYNLypFAvaL3UvgZLR0U4=
github.com/satori/go.uuid v1.2.0/go.mod h1:dA0hQrYB0VpLJoorglMZABFdXlWrHn1NEOzdhQKdks0=
github.com/satori/go.uuid v1.2.1-0.20181028125025-b2ce2384e17b h1:gQZ0qzfKHQIybLANtM3mBXNUtOfsCFXeTsnBqCsx1KM=
Expand Down Expand Up @@ -974,8 +976,8 @@ github.com/syndtr/gocapability v0.0.0-20180916011248-d98352740cb2 h1:b6uOv7YOFK0
github.com/syndtr/gocapability v0.0.0-20180916011248-d98352740cb2/go.mod h1:hkRG7XYTFWNJGYcbNJQlaLq0fg1yr4J4t/NcTQtrfww=
github.com/talos-systems/crypto v0.2.1-0.20210202170911-39584f1b6e54 h1:2IGs3f0qG7f33Fv37XuYzIMxLARl4dcpwahZXLmdT2A=
github.com/talos-systems/crypto v0.2.1-0.20210202170911-39584f1b6e54/go.mod h1:OXCK52Q0dzm88YRG4VdTBdidkPUtqrCxCyW7bUs4DAw=
github.com/talos-systems/go-blockdevice v0.2.1-0.20210305142622-bb3ad73f6983 h1:tYxXpKNUNvrBKFE4itbU3lJVBws/XUmlYkuEI93kCAE=
github.com/talos-systems/go-blockdevice v0.2.1-0.20210305142622-bb3ad73f6983/go.mod h1:Wt0/JMsa+WysYRDlNo6fu9rdn/re73uRnktFWyVUqjs=
github.com/talos-systems/go-blockdevice v0.2.1-0.20210322192639-776b37d31de0 h1:ZOIwpHUR0i2e4VaCe8nwHZuBRjD25XUv+xImyPWHw50=
github.com/talos-systems/go-blockdevice v0.2.1-0.20210322192639-776b37d31de0/go.mod h1:qnn/zDc09I1DA2BUDDCOSA2D0P8pIDjN8pGiRoRaQig=
github.com/talos-systems/go-cmd v0.0.0-20210216164758-68eb0067e0f0 h1:DI+BjK+fcrLBc70Fi50dZocQcaHosqsuWHrGHKp2NzE=
github.com/talos-systems/go-cmd v0.0.0-20210216164758-68eb0067e0f0/go.mod h1:kf+rZzTEmlDiYQ6ulslvRONnKLQH8x83TowltGMhO+k=
github.com/talos-systems/go-loadbalancer v0.1.0 h1:MQFONvSjoleU8RrKq1O1Z8CyTCJGd4SLqdAHDlR6o9s=
Expand Down
4 changes: 4 additions & 0 deletions hack/docgen/main.go
Original file line number Diff line number Diff line change
Expand Up @@ -319,6 +319,10 @@ func collectFields(s *structType) (fields []*Field) {
log.Fatalf("field %q is missing a documentation", f.Names[0].Name)
}

if strings.Contains(f.Doc.Text(), "docgen:nodoc") {
continue
}

name := f.Names[0].Name

fieldType := formatFieldType(f.Type)
Expand Down
21 changes: 19 additions & 2 deletions hack/release.toml
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ github_repo = "talos-systems/talos"
match_deps = "^github.com/(talos-systems/[a-zA-Z0-9-]+)$"

# previous release
previous = "v0.9.0-beta.0"
previous = "v0.9.0"

pre_release = true

Expand All @@ -24,8 +24,25 @@ preface = """\

[notes.optimize]
title = "Optmizations"
descriptions = """\
description = """\
* Talos `system` services now run without container images on initramfs from the single executable; this change reduces RAM usage, initramfs size and boot time..
"""

[notes.machineconfig]
title = "Install Disk Selector"
description = """\
Install section of the machine config now has `diskSelector` field that allows querying install disk using the list of qualifiers:
```yaml
...
install:
diskSelector:
size: >= 500GB
model: WDC*
...
```
`talosctl disks -n <node> -i` can be used to check allowed disk qualifiers when the node is running in the maintenance mode.
"""

[make_deps]
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -246,8 +246,13 @@ func (s *Server) Rollback(ctx context.Context, in *machine.RollbackRequest) (*ma
}
}()

disk, err := s.Controller.Runtime().Config().Machine().Install().Disk()
if err != nil {
return err
}

grub := &grub.Grub{
BootDisk: s.Controller.Runtime().Config().Machine().Install().Disk(),
BootDisk: disk,
}

_, next, err := grub.Labels()
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -740,10 +740,16 @@ func VerifyInstallation(seq runtime.Sequence, data interface{}) (runtime.TaskExe
var (
current string
next string
disk string
)

disk, err = r.Config().Machine().Install().Disk()
if err != nil {
return err
}

grub := &grub.Grub{
BootDisk: r.Config().Machine().Install().Disk(),
BootDisk: disk,
}

current, next, err = grub.Labels()
Expand Down Expand Up @@ -1630,8 +1636,15 @@ func Install(seq runtime.Sequence, data interface{}) (runtime.TaskExecutionFunc,
installerImage = images.DefaultInstallerImage
}

var disk string

disk, err = r.Config().Machine().Install().Disk()
if err != nil {
return err
}

err = install.RunInstallerContainer(
r.Config().Machine().Install().Disk(),
disk,
r.State().Platform().Name(),
installerImage,
r.Config().Machine().Registries(),
Expand Down
8 changes: 6 additions & 2 deletions internal/app/storaged/server.go
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@ import (
"context"

"github.com/golang/protobuf/ptypes/empty"
"github.com/talos-systems/go-blockdevice/blockdevice/util"
"github.com/talos-systems/go-blockdevice/blockdevice/util/disk"

"github.com/talos-systems/talos/pkg/machinery/api/storage"
)
Expand All @@ -19,7 +19,7 @@ type Server struct{}

// Disks implements storage.StorageService.
func (s *Server) Disks(ctx context.Context, in *empty.Empty) (reply *storage.DisksResponse, err error) {
disks, err := util.GetDisks()
disks, err := disk.List()
if err != nil {
return nil, err
}
Expand All @@ -31,6 +31,10 @@ func (s *Server) Disks(ctx context.Context, in *empty.Empty) (reply *storage.Dis
DeviceName: disk.DeviceName,
Model: disk.Model,
Size: disk.Size,
Name: disk.Name,
Serial: disk.Serial,
Modalias: disk.Modalias,
Type: storage.Disk_DiskType(disk.Type),
}
}

Expand Down
10 changes: 8 additions & 2 deletions internal/integration/api/generate-config.go
Original file line number Diff line number Diff line change
Expand Up @@ -95,6 +95,9 @@ func (suite *GenerateConfigSuite) TestGenerate() {

suite.Require().NoError(err)

disk, err := config.Machine().Install().Disk()
suite.Require().NoError(err)

suite.Require().EqualValues(request.MachineConfig.Type, config.Machine().Type())
suite.Require().EqualValues(request.ConfigVersion, config.Version())
suite.Require().EqualValues(request.ClusterConfig.Name, config.Cluster().Name())
Expand All @@ -103,7 +106,7 @@ func (suite *GenerateConfigSuite) TestGenerate() {
suite.Require().EqualValues(request.ClusterConfig.ClusterNetwork.CniConfig.Name, config.Cluster().Network().CNI().Name())
suite.Require().EqualValues(request.ClusterConfig.ClusterNetwork.CniConfig.Urls, config.Cluster().Network().CNI().URLs())
suite.Require().EqualValues(fmt.Sprintf("%s:v%s", constants.KubeletImage, request.MachineConfig.KubernetesVersion), config.Machine().Kubelet().Image())
suite.Require().EqualValues(request.MachineConfig.InstallConfig.InstallDisk, config.Machine().Install().Disk())
suite.Require().EqualValues(request.MachineConfig.InstallConfig.InstallDisk, disk)
suite.Require().EqualValues(request.MachineConfig.InstallConfig.InstallImage, config.Machine().Install().Image())
suite.Require().EqualValues(request.MachineConfig.NetworkConfig.Hostname, config.Machine().Network().Hostname())
suite.Require().EqualValues(request.MachineConfig.NetworkConfig.Hostname, config.Machine().Network().Hostname())
Expand Down Expand Up @@ -138,13 +141,16 @@ func (suite *GenerateConfigSuite) TestGenerate() {

suite.Require().NoError(err)

disk, err = config.Machine().Install().Disk()
suite.Require().NoError(err)

suite.Require().EqualValues(request.MachineConfig.Type, joinedConfig.Machine().Type())
suite.Require().EqualValues(request.ConfigVersion, joinedConfig.Version())
suite.Require().EqualValues(request.ClusterConfig.Name, joinedConfig.Cluster().Name())
suite.Require().EqualValues(request.ClusterConfig.ControlPlane.Endpoint, joinedConfig.Cluster().Endpoint().String())
suite.Require().EqualValues(request.ClusterConfig.ClusterNetwork.DnsDomain, joinedConfig.Cluster().Network().DNSDomain())
suite.Require().EqualValues(fmt.Sprintf("%s:v%s", constants.KubeletImage, request.MachineConfig.KubernetesVersion), joinedConfig.Machine().Kubelet().Image())
suite.Require().EqualValues(request.MachineConfig.InstallConfig.InstallDisk, joinedConfig.Machine().Install().Disk())
suite.Require().EqualValues(request.MachineConfig.InstallConfig.InstallDisk, disk)
suite.Require().EqualValues(request.MachineConfig.InstallConfig.InstallImage, joinedConfig.Machine().Install().Image())
suite.Require().EqualValues(request.MachineConfig.NetworkConfig.Hostname, joinedConfig.Machine().Network().Hostname())

Expand Down
Loading

0 comments on commit 5329346

Please sign in to comment.