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.

93 lines
2.8 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 weed_server
  2. import (
  3. "fmt"
  4. "os"
  5. "time"
  6. "github.com/chrislusf/seaweedfs/weed/pb/volume_server_pb"
  7. "github.com/chrislusf/seaweedfs/weed/storage/backend"
  8. "github.com/chrislusf/seaweedfs/weed/storage/needle"
  9. )
  10. // VolumeTierCopyDatToRemote copy dat file to a remote tier
  11. func (vs *VolumeServer) VolumeTierCopyDatToRemote(req *volume_server_pb.VolumeTierCopyDatToRemoteRequest, stream volume_server_pb.VolumeServer_VolumeTierCopyDatToRemoteServer) error {
  12. // find existing volume
  13. v := vs.store.GetVolume(needle.VolumeId(req.VolumeId))
  14. if v == nil {
  15. return fmt.Errorf("volume %d not found", req.VolumeId)
  16. }
  17. // verify the collection
  18. if v.Collection != req.Collection {
  19. return fmt.Errorf("existing collection:%v unexpected input: %v", v.Collection, req.Collection)
  20. }
  21. // locate the disk file
  22. diskFile, ok := v.DataBackend.(*backend.DiskFile)
  23. if !ok {
  24. return fmt.Errorf("volume %d is not on local disk", req.VolumeId)
  25. }
  26. // check valid storage backend type
  27. backendStorage, found := backend.BackendStorages[req.DestinationBackendName]
  28. if !found {
  29. var keys []string
  30. for key := range backend.BackendStorages {
  31. keys = append(keys, key)
  32. }
  33. return fmt.Errorf("destination %s not found, suppported: %v", req.DestinationBackendName, keys)
  34. }
  35. // check whether the existing backend storage is the same as requested
  36. // if same, skip
  37. backendType, backendId := backend.BackendNameToTypeId(req.DestinationBackendName)
  38. for _, remoteFile := range v.GetVolumeTierInfo().GetFiles() {
  39. if remoteFile.BackendType == backendType && remoteFile.BackendId == backendId {
  40. return fmt.Errorf("destination %s already exists", req.DestinationBackendName)
  41. }
  42. }
  43. startTime := time.Now()
  44. fn := func(progressed int64, percentage float32) error {
  45. now := time.Now()
  46. if now.Sub(startTime) < time.Second {
  47. return nil
  48. }
  49. startTime = now
  50. return stream.Send(&volume_server_pb.VolumeTierCopyDatToRemoteResponse{
  51. Processed: progressed,
  52. ProcessedPercentage: percentage,
  53. })
  54. }
  55. // copy the data file
  56. key, size, err := backendStorage.CopyFile(diskFile.File, fn)
  57. if err != nil {
  58. return fmt.Errorf("backend %s copy file %s: %v", req.DestinationBackendName, diskFile.Name(), err)
  59. }
  60. // save the remote file to volume tier info
  61. v.GetVolumeTierInfo().Files = append(v.GetVolumeTierInfo().GetFiles(), &volume_server_pb.RemoteFile{
  62. BackendType: backendType,
  63. BackendId: backendId,
  64. Key: key,
  65. Offset: 0,
  66. FileSize: uint64(size),
  67. ModifiedTime: uint64(time.Now().Unix()),
  68. })
  69. if err := v.SaveVolumeTierInfo(); err != nil {
  70. return fmt.Errorf("volume %d fail to save remote file info: %v", v.Id, err)
  71. }
  72. if err := v.LoadRemoteFile(); err != nil {
  73. return fmt.Errorf("volume %d fail to load remote file: %v", v.Id, err)
  74. }
  75. if !req.KeepLocalDatFile {
  76. os.Remove(v.FileName() + ".dat")
  77. }
  78. return nil
  79. }