mount.go 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163
  1. // Copyright 2015 Google Inc. All Rights Reserved.
  2. //
  3. // Licensed under the Apache License, Version 2.0 (the "License");
  4. // you may not use this file except in compliance with the License.
  5. // You may obtain a copy of the License at
  6. //
  7. // http://www.apache.org/licenses/LICENSE-2.0
  8. //
  9. // Unless required by applicable law or agreed to in writing, software
  10. // distributed under the License is distributed on an "AS IS" BASIS,
  11. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. // See the License for the specific language governing permissions and
  13. // limitations under the License.
  14. // A simple tool for mounting sample file systems, used by the tests in
  15. // samples/.
  16. package main
  17. import (
  18. "errors"
  19. "flag"
  20. "fmt"
  21. "log"
  22. "os"
  23. "runtime"
  24. "syscall"
  25. "github.com/jacobsa/fuse"
  26. "github.com/jacobsa/fuse/samples/flushfs"
  27. "golang.org/x/net/context"
  28. )
  29. var fType = flag.String("type", "", "The name of the samples/ sub-dir.")
  30. var fMountPoint = flag.String("mount_point", "", "Path to mount point.")
  31. var fReadyFile = flag.Uint64("ready_file", 0, "FD to signal when ready.")
  32. var fFlushesFile = flag.Uint64("flushfs.flushes_file", 0, "")
  33. var fFsyncsFile = flag.Uint64("flushfs.fsyncs_file", 0, "")
  34. var fFlushError = flag.Int("flushfs.flush_error", 0, "")
  35. var fFsyncError = flag.Int("flushfs.fsync_error", 0, "")
  36. var fReadOnly = flag.Bool("read_only", false, "Mount in read-only mode.")
  37. var fDebug = flag.Bool("debug", false, "Enable debug logging.")
  38. func makeFlushFS() (server fuse.Server, err error) {
  39. // Check the flags.
  40. if *fFlushesFile == 0 || *fFsyncsFile == 0 {
  41. err = fmt.Errorf("You must set the flushfs flags.")
  42. return
  43. }
  44. // Set up the files.
  45. flushes := os.NewFile(uintptr(*fFlushesFile), "(flushes file)")
  46. fsyncs := os.NewFile(uintptr(*fFsyncsFile), "(fsyncs file)")
  47. // Set up errors.
  48. var flushErr error
  49. var fsyncErr error
  50. if *fFlushError != 0 {
  51. flushErr = syscall.Errno(*fFlushError)
  52. }
  53. if *fFsyncError != 0 {
  54. fsyncErr = syscall.Errno(*fFsyncError)
  55. }
  56. // Report flushes and fsyncs by writing the contents followed by a newline.
  57. report := func(f *os.File, outErr error) func(string) error {
  58. return func(s string) (err error) {
  59. buf := []byte(s)
  60. buf = append(buf, '\n')
  61. _, err = f.Write(buf)
  62. if err != nil {
  63. err = fmt.Errorf("Write: %v", err)
  64. return
  65. }
  66. err = outErr
  67. return
  68. }
  69. }
  70. reportFlush := report(flushes, flushErr)
  71. reportFsync := report(fsyncs, fsyncErr)
  72. // Create the file system.
  73. server, err = flushfs.NewFileSystem(reportFlush, reportFsync)
  74. return
  75. }
  76. func makeFS() (server fuse.Server, err error) {
  77. switch *fType {
  78. default:
  79. err = fmt.Errorf("Unknown FS type: %v", *fType)
  80. case "flushfs":
  81. server, err = makeFlushFS()
  82. }
  83. return
  84. }
  85. func getReadyFile() (f *os.File, err error) {
  86. if *fReadyFile == 0 {
  87. err = errors.New("You must set --ready_file.")
  88. return
  89. }
  90. f = os.NewFile(uintptr(*fReadyFile), "(ready file)")
  91. return
  92. }
  93. func main() {
  94. flag.Parse()
  95. // Allow parallelism in the file system implementation, to help flush out
  96. // bugs like https://github.com/jacobsa/fuse/issues/4.
  97. runtime.GOMAXPROCS(2)
  98. // Grab the file to signal when ready.
  99. readyFile, err := getReadyFile()
  100. if err != nil {
  101. log.Fatalf("getReadyFile:", err)
  102. }
  103. // Create an appropriate file system.
  104. server, err := makeFS()
  105. if err != nil {
  106. log.Fatalf("makeFS: %v", err)
  107. }
  108. // Mount the file system.
  109. if *fMountPoint == "" {
  110. log.Fatalf("You must set --mount_point.")
  111. }
  112. cfg := &fuse.MountConfig{
  113. ReadOnly: *fReadOnly,
  114. }
  115. if *fDebug {
  116. cfg.DebugLogger = log.New(os.Stderr, "fuse: ", 0)
  117. }
  118. mfs, err := fuse.Mount(*fMountPoint, server, cfg)
  119. if err != nil {
  120. log.Fatalf("Mount: %v", err)
  121. }
  122. // Signal that it is ready.
  123. _, err = readyFile.Write([]byte("x"))
  124. if err != nil {
  125. log.Fatalf("readyFile.Write: %v", err)
  126. }
  127. // Wait for it to be unmounted.
  128. if err = mfs.Join(context.Background()); err != nil {
  129. log.Fatalf("Join: %v", err)
  130. }
  131. }