mtimefs.go 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217
  1. // Copyright (C) 2016 The Syncthing Authors.
  2. //
  3. // This Source Code Form is subject to the terms of the Mozilla Public
  4. // License, v. 2.0. If a copy of the MPL was not distributed with this file,
  5. // You can obtain one at https://mozilla.org/MPL/2.0/.
  6. package fs
  7. import (
  8. "time"
  9. )
  10. // The database is where we store the virtual mtimes
  11. type database interface {
  12. GetMtime(folder, name string) (ondisk, virtual time.Time)
  13. PutMtime(folder, name string, ondisk, virtual time.Time) error
  14. DeleteMtime(folder, name string) error
  15. }
  16. type mtimeFS struct {
  17. Filesystem
  18. chtimes func(string, time.Time, time.Time) error
  19. db database
  20. folderID string
  21. caseInsensitive bool
  22. }
  23. type MtimeFSOption func(*mtimeFS)
  24. func WithCaseInsensitivity(v bool) MtimeFSOption {
  25. return func(f *mtimeFS) {
  26. f.caseInsensitive = v
  27. }
  28. }
  29. type optionMtime struct {
  30. db database
  31. folderID string
  32. options []MtimeFSOption
  33. }
  34. // NewMtimeOption makes any filesystem provide nanosecond mtime precision,
  35. // regardless of what shenanigans the underlying filesystem gets up to.
  36. func NewMtimeOption(db database, folderID string, options ...MtimeFSOption) Option {
  37. return &optionMtime{
  38. db: db,
  39. folderID: folderID,
  40. options: options,
  41. }
  42. }
  43. func (o *optionMtime) apply(fs Filesystem) Filesystem {
  44. f := &mtimeFS{
  45. Filesystem: fs,
  46. chtimes: fs.Chtimes, // for mocking it out in the tests
  47. db: o.db,
  48. folderID: o.folderID,
  49. }
  50. for _, opt := range o.options {
  51. opt(f)
  52. }
  53. return f
  54. }
  55. func (*optionMtime) String() string {
  56. return "mtime"
  57. }
  58. func (f *mtimeFS) Chtimes(name string, atime, mtime time.Time) error {
  59. // Do a normal Chtimes call, don't care if it succeeds or not.
  60. _ = f.chtimes(name, atime, mtime)
  61. // Stat the file to see what happened. Here we *do* return an error,
  62. // because it might be "does not exist" or similar.
  63. info, err := f.Filesystem.Lstat(name)
  64. if err != nil {
  65. return err
  66. }
  67. f.save(name, info.ModTime(), mtime)
  68. return nil
  69. }
  70. func (f *mtimeFS) Stat(name string) (FileInfo, error) {
  71. info, err := f.Filesystem.Stat(name)
  72. if err != nil {
  73. return nil, err
  74. }
  75. ondisk, virtual := f.load(name)
  76. if ondisk.Equal(info.ModTime()) {
  77. info = mtimeFileInfo{
  78. FileInfo: info,
  79. mtime: virtual,
  80. }
  81. }
  82. return info, nil
  83. }
  84. func (f *mtimeFS) Lstat(name string) (FileInfo, error) {
  85. info, err := f.Filesystem.Lstat(name)
  86. if err != nil {
  87. return nil, err
  88. }
  89. ondisk, virtual := f.load(name)
  90. if ondisk.Equal(info.ModTime()) {
  91. info = mtimeFileInfo{
  92. FileInfo: info,
  93. mtime: virtual,
  94. }
  95. }
  96. return info, nil
  97. }
  98. func (f *mtimeFS) Create(name string) (File, error) {
  99. fd, err := f.Filesystem.Create(name)
  100. if err != nil {
  101. return nil, err
  102. }
  103. return mtimeFile{fd, f}, nil
  104. }
  105. func (f *mtimeFS) Open(name string) (File, error) {
  106. fd, err := f.Filesystem.Open(name)
  107. if err != nil {
  108. return nil, err
  109. }
  110. return mtimeFile{fd, f}, nil
  111. }
  112. func (f *mtimeFS) OpenFile(name string, flags int, mode FileMode) (File, error) {
  113. fd, err := f.Filesystem.OpenFile(name, flags, mode)
  114. if err != nil {
  115. return nil, err
  116. }
  117. return mtimeFile{fd, f}, nil
  118. }
  119. func (f *mtimeFS) underlying() (Filesystem, bool) {
  120. return f.Filesystem, true
  121. }
  122. func (f *mtimeFS) save(name string, ondisk, virtual time.Time) {
  123. if f.caseInsensitive {
  124. name = UnicodeLowercaseNormalized(name)
  125. }
  126. if ondisk.Equal(virtual) {
  127. // If the virtual time and the real on disk time are equal we don't
  128. // need to store anything.
  129. _ = f.db.DeleteMtime(f.folderID, name)
  130. return
  131. }
  132. _ = f.db.PutMtime(f.folderID, name, ondisk, virtual)
  133. }
  134. func (f *mtimeFS) load(name string) (ondisk, virtual time.Time) {
  135. if f.caseInsensitive {
  136. name = UnicodeLowercaseNormalized(name)
  137. }
  138. return f.db.GetMtime(f.folderID, name)
  139. }
  140. // The mtimeFileInfo is an os.FileInfo that lies about the ModTime().
  141. type mtimeFileInfo struct {
  142. FileInfo
  143. mtime time.Time
  144. }
  145. func (m mtimeFileInfo) ModTime() time.Time {
  146. return m.mtime
  147. }
  148. type mtimeFile struct {
  149. File
  150. fs *mtimeFS
  151. }
  152. func (f mtimeFile) Stat() (FileInfo, error) {
  153. info, err := f.File.Stat()
  154. if err != nil {
  155. return nil, err
  156. }
  157. ondisk, virtual := f.fs.load(f.Name())
  158. if ondisk.Equal(info.ModTime()) {
  159. info = mtimeFileInfo{
  160. FileInfo: info,
  161. mtime: virtual,
  162. }
  163. }
  164. return info, nil
  165. }
  166. // Used by copyRange to unwrap to the real file and access SyscallConn
  167. func (f mtimeFile) unwrap() File {
  168. return f.File
  169. }
  170. func GetMtimeMapping(fs Filesystem, file string) (ondisk, virtual time.Time) {
  171. fs, ok := unwrapFilesystem[*mtimeFS](fs)
  172. if !ok {
  173. return time.Time{}, time.Time{}
  174. }
  175. mtimeFs, ok := fs.(*mtimeFS)
  176. if !ok {
  177. return time.Time{}, time.Time{}
  178. }
  179. return mtimeFs.load(file)
  180. }