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.

305 lines
9.6 KiB

more solid weed mount (#4089) * compare chunks by timestamp * fix slab clearing error * fix test compilation * move oldest chunk to sealed, instead of by fullness * lock on fh.entryViewCache * remove verbose logs * revert slat clearing * less logs * less logs * track write and read by timestamp * remove useless logic * add entry lock on file handle release * use mem chunk only, swap file chunk has problems * comment out code that maybe used later * add debug mode to compare data read and write * more efficient readResolvedChunks with linked list * small optimization * fix test compilation * minor fix on writer * add SeparateGarbageChunks * group chunks into sections * turn off debug mode * fix tests * fix tests * tmp enable swap file chunk * Revert "tmp enable swap file chunk" This reverts commit 985137ec472924e4815f258189f6ca9f2168a0a7. * simple refactoring * simple refactoring * do not re-use swap file chunk. Sealed chunks should not be re-used. * comment out debugging facilities * either mem chunk or swap file chunk is fine now * remove orderedMutex as *semaphore.Weighted not found impactful * optimize size calculation for changing large files * optimize performance to avoid going through the long list of chunks * still problems with swap file chunk * rename * tiny optimization * swap file chunk save only successfully read data * fix * enable both mem and swap file chunk * resolve chunks with range * rename * fix chunk interval list * also change file handle chunk group when adding chunks * pick in-active chunk with time-decayed counter * fix compilation * avoid nil with empty fh.entry * refactoring * rename * rename * refactor visible intervals to *list.List * refactor chunkViews to *list.List * add IntervalList for generic interval list * change visible interval to use IntervalList in generics * cahnge chunkViews to *IntervalList[*ChunkView] * use NewFileChunkSection to create * rename variables * refactor * fix renaming leftover * renaming * renaming * add insert interval * interval list adds lock * incrementally add chunks to readers Fixes: 1. set start and stop offset for the value object 2. clone the value object 3. use pointer instead of copy-by-value when passing to interval.Value 4. use insert interval since adding chunk could be out of order * fix tests compilation * fix tests compilation
2 years ago
  1. package shell
  2. import (
  3. "bytes"
  4. "context"
  5. "flag"
  6. "fmt"
  7. "github.com/seaweedfs/seaweedfs/weed/filer"
  8. "github.com/seaweedfs/seaweedfs/weed/operation"
  9. "github.com/seaweedfs/seaweedfs/weed/pb"
  10. "github.com/seaweedfs/seaweedfs/weed/pb/filer_pb"
  11. "github.com/seaweedfs/seaweedfs/weed/pb/master_pb"
  12. "github.com/seaweedfs/seaweedfs/weed/pb/volume_server_pb"
  13. "github.com/seaweedfs/seaweedfs/weed/storage"
  14. "github.com/seaweedfs/seaweedfs/weed/util"
  15. "go.uber.org/atomic"
  16. "golang.org/x/exp/slices"
  17. "io"
  18. "math"
  19. "strings"
  20. "sync"
  21. "time"
  22. )
  23. func init() {
  24. Commands = append(Commands, &commandFsVerify{})
  25. }
  26. type commandFsVerify struct {
  27. env *CommandEnv
  28. volumeServers []pb.ServerAddress
  29. volumeIds map[uint32][]pb.ServerAddress
  30. verbose *bool
  31. metadataFromLog *bool
  32. concurrency *int
  33. modifyTimeAgoAtSec int64
  34. writer io.Writer
  35. waitChan map[string]chan struct{}
  36. waitChanLock sync.RWMutex
  37. }
  38. func (c *commandFsVerify) Name() string {
  39. return "fs.verify"
  40. }
  41. func (c *commandFsVerify) Help() string {
  42. return `recursively verify all files under a directory
  43. fs.verify [-v] [-modifyTimeAgo 1h] /buckets/dir
  44. `
  45. }
  46. func (c *commandFsVerify) Do(args []string, commandEnv *CommandEnv, writer io.Writer) (err error) {
  47. c.env = commandEnv
  48. c.writer = writer
  49. fsVerifyCommand := flag.NewFlagSet(c.Name(), flag.ContinueOnError)
  50. c.verbose = fsVerifyCommand.Bool("v", false, "print out each processed files")
  51. modifyTimeAgo := fsVerifyCommand.Duration("modifyTimeAgo", 0, "only include files after this modify time to verify")
  52. c.concurrency = fsVerifyCommand.Int("concurrency", 0, "number of parallel verification per volume server")
  53. c.metadataFromLog = fsVerifyCommand.Bool("metadataFromLog", false, "Using filer log to get metadata")
  54. if err = fsVerifyCommand.Parse(args); err != nil {
  55. return err
  56. }
  57. path, parseErr := commandEnv.parseUrl(findInputDirectory(fsVerifyCommand.Args()))
  58. if parseErr != nil {
  59. return parseErr
  60. }
  61. c.modifyTimeAgoAtSec = int64(modifyTimeAgo.Seconds())
  62. c.volumeIds = make(map[uint32][]pb.ServerAddress)
  63. c.waitChan = make(map[string]chan struct{})
  64. c.volumeServers = []pb.ServerAddress{}
  65. defer func() {
  66. c.modifyTimeAgoAtSec = 0
  67. c.volumeIds = nil
  68. c.waitChan = nil
  69. c.volumeServers = nil
  70. }()
  71. if err := c.collectVolumeIds(); err != nil {
  72. return parseErr
  73. }
  74. if *c.concurrency > 0 {
  75. for _, volumeServer := range c.volumeServers {
  76. volumeServerStr := string(volumeServer)
  77. c.waitChan[volumeServerStr] = make(chan struct{}, *c.concurrency)
  78. defer close(c.waitChan[volumeServerStr])
  79. }
  80. }
  81. var fCount, eCount uint64
  82. if *c.metadataFromLog {
  83. var wg sync.WaitGroup
  84. fCount, eCount, err = c.verifyProcessMetadata(path, &wg)
  85. wg.Wait()
  86. if err != nil {
  87. return err
  88. }
  89. } else {
  90. fCount, eCount, err = c.verifyTraverseBfs(path)
  91. }
  92. fmt.Fprintf(writer, "verified %d files, error %d files \n", fCount, eCount)
  93. return err
  94. }
  95. func (c *commandFsVerify) collectVolumeIds() error {
  96. topologyInfo, _, err := collectTopologyInfo(c.env, 0)
  97. if err != nil {
  98. return err
  99. }
  100. eachDataNode(topologyInfo, func(dc string, rack RackId, nodeInfo *master_pb.DataNodeInfo) {
  101. for _, diskInfo := range nodeInfo.DiskInfos {
  102. for _, vi := range diskInfo.VolumeInfos {
  103. volumeServer := pb.NewServerAddressFromDataNode(nodeInfo)
  104. c.volumeIds[vi.Id] = append(c.volumeIds[vi.Id], volumeServer)
  105. if !slices.Contains(c.volumeServers, volumeServer) {
  106. c.volumeServers = append(c.volumeServers, volumeServer)
  107. }
  108. }
  109. }
  110. })
  111. return nil
  112. }
  113. func (c *commandFsVerify) verifyChunk(volumeServer pb.ServerAddress, fileId *filer_pb.FileId) error {
  114. err := operation.WithVolumeServerClient(false, volumeServer, c.env.option.GrpcDialOption,
  115. func(client volume_server_pb.VolumeServerClient) error {
  116. _, err := client.VolumeNeedleStatus(context.Background(),
  117. &volume_server_pb.VolumeNeedleStatusRequest{
  118. VolumeId: fileId.VolumeId,
  119. NeedleId: fileId.FileKey})
  120. return err
  121. },
  122. )
  123. if err != nil && !strings.Contains(err.Error(), storage.ErrorDeleted.Error()) {
  124. return err
  125. }
  126. return nil
  127. }
  128. type ItemEntry struct {
  129. chunks []*filer_pb.FileChunk
  130. path util.FullPath
  131. }
  132. func (c *commandFsVerify) verifyProcessMetadata(path string, wg *sync.WaitGroup) (fileCount uint64, errCount uint64, err error) {
  133. processEventFn := func(resp *filer_pb.SubscribeMetadataResponse) error {
  134. message := resp.EventNotification
  135. if resp.EventNotification.NewEntry == nil {
  136. return nil
  137. }
  138. chunkCount := len(message.NewEntry.Chunks)
  139. if chunkCount == 0 {
  140. return nil
  141. }
  142. entryPath := fmt.Sprintf("%s/%s", message.NewParentPath, message.NewEntry.Name)
  143. errorChunksCount := atomic.NewUint64(0)
  144. if !c.verifyEntry(entryPath, message.NewEntry.Chunks, errorChunksCount, wg) {
  145. if err = c.env.WithFilerClient(false, func(client filer_pb.SeaweedFilerClient) error {
  146. entryResp, errReq := client.LookupDirectoryEntry(context.Background(), &filer_pb.LookupDirectoryEntryRequest{
  147. Directory: message.NewParentPath,
  148. Name: message.NewEntry.Name,
  149. })
  150. if strings.HasSuffix(errReq.Error(), "no entry is found in filer store") {
  151. return nil
  152. } else if errReq != nil {
  153. return errReq
  154. }
  155. if entryResp.Entry.Attributes.Mtime == message.NewEntry.Attributes.Mtime &&
  156. bytes.Equal(entryResp.Entry.Attributes.Md5, message.NewEntry.Attributes.Md5) {
  157. fmt.Fprintf(c.writer, "file: %s needles:%d failed:%d\n", entryPath, chunkCount, errorChunksCount.Load())
  158. errCount++
  159. }
  160. return nil
  161. }); err != nil {
  162. return err
  163. }
  164. return nil
  165. }
  166. if *c.verbose {
  167. fmt.Fprintf(c.writer, "file: %s needles:%d verifed\n", entryPath, chunkCount)
  168. }
  169. fileCount++
  170. return nil
  171. }
  172. metadataFollowOption := &pb.MetadataFollowOption{
  173. ClientName: "shell_verify",
  174. ClientId: util.RandomInt32(),
  175. ClientEpoch: 0,
  176. SelfSignature: 0,
  177. PathPrefix: path,
  178. AdditionalPathPrefixes: nil,
  179. DirectoriesToWatch: nil,
  180. StartTsNs: time.Now().Add(-1 * time.Second * time.Duration(c.modifyTimeAgoAtSec)).UnixNano(),
  181. StopTsNs: time.Now().UnixNano(),
  182. EventErrorType: pb.DontLogError,
  183. }
  184. return fileCount, errCount, pb.FollowMetadata(c.env.option.FilerAddress, c.env.option.GrpcDialOption, metadataFollowOption, processEventFn)
  185. }
  186. func (c *commandFsVerify) verifyEntry(path string, chunks []*filer_pb.FileChunk, errorCount *atomic.Uint64, wg *sync.WaitGroup) bool {
  187. fileMsg := fmt.Sprintf("file:%s", path)
  188. itemIsVerifed := atomic.NewBool(true)
  189. for _, chunk := range chunks {
  190. if volumeIds, ok := c.volumeIds[chunk.Fid.VolumeId]; ok {
  191. for _, volumeServer := range volumeIds {
  192. if *c.concurrency == 0 {
  193. if err := c.verifyChunk(volumeServer, chunk.Fid); err != nil {
  194. if !(*c.metadataFromLog && strings.HasSuffix(err.Error(), "not found")) {
  195. fmt.Fprintf(c.writer, "%s failed verify fileId %s: %+v\n",
  196. fileMsg, chunk.GetFileIdString(), err)
  197. }
  198. if itemIsVerifed.Load() {
  199. itemIsVerifed.Store(false)
  200. errorCount.Add(1)
  201. }
  202. }
  203. continue
  204. }
  205. c.waitChanLock.RLock()
  206. waitChan, ok := c.waitChan[string(volumeServer)]
  207. c.waitChanLock.RUnlock()
  208. if !ok {
  209. fmt.Fprintf(c.writer, "%s failed to get channel for %s fileId: %s\n",
  210. string(volumeServer), fileMsg, chunk.GetFileIdString())
  211. if itemIsVerifed.Load() {
  212. itemIsVerifed.Store(false)
  213. errorCount.Add(1)
  214. }
  215. continue
  216. }
  217. wg.Add(1)
  218. waitChan <- struct{}{}
  219. go func(fChunk *filer_pb.FileChunk, path string, volumeServer pb.ServerAddress, msg string) {
  220. defer wg.Done()
  221. if err := c.verifyChunk(volumeServer, fChunk.Fid); err != nil {
  222. if !(*c.metadataFromLog && strings.HasSuffix(err.Error(), "not found")) {
  223. fmt.Fprintf(c.writer, "%s failed verify fileId %s: %+v\n",
  224. msg, fChunk.GetFileIdString(), err)
  225. }
  226. if itemIsVerifed.Load() {
  227. itemIsVerifed.Store(false)
  228. errorCount.Add(1)
  229. }
  230. }
  231. <-waitChan
  232. }(chunk, path, volumeServer, fileMsg)
  233. }
  234. } else {
  235. if !*c.metadataFromLog {
  236. err := fmt.Errorf("volumeId %d not found", chunk.Fid.VolumeId)
  237. fmt.Fprintf(c.writer, "%s failed verify fileId %s: %+v\n",
  238. fileMsg, chunk.GetFileIdString(), err)
  239. }
  240. if itemIsVerifed.Load() {
  241. itemIsVerifed.Store(false)
  242. errorCount.Add(1)
  243. }
  244. break
  245. }
  246. }
  247. return itemIsVerifed.Load()
  248. }
  249. func (c *commandFsVerify) verifyTraverseBfs(path string) (fileCount uint64, errCount uint64, err error) {
  250. timeNowAtSec := time.Now().Unix()
  251. return fileCount, errCount, doTraverseBfsAndSaving(c.env, c.writer, path, false,
  252. func(entry *filer_pb.FullEntry, outputChan chan interface{}) (err error) {
  253. if c.modifyTimeAgoAtSec > 0 {
  254. if entry.Entry.Attributes != nil && c.modifyTimeAgoAtSec < timeNowAtSec-entry.Entry.Attributes.Mtime {
  255. return nil
  256. }
  257. }
  258. dataChunks, manifestChunks, resolveErr := filer.ResolveChunkManifest(filer.LookupFn(c.env), entry.Entry.GetChunks(), 0, math.MaxInt64)
  259. if resolveErr != nil {
  260. return fmt.Errorf("failed to ResolveChunkManifest: %+v", resolveErr)
  261. }
  262. dataChunks = append(dataChunks, manifestChunks...)
  263. if len(dataChunks) > 0 {
  264. outputChan <- &ItemEntry{
  265. chunks: dataChunks,
  266. path: util.NewFullPath(entry.Dir, entry.Entry.Name),
  267. }
  268. }
  269. return nil
  270. },
  271. func(outputChan chan interface{}) {
  272. var wg sync.WaitGroup
  273. itemErrCount := atomic.NewUint64(0)
  274. for itemEntry := range outputChan {
  275. i := itemEntry.(*ItemEntry)
  276. itemPath := string(i.path)
  277. if c.verifyEntry(itemPath, i.chunks, itemErrCount, &wg) {
  278. if *c.verbose {
  279. fmt.Fprintf(c.writer, "file: %s needles:%d verifed\n", itemPath, len(i.chunks))
  280. }
  281. fileCount++
  282. }
  283. }
  284. wg.Wait()
  285. errCount = itemErrCount.Load()
  286. })
  287. }