device_tool.go 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168
  1. //go:build !windows
  2. // +build !windows
  3. package main
  4. import (
  5. "flag"
  6. "fmt"
  7. "os"
  8. "path"
  9. "sort"
  10. "strconv"
  11. "strings"
  12. "github.com/docker/docker/daemon/graphdriver/devmapper"
  13. "github.com/docker/docker/pkg/devicemapper"
  14. "github.com/sirupsen/logrus"
  15. )
  16. func usage() {
  17. fmt.Fprintf(os.Stderr, "Usage: %s <flags> [status] | [list] | [device id] | [resize new-pool-size] | [snap new-id base-id] | [remove id] | [mount id mountpoint]\n", os.Args[0])
  18. flag.PrintDefaults()
  19. os.Exit(1)
  20. }
  21. func byteSizeFromString(arg string) (int64, error) {
  22. digits := ""
  23. rest := ""
  24. last := strings.LastIndexAny(arg, "0123456789")
  25. if last >= 0 {
  26. digits = arg[:last+1]
  27. rest = arg[last+1:]
  28. }
  29. val, err := strconv.ParseInt(digits, 10, 64)
  30. if err != nil {
  31. return val, err
  32. }
  33. rest = strings.ToLower(strings.TrimSpace(rest))
  34. var multiplier int64
  35. switch rest {
  36. case "":
  37. multiplier = 1
  38. case "k", "kb":
  39. multiplier = 1024
  40. case "m", "mb":
  41. multiplier = 1024 * 1024
  42. case "g", "gb":
  43. multiplier = 1024 * 1024 * 1024
  44. case "t", "tb":
  45. multiplier = 1024 * 1024 * 1024 * 1024
  46. default:
  47. return 0, fmt.Errorf("Unknown size unit: %s", rest)
  48. }
  49. return val * multiplier, nil
  50. }
  51. func main() {
  52. root := flag.String("r", "/var/lib/docker", "Docker root dir")
  53. flDebug := flag.Bool("D", false, "Debug mode")
  54. flag.Parse()
  55. if *flDebug {
  56. os.Setenv("DEBUG", "1")
  57. logrus.SetLevel(logrus.DebugLevel)
  58. }
  59. if flag.NArg() < 1 {
  60. usage()
  61. }
  62. args := flag.Args()
  63. home := path.Join(*root, "devicemapper")
  64. devices, err := devmapper.NewDeviceSet(home, false, nil, nil, nil)
  65. if err != nil {
  66. fmt.Println("Can't initialize device mapper: ", err)
  67. os.Exit(1)
  68. }
  69. switch args[0] {
  70. case "status":
  71. status := devices.Status()
  72. fmt.Printf("Pool name: %s\n", status.PoolName)
  73. fmt.Printf("Data Loopback file: %s\n", status.DataLoopback)
  74. fmt.Printf("Metadata Loopback file: %s\n", status.MetadataLoopback)
  75. fmt.Printf("Sector size: %d\n", status.SectorSize)
  76. fmt.Printf("Data use: %d of %d (%.1f %%)\n", status.Data.Used, status.Data.Total, 100.0*float64(status.Data.Used)/float64(status.Data.Total))
  77. fmt.Printf("Metadata use: %d of %d (%.1f %%)\n", status.Metadata.Used, status.Metadata.Total, 100.0*float64(status.Metadata.Used)/float64(status.Metadata.Total))
  78. case "list":
  79. ids := devices.List()
  80. sort.Strings(ids)
  81. for _, id := range ids {
  82. fmt.Println(id)
  83. }
  84. case "device":
  85. if flag.NArg() < 2 {
  86. usage()
  87. }
  88. status, err := devices.GetDeviceStatus(args[1])
  89. if err != nil {
  90. fmt.Println("Can't get device info: ", err)
  91. os.Exit(1)
  92. }
  93. fmt.Printf("Id: %d\n", status.DeviceID)
  94. fmt.Printf("Size: %d\n", status.Size)
  95. fmt.Printf("Transaction Id: %d\n", status.TransactionID)
  96. fmt.Printf("Size in Sectors: %d\n", status.SizeInSectors)
  97. fmt.Printf("Mapped Sectors: %d\n", status.MappedSectors)
  98. fmt.Printf("Highest Mapped Sector: %d\n", status.HighestMappedSector)
  99. case "resize":
  100. if flag.NArg() < 2 {
  101. usage()
  102. }
  103. size, err := byteSizeFromString(args[1])
  104. if err != nil {
  105. fmt.Println("Invalid size: ", err)
  106. os.Exit(1)
  107. }
  108. err = devices.ResizePool(size)
  109. if err != nil {
  110. fmt.Println("Error resizing pool: ", err)
  111. os.Exit(1)
  112. }
  113. case "snap":
  114. if flag.NArg() < 3 {
  115. usage()
  116. }
  117. err := devices.AddDevice(args[1], args[2], nil)
  118. if err != nil {
  119. fmt.Println("Can't create snap device: ", err)
  120. os.Exit(1)
  121. }
  122. case "remove":
  123. if flag.NArg() < 2 {
  124. usage()
  125. }
  126. err := devicemapper.RemoveDevice(args[1])
  127. if err != nil {
  128. fmt.Println("Can't remove device: ", err)
  129. os.Exit(1)
  130. }
  131. case "mount":
  132. if flag.NArg() < 3 {
  133. usage()
  134. }
  135. err := devices.MountDevice(args[1], args[2], "")
  136. if err != nil {
  137. fmt.Println("Can't mount device: ", err)
  138. os.Exit(1)
  139. }
  140. default:
  141. fmt.Printf("Unknown command %s\n", args[0])
  142. usage()
  143. os.Exit(1)
  144. }
  145. }