blkid.go 2.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118
  1. package blkid
  2. import (
  3. "bufio"
  4. "errors"
  5. "os/exec"
  6. "regexp"
  7. "strconv"
  8. "strings"
  9. )
  10. type BlockDevice struct {
  11. Device string
  12. UUID string
  13. BlockSize int
  14. Type string
  15. PartUUID string
  16. PartLabel string
  17. }
  18. // GetBlockDevices retrieves block devices using the `blkid` command.
  19. func GetPartitionIdInfo() ([]BlockDevice, error) {
  20. //Check if the current user have superuser privileges
  21. cmd := exec.Command("id", "-u")
  22. userIDOutput, err := cmd.Output()
  23. if err != nil {
  24. return nil, err
  25. }
  26. // Check if the user ID is 0 (root)
  27. // If not, run blkid without sudo
  28. if strings.TrimSpace(string(userIDOutput)) == "0" {
  29. cmd = exec.Command("blkid")
  30. } else {
  31. cmd = exec.Command("sudo", "blkid")
  32. }
  33. output, err := cmd.Output()
  34. if err != nil {
  35. return nil, err
  36. }
  37. scanner := bufio.NewScanner(strings.NewReader(string(output)))
  38. devices := []BlockDevice{}
  39. re := regexp.MustCompile(`(\S+):\s+(.*)`)
  40. for scanner.Scan() {
  41. line := scanner.Text()
  42. matches := re.FindStringSubmatch(line)
  43. if len(matches) != 3 {
  44. continue
  45. }
  46. device := matches[1]
  47. attributes := matches[2]
  48. deviceInfo := BlockDevice{Device: device}
  49. for _, attr := range strings.Split(attributes, " ") {
  50. kv := strings.SplitN(attr, "=", 2)
  51. if len(kv) != 2 {
  52. continue
  53. }
  54. key := kv[0]
  55. value := strings.Trim(kv[1], `"`)
  56. switch key {
  57. case "UUID":
  58. deviceInfo.UUID = value
  59. case "BLOCK_SIZE":
  60. // Convert block size to int if possible
  61. blockSize, err := strconv.Atoi(value)
  62. if err == nil {
  63. deviceInfo.BlockSize = blockSize
  64. } else {
  65. deviceInfo.BlockSize = 0
  66. }
  67. case "TYPE":
  68. deviceInfo.Type = value
  69. case "PARTUUID":
  70. deviceInfo.PartUUID = value
  71. case "PARTLABEL":
  72. deviceInfo.PartLabel = value
  73. }
  74. }
  75. devices = append(devices, deviceInfo)
  76. }
  77. if err := scanner.Err(); err != nil {
  78. return nil, err
  79. }
  80. return devices, nil
  81. }
  82. // GetBlockDeviceIDFromDevicePath retrieves block device information for a given device path.
  83. func GetPartitionIDFromDevicePath(devpath string) (*BlockDevice, error) {
  84. devpath = strings.TrimPrefix(devpath, "/dev/")
  85. if strings.Contains(devpath, "/") {
  86. return nil, errors.New("invalid device path")
  87. }
  88. devpath = "/dev/" + devpath
  89. devices, err := GetPartitionIdInfo()
  90. if err != nil {
  91. return nil, err
  92. }
  93. for _, device := range devices {
  94. if device.Device == devpath {
  95. return &device, nil
  96. }
  97. }
  98. return nil, errors.New("device not found")
  99. }