123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258 |
- /*
- Copyright 2016 The Rook Authors. All rights reserved.
- Licensed under the Apache License, Version 2.0 (the "License");
- you may not use this file except in compliance with the License.
- You may obtain a copy of the License at
- http://www.apache.org/licenses/LICENSE-2.0
- Unless required by applicable law or agreed to in writing, software
- distributed under the License is distributed on an "AS IS" BASIS,
- WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- See the License for the specific language governing permissions and
- limitations under the License.
- */
- package clusterd
- import (
- "errors"
- "fmt"
- "os"
- "path"
- "regexp"
- "strconv"
- "github.com/coreos/pkg/capnslog"
- "github.com/rook/rook/pkg/util/exec"
- "github.com/rook/rook/pkg/util/sys"
- )
- var (
- logger = capnslog.NewPackageLogger("github.com/rook/rook", "inventory")
- isRBD = regexp.MustCompile("^rbd[0-9]+p?[0-9]{0,}$")
- listAllDevices = "all"
- // We need to allow dm- devices (Kubernetes mountpoints) because dm- devices name are used as names for meta device
- allowDeviceNamePattern = regexp.MustCompile("dm-")
- )
- func supportedDeviceType(device string) bool {
- return device == sys.DiskType ||
- device == sys.SSDType ||
- device == sys.CryptType ||
- device == sys.LVMType ||
- device == sys.MultiPath ||
- device == sys.PartType ||
- device == sys.LinearType ||
- (getAllowLoopDevices() && device == sys.LoopType)
- }
- // GetDeviceEmpty check whether a device is completely empty
- func GetDeviceEmpty(device *sys.LocalDisk) bool {
- return supportedDeviceType(device.Type) && len(device.Partitions) == 0 && device.Filesystem == ""
- }
- func ignoreDevice(d string) bool {
- return isRBD.MatchString(d)
- }
- func DiscoverDevicesWithFilter(executor exec.Executor, deviceFilter, metaDevice string) ([]*sys.LocalDisk, error) {
- var disks []*sys.LocalDisk
- devices, err := sys.ListDevices(executor)
- if err != nil {
- return nil, err
- }
- for _, d := range devices {
- // Ignore RBD device
- if ignoreDevice(d) {
- // skip device
- logger.Warningf("skipping rbd device %q", d)
- continue
- }
- if deviceFilter != "" && !deviceMatchWithFilter(d, deviceFilter, metaDevice) {
- logger.Warningf("device %q skipped due to filter %q", d, deviceFilter)
- continue
- }
- // Populate device information coming from lsblk
- disk, err := PopulateDeviceInfo(d, executor)
- if err != nil {
- logger.Warningf("skipping device %q. %v", d, err)
- continue
- }
- // Populate udev information coming from udev
- disk, err = PopulateDeviceUdevInfo(d, executor, disk)
- if err != nil {
- // go on without udev info
- // not ideal for our filesystem check later but we can't really fail either...
- logger.Warningf("failed to get udev info for device %q. %v", d, err)
- }
- // Test if device has child, if so we skip it and only consider the partitions
- // which will come in later iterations of the loop
- // We only test if the type is 'disk', this is a property reported by lsblk
- // and means it's a parent block device
- if disk.Type == sys.DiskType {
- deviceChild, err := sys.ListDevicesChild(executor, fmt.Sprintf("/dev/%s", d))
- if err != nil {
- logger.Warningf("failed to detect child devices for device %q, assuming they are none. %v", d, err)
- }
- // lsblk will output at least 2 lines if they are partitions, one for the parent
- // and N for the child
- if len(deviceChild) > 1 {
- logger.Infof("skipping device %q because it has child, considering the child instead.", d)
- continue
- }
- }
- disks = append(disks, disk)
- }
- logger.Debug("discovered disks are:")
- for _, disk := range disks {
- logger.Debugf("%+v", disk)
- }
- return disks, nil
- }
- func deviceMatchWithFilter(device string, filter, metaDevice string) bool {
- if filter == listAllDevices || device == metaDevice {
- return true
- }
- if allowDeviceNamePattern.MatchString(device) {
- return true
- }
- pattern, err := regexp.Compile(filter)
- if err != nil {
- return false
- }
- if !pattern.MatchString(device) {
- return false
- }
- return true
- }
- // DiscoverDevices returns all the details of devices available on the local node
- func DiscoverDevices(executor exec.Executor) ([]*sys.LocalDisk, error) {
- disks, err := DiscoverDevicesWithFilter(executor, "", "")
- if err != nil {
- return nil, err
- }
- return disks, nil
- }
- // PopulateDeviceInfo returns the information of the specified block device
- func PopulateDeviceInfo(d string, executor exec.Executor) (*sys.LocalDisk, error) {
- diskProps, err := sys.GetDeviceProperties(d, executor)
- if err != nil {
- return nil, err
- }
- diskType, ok := diskProps["TYPE"]
- if !ok {
- return nil, errors.New("diskType is empty")
- }
- if !supportedDeviceType(diskType) {
- return nil, fmt.Errorf("unsupported diskType %+s", diskType)
- }
- // get the UUID for disks
- var diskUUID string
- if diskType == sys.DiskType {
- uuid, err := sys.GetDiskUUID(d, executor)
- if err != nil {
- logger.Warning(err)
- } else {
- diskUUID = uuid
- }
- }
- disk := &sys.LocalDisk{Name: d, UUID: diskUUID}
- if val, ok := diskProps["TYPE"]; ok {
- disk.Type = val
- }
- if val, ok := diskProps["SIZE"]; ok {
- if size, err := strconv.ParseUint(val, 10, 64); err == nil {
- disk.Size = size
- }
- }
- if val, ok := diskProps["ROTA"]; ok {
- if rotates, err := strconv.ParseBool(val); err == nil {
- disk.Rotational = rotates
- }
- }
- if val, ok := diskProps["RO"]; ok {
- if ro, err := strconv.ParseBool(val); err == nil {
- disk.Readonly = ro
- }
- }
- if val, ok := diskProps["PKNAME"]; ok {
- if val != "" {
- disk.Parent = path.Base(val)
- }
- }
- if val, ok := diskProps["NAME"]; ok {
- disk.RealPath = val
- }
- if val, ok := diskProps["KNAME"]; ok {
- disk.KernelName = path.Base(val)
- }
- if val, ok := diskProps["FSTYPE"]; ok && val != "" {
- disk.Filesystem = path.Base(val)
- }
- if val, ok := diskProps["MOUNTPOINT"]; ok && val != "" {
- disk.Mountpoint = path.Base(val)
- }
- return disk, nil
- }
- // PopulateDeviceUdevInfo fills the udev info into the block device information
- func PopulateDeviceUdevInfo(d string, executor exec.Executor, disk *sys.LocalDisk) (*sys.LocalDisk, error) {
- udevInfo, err := sys.GetUdevInfo(d, executor)
- if err != nil {
- return disk, err
- }
- // parse udev info output
- if val, ok := udevInfo["DEVLINKS"]; ok {
- disk.DevLinks = val
- }
- if val, ok := udevInfo["ID_FS_TYPE"]; ok {
- disk.Filesystem = val
- }
- if val, ok := udevInfo["ID_SERIAL"]; ok {
- disk.Serial = val
- }
- if val, ok := udevInfo["ID_VENDOR"]; ok {
- disk.Vendor = val
- }
- if val, ok := udevInfo["ID_MODEL"]; ok {
- disk.Model = val
- }
- if val, ok := udevInfo["ID_WWN_WITH_EXTENSION"]; ok {
- disk.WWNVendorExtension = val
- }
- if val, ok := udevInfo["ID_WWN"]; ok {
- disk.WWN = val
- }
- return disk, nil
- }
- func getAllowLoopDevices() bool {
- return os.Getenv("CEPH_VOLUME_ALLOW_LOOP_DEVICES") == "true"
- }
|