retrieve.go 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174
  1. // Copyright (c) Tailscale Inc & AUTHORS
  2. // SPDX-License-Identifier: BSD-3-Clause
  3. package taildrop
  4. import (
  5. "context"
  6. "errors"
  7. "io"
  8. "io/fs"
  9. "os"
  10. "runtime"
  11. "sort"
  12. "time"
  13. "tailscale.com/client/tailscale/apitype"
  14. "tailscale.com/util/backoff"
  15. "tailscale.com/util/set"
  16. )
  17. // HasFilesWaiting reports whether any files are buffered in [Handler.Dir].
  18. // This always returns false when [Handler.DirectFileMode] is false.
  19. func (m *manager) HasFilesWaiting() bool {
  20. if m == nil || m.opts.fileOps == nil || m.opts.DirectFileMode {
  21. return false
  22. }
  23. // Optimization: this is usually empty, so avoid opening
  24. // the directory and checking. We can't cache the actual
  25. // has-files-or-not values as the macOS/iOS client might
  26. // in the future use+delete the files directly. So only
  27. // keep this negative cache.
  28. total := m.totalReceived.Load()
  29. if total == m.emptySince.Load() {
  30. return false
  31. }
  32. files, err := m.opts.fileOps.ListFiles()
  33. if err != nil {
  34. return false
  35. }
  36. // Build a set of filenames present in Dir
  37. fileSet := set.Of(files...)
  38. for _, filename := range files {
  39. if isPartialOrDeleted(filename) {
  40. continue
  41. }
  42. if fileSet.Contains(filename + deletedSuffix) {
  43. continue // already handled
  44. }
  45. // Found at least one downloadable file
  46. return true
  47. }
  48. // No waiting files → update negative‑result cache
  49. m.emptySince.Store(total)
  50. return false
  51. }
  52. // WaitingFiles returns the list of files that have been sent by a
  53. // peer that are waiting in [Handler.Dir].
  54. // This always returns nil when [Handler.DirectFileMode] is false.
  55. func (m *manager) WaitingFiles() ([]apitype.WaitingFile, error) {
  56. if m == nil || m.opts.fileOps == nil {
  57. return nil, ErrNoTaildrop
  58. }
  59. if m.opts.DirectFileMode {
  60. return nil, nil
  61. }
  62. names, err := m.opts.fileOps.ListFiles()
  63. if err != nil {
  64. return nil, redactError(err)
  65. }
  66. var ret []apitype.WaitingFile
  67. for _, name := range names {
  68. if isPartialOrDeleted(name) {
  69. continue
  70. }
  71. // A corresponding .deleted marker means the file was already handled.
  72. if _, err := m.opts.fileOps.Stat(name + deletedSuffix); err == nil {
  73. continue
  74. }
  75. fi, err := m.opts.fileOps.Stat(name)
  76. if err != nil {
  77. continue
  78. }
  79. ret = append(ret, apitype.WaitingFile{
  80. Name: name,
  81. Size: fi.Size(),
  82. })
  83. }
  84. sort.Slice(ret, func(i, j int) bool { return ret[i].Name < ret[j].Name })
  85. return ret, nil
  86. }
  87. // DeleteFile deletes a file of the given baseName from [Handler.Dir].
  88. // This method is only allowed when [Handler.DirectFileMode] is false.
  89. func (m *manager) DeleteFile(baseName string) error {
  90. if m == nil || m.opts.fileOps == nil {
  91. return ErrNoTaildrop
  92. }
  93. if m.opts.DirectFileMode {
  94. return errors.New("deletes not allowed in direct mode")
  95. }
  96. var bo *backoff.Backoff
  97. logf := m.opts.Logf
  98. t0 := m.opts.Clock.Now()
  99. for {
  100. err := m.opts.fileOps.Remove(baseName)
  101. if err != nil && !os.IsNotExist(err) {
  102. err = redactError(err)
  103. // Put a retry loop around deletes on Windows.
  104. //
  105. // Windows file descriptor closes are effectively asynchronous,
  106. // as a bunch of hooks run on/after close,
  107. // and we can't necessarily delete the file for a while after close,
  108. // as we need to wait for everybody to be done with it.
  109. // On Windows, unlike Unix, a file can't be deleted if it's open anywhere.
  110. // So try a few times but ultimately just leave a "foo.jpg.deleted"
  111. // marker file to note that it's deleted and we clean it up later.
  112. if runtime.GOOS == "windows" {
  113. if bo == nil {
  114. bo = backoff.NewBackoff("delete-retry", logf, 1*time.Second)
  115. }
  116. if m.opts.Clock.Since(t0) < 5*time.Second {
  117. bo.BackOff(context.Background(), err)
  118. continue
  119. }
  120. if err := m.touchFile(baseName + deletedSuffix); err != nil {
  121. logf("peerapi: failed to leave deleted marker: %v", err)
  122. }
  123. m.deleter.Insert(baseName + deletedSuffix)
  124. }
  125. logf("peerapi: failed to DeleteFile: %v", err)
  126. return err
  127. }
  128. return nil
  129. }
  130. }
  131. func (m *manager) touchFile(name string) error {
  132. wc, _, err := m.opts.fileOps.OpenWriter(name /* offset= */, 0, 0666)
  133. if err != nil {
  134. return redactError(err)
  135. }
  136. return wc.Close()
  137. }
  138. // OpenFile opens a file of the given baseName from [Handler.Dir].
  139. // This method is only allowed when [Handler.DirectFileMode] is false.
  140. func (m *manager) OpenFile(baseName string) (rc io.ReadCloser, size int64, err error) {
  141. if m == nil || m.opts.fileOps == nil {
  142. return nil, 0, ErrNoTaildrop
  143. }
  144. if m.opts.DirectFileMode {
  145. return nil, 0, errors.New("opens not allowed in direct mode")
  146. }
  147. if _, err := m.opts.fileOps.Stat(baseName + deletedSuffix); err == nil {
  148. return nil, 0, redactError(&fs.PathError{Op: "open", Path: baseName, Err: fs.ErrNotExist})
  149. }
  150. f, err := m.opts.fileOps.OpenReader(baseName)
  151. if err != nil {
  152. return nil, 0, redactError(err)
  153. }
  154. fi, err := m.opts.fileOps.Stat(baseName)
  155. if err != nil {
  156. f.Close()
  157. return nil, 0, redactError(err)
  158. }
  159. return f, fi.Size(), nil
  160. }