You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

133 lines
2.9 KiB

5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
  1. package shell
  2. import (
  3. "fmt"
  4. "io"
  5. "net/url"
  6. "strconv"
  7. "strings"
  8. "google.golang.org/grpc"
  9. "github.com/chrislusf/seaweedfs/weed/pb"
  10. "github.com/chrislusf/seaweedfs/weed/pb/filer_pb"
  11. "github.com/chrislusf/seaweedfs/weed/util"
  12. "github.com/chrislusf/seaweedfs/weed/wdclient"
  13. "github.com/chrislusf/seaweedfs/weed/wdclient/exclusive_locks"
  14. )
  15. type ShellOptions struct {
  16. Masters *string
  17. GrpcDialOption grpc.DialOption
  18. // shell transient context
  19. FilerHost string
  20. FilerPort int64
  21. Directory string
  22. }
  23. type CommandEnv struct {
  24. env map[string]string
  25. MasterClient *wdclient.MasterClient
  26. option ShellOptions
  27. locker *exclusive_locks.ExclusiveLocker
  28. }
  29. type command interface {
  30. Name() string
  31. Help() string
  32. Do([]string, *CommandEnv, io.Writer) error
  33. }
  34. var (
  35. Commands = []command{}
  36. )
  37. func NewCommandEnv(options ShellOptions) *CommandEnv {
  38. ce := &CommandEnv{
  39. env: make(map[string]string),
  40. MasterClient: wdclient.NewMasterClient(options.GrpcDialOption, pb.AdminShellClient, "", 0, "", strings.Split(*options.Masters, ",")),
  41. option: options,
  42. }
  43. ce.locker = exclusive_locks.NewExclusiveLocker(ce.MasterClient)
  44. return ce
  45. }
  46. func (ce *CommandEnv) parseUrl(input string) (path string, err error) {
  47. if strings.HasPrefix(input, "http") {
  48. err = fmt.Errorf("http://<filer>:<port> prefix is not supported any more")
  49. return
  50. }
  51. if !strings.HasPrefix(input, "/") {
  52. input = util.Join(ce.option.Directory, input)
  53. }
  54. return input, err
  55. }
  56. func (ce *CommandEnv) isDirectory(path string) bool {
  57. return ce.checkDirectory(path) == nil
  58. }
  59. func (ce *CommandEnv) confirmIsLocked() error {
  60. if ce.locker.IsLocking() {
  61. return nil
  62. }
  63. return fmt.Errorf("need to lock to continue")
  64. }
  65. func (ce *CommandEnv) checkDirectory(path string) error {
  66. dir, name := util.FullPath(path).DirAndName()
  67. exists, err := filer_pb.Exists(ce, dir, name, true)
  68. if !exists {
  69. return fmt.Errorf("%s is not a directory", path)
  70. }
  71. return err
  72. }
  73. var _ = filer_pb.FilerClient(&CommandEnv{})
  74. func (ce *CommandEnv) WithFilerClient(fn func(filer_pb.SeaweedFilerClient) error) error {
  75. filerGrpcAddress := fmt.Sprintf("%s:%d", ce.option.FilerHost, ce.option.FilerPort+10000)
  76. return pb.WithGrpcFilerClient(filerGrpcAddress, ce.option.GrpcDialOption, fn)
  77. }
  78. func parseFilerUrl(entryPath string) (filerServer string, filerPort int64, path string, err error) {
  79. if strings.HasPrefix(entryPath, "http") {
  80. var u *url.URL
  81. u, err = url.Parse(entryPath)
  82. if err != nil {
  83. return
  84. }
  85. filerServer = u.Hostname()
  86. portString := u.Port()
  87. if portString != "" {
  88. filerPort, err = strconv.ParseInt(portString, 10, 32)
  89. }
  90. path = u.Path
  91. } else {
  92. err = fmt.Errorf("path should have full url /path/to/dirOrFile : %s", entryPath)
  93. }
  94. return
  95. }
  96. func findInputDirectory(args []string) (input string) {
  97. input = "."
  98. if len(args) > 0 {
  99. input = args[len(args)-1]
  100. if strings.HasPrefix(input, "-") {
  101. input = "."
  102. }
  103. }
  104. return input
  105. }