Commit 4aed08d3 authored by Han-Wen Nienhuys's avatar Han-Wen Nienhuys

fs: add more package docs.

Add dynamic_example_test.go to Readdir/Lookup.

Change-Id: If7cd2a29166b9952af0ca362cba89df1d98b634b
parent aedffc58
...@@ -4,10 +4,57 @@ ...@@ -4,10 +4,57 @@
// Package fs provides infrastructure to build tree-organized filesystems. // Package fs provides infrastructure to build tree-organized filesystems.
// //
// A tree-organized filesystem is similar to UNIX or Plan 9 filesystem: it // Structure of a file system implementation
// consists of nodes with each node being either a file or a directory. Files //
// are located at tree leafs. A directory node can have other nodes as its // To create a file system, you should first define types for the
// children and refer to each child by name unique through the directory. // nodes of the file system tree.
//
// struct myNode {
// fs.Inode
// }
//
// // Node types must be InodeEmbedders
// var _ = (fs.InodeEmbedder)((*myNode)(nil))
//
// // Node types should implement some file system operations, eg. Lookup
// var _ = (fs.NodeLookuper)((*myNode)(nil))
//
// func (n *myNode) Lookup(ctx context.Context, name string, ... ) (*Inode, syscall.Errno) {
// ops := myNode{}
// return n.NewInode(ctx, &ops, fs.StableAttr{Mode: syscall.S_IFDIR}), 0
// }
//
// The method names are inspired on the system call names, so we have
// Listxattr rather than ListXAttr.
//
// the file system is mounted by calling mount on the root of the tree,
//
// server, err := fs.Mount("/tmp/mnt", &myNode{}, &fs.Options{})
// ..
// // start serving the file system
// server.Wait()
//
// Error handling
//
// All error reporting must use the syscall.Errno type. This is an
// integer with predefined error codes, where the value 0 (`OK`)
// should be used to indicate success.
//
// File system concepts
//
// The FUSE API is very similar to Linux' internal VFS API for
// defining file systems in the kernel. It is therefore useful to
// understand some terminology.
//
// File content: the raw bytes that we store inside regular files.
//
// Path: a /-separated string path that describes location of a node
// in the file system tree. For example
//
// dir1/file
//
// describes path root → dir1 → file.
//
// There can be several paths leading from tree root to a particular node, // There can be several paths leading from tree root to a particular node,
// known as hard-linking, for example // known as hard-linking, for example
// //
...@@ -17,59 +64,105 @@ ...@@ -17,59 +64,105 @@
// \ / // \ /
// file // file
// //
// A /-separated string path describes location of a node in the tree. For example // Inode: ("index node") points to the file content, and stores
// metadata (size, timestamps) about a file or directory. Each
// directory has a type (directory, symlink, regular file, etc.) and
// an identity (a 64-bit number, unique to the file
// system). Directories can have children.
// //
// dir1/file // The inode in the kernel is represented in Go-FUSE as the Inode
// type.
// //
// describes path root → dir1 → file. // While common OS APIs are phrased in terms of paths (strings), the
// precise semantics of a file system are better described in terms of
// Inodes. This allows us to specify what happens in corner cases,
// such as writing data to deleted files.
// //
// Each node is associated with integer ID uniquely identifying the node // File descriptor: a handle returned to opening a file. File
// throughout filesystem. The tree-level structure of any filesystem is // descriptors always refer to a single inode.
// expressed through index-nodes (also known as "inode", see Inode) which
// describe parent/child relation in between nodes and node-ID association.
// //
// The filesystem nodes are struct that embed the Inode type, so they // Dirent: a dirent maps (parent inode number, name string) tuple to
// comply with the InodeEmbedder interface. They should be // child inode, thus representing a parent/child relation (or the
// initialized by calling NewInode or NewPersistentInode before being // absense thereof). Dirents do not have an equivalent type inside
// manipulated further, eg. // Go-FUSE, but the result of Lookup operation essentially is a
// dirent, which the kernel puts in a cache.
// //
// type myNode struct {
// Inode
// }
// //
// func (n *myNode) Lookup(ctx context.Context, name string, ... ) (*Inode, syscall.Errno) { // Kernel caching
// child := myNode{} //
// return n.NewInode(ctx, &myNode{}, StableAttr{Mode: syscall.S_IFDIR}), 0 // The kernel caches several pieces of information from the FUSE process:
// } //
// 1. File contents: enabled with the fuse.FOPEN_KEEP_CACHE return flag
// in Open, manipulated with ReadCache and WriteCache, and invalidated
// with Inode.NotifyContent
//
// 2. File Attributes (size, mtime, etc.): controlled with the
// attribute timeout fields in fuse.AttrOut and fuse.EntryOut, which
// get be populated from Getattr and Lookup
// //
// On mounting, the root InodeEmbedder is associated with root of the // 3. Directory entries (parent/child relations in the FS tree):
// tree. // controlled with the timeout fields in fuse.EntryOut, and
// invalidated with Inode.NotifyEntry and Inode.NotifyDelete.
// //
// The kernel can evict inode data to free up memory. It does so by // Without Directory Entry timeouts, every operation on file "a/b/c"
// issuing FORGET calls. When a node has no children, and no kernel // must first do lookups for "a", "a/b" and "a/b/c", which is
// references, it is removed from the file system trees. // expensive because of context switches between the kernel and the
// FUSE process.
// //
// File system trees can also be constructed in advance. This is done // Unsuccessful entry lookups can also be cached by setting an entry
// by instantiating "persistent" inodes from the OnAdder // timeout when Lookup returns ENOENT.
// implementation. Persistent inodes remain in memory even if the
// kernel has forgotten them. See zip_test.go for an example of how
// to do this.
// //
// File systems whose tree structures are on backing storage typically // The libfuse C library specifies 1 second timeouts for both
// discover the file system tree on-demand, and if the kernel is tight // attribute and directory entries, but no timeout for negative
// on memory, parts of the tree are forgotten again. These file // entries. by default. This can be achieve in go-fuse by setting
// systems should implement Lookuper instead. The loopback file // options on mount, eg.
// system created by `NewLoopbackRoot` provides a straightforward
// example.
// //
// All error reporting must use the syscall.Errno type. The value 0 // sec := time.Second
// (`OK`) should be used to indicate success. The method names are // opts := fs.Options{
// inspired on the system call names, so we have Listxattr rather than // EntryTimeout: &sec,
// ListXAttr. // AttrTimeout: &sec,
// }
//
// Locking
// //
// Locks for networked filesystems are supported through the suite of // Locks for networked filesystems are supported through the suite of
// Getlk, Setlk and Setlkw methods. They alllow locks on regions of // Getlk, Setlk and Setlkw methods. They alllow locks on regions of
// regular files. // regular files.
//
// Parallelism
//
// The VFS layer in the kernel is optimized to be highly parallel, and
// this parallelism also affects FUSE file systems: many FUSE
// operations can run in parallel, and this invites race
// conditions. It is strongly recommended to test your FUSE file
// system issuing file operations in parallel, and using the race
// detector to weed out data races.
//
// Dynamically discovered file systems
//
// File system data usually cannot fit all in RAM, so the kernel must
// discover the file system dynamically: as you are entering and list
// directory contents, the kernel asks the FUSE server about the files
// and directories you are busy reading/writing, and forgets parts of
// your file system when it is low on memory.
//
// The two important operations for dynamic file systems are:
// 1. Lookup, part of the NodeLookuper interface for discovering
// individual children of directories, and 2. Readdir, part of the
// NodeReaddirer interface for listing the contents of a directory.
//
// Static in-memory file systems
//
// For small, read-only file systems, getting the locking mechanics of
// Lookup correct is tedious, so Go-FUSE provides a feature to
// simplify building such file systems.
//
// Instead of discovering the FS tree on the fly, you can construct
// the entire tree from an OnAdd method. Then, that in-memory tree
// structure becomes the source of truth. This means you Go-FUSE must
// remember Inodes even if the kernel is no longer interested in
// them. This is done by instantiating "persistent" inodes. See
// zip_test.go for an example of how to do this.
package fs package fs
import ( import (
...@@ -278,16 +371,31 @@ type DirStream interface { ...@@ -278,16 +371,31 @@ type DirStream interface {
Close() Close()
} }
// Lookup should find a direct child of the node by child // Lookup should find a direct child of a directory by the child's name. If
// name. If the entry does not exist, it should return ENOENT // the entry does not exist, it should return ENOENT and optionally
// and optionally set a NegativeTimeout in `out`. If it does // set a NegativeTimeout in `out`. If it does exist, it should return
// exist, it should return attribute data in `out` and return // attribute data in `out` and return the Inode for the child. A new
// the Inode for the child. A new inode can be created using // inode can be created using `Inode.NewInode`. The new Inode will be
// `Inode.NewInode`. The new Inode will be added to the FS // added to the FS tree automatically if the return status is OK.
// tree automatically if the return status is OK. //
// If a directory does not implement NodeLookuper, the library looks
// for an existing child with the given name.
//
// The input to a Lookup is {parent directory, name string}.
//
// Lookup, if successful, must return an *Inode. Once the Inode is
// returned to the kernel, the kernel can issue further operations,
// such as Open or Getxattr on that node.
//
// A successful Lookup also returns an EntryOut. Among others, this
// contains file attributes (mode, size, mtime, etc.).
//
// FUSE supports other operations that modify the namespace. For
// example, the Symlink, Create, Mknod, Link methods all create new
// children in directories. Hence, they also return *Inode and must
// populate their fuse.EntryOut arguments.
// //
// If not defined, we look for an existing child with the given name,
// or returns ENOENT.
type NodeLookuper interface { type NodeLookuper interface {
Lookup(ctx context.Context, name string, out *fuse.EntryOut) (*Inode, syscall.Errno) Lookup(ctx context.Context, name string, out *fuse.EntryOut) (*Inode, syscall.Errno)
} }
...@@ -302,8 +410,15 @@ type NodeOpendirer interface { ...@@ -302,8 +410,15 @@ type NodeOpendirer interface {
// ReadDir opens a stream of directory entries. // ReadDir opens a stream of directory entries.
// //
// The default ReadDir returns the list of currently known children // Readdir essentiallly returns a list of strings, and it is allowed
// from the tree // for Readdir to return different results from Lookup. For example,
// you can return nothing for Readdir ("ls my-fuse-mount" is empty),
// while still implementing Lookup ("ls my-fuse-mount/a-specific-file"
// shows a single file).
//
// If a directory does not implement NodeReaddirer, a list of
// currently known children from the tree is returned. This means that
// static in-memory file systems need not implement NodeReaddirer.
type NodeReaddirer interface { type NodeReaddirer interface {
Readdir(ctx context.Context) (DirStream, syscall.Errno) Readdir(ctx context.Context) (DirStream, syscall.Errno)
} }
......
// Copyright 2019 the Go-FUSE Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
package fs_test
import (
"context"
"log"
"os"
"strconv"
"syscall"
"github.com/hanwen/go-fuse/fs"
"github.com/hanwen/go-fuse/fuse"
)
// numberNode is a filesystem node representing an integer. Prime
// numbers are regular files, while composite numbers are directories
// containing all smaller numbers, eg.
//
// $ ls -F /tmp/x/6
// 2 3 4/ 5
//
// the file system nodes are deduplicated using inode numbers. The
// number 2 appears in many directories, but it is actually the represented
// by the same numberNode{} object, with inode number 2.
//
// $ ls -i1 /tmp/x/2 /tmp/x/8/6/4/2
// 2 /tmp/x/2
// 2 /tmp/x/8/6/4/2
//
type numberNode struct {
// Must embed an Inode for the struct to work as a node.
fs.Inode
// num is the integer represented in this file/directory
num int
}
// isPrime returns whether n is prime
func isPrime(n int) bool {
for i := 2; i*i <= n; i++ {
if n%i == 0 {
return false
}
}
return true
}
func numberToMode(n int) uint32 {
// prime numbers are files
if isPrime(n) {
return fuse.S_IFREG
}
// composite numbers are directories
return fuse.S_IFDIR
}
// Ensure we are implementing the NodeReaddirer interface
var _ = (fs.NodeReaddirer)((*numberNode)(nil))
// Readdir is part of the NodeReaddirer interface
func (n *numberNode) Readdir(ctx context.Context) (fs.DirStream, syscall.Errno) {
r := make([]fuse.DirEntry, 0, n.num)
for i := 2; i < n.num; i++ {
d := fuse.DirEntry{
Name: strconv.Itoa(i),
Ino: uint64(i),
Mode: numberToMode(i),
}
r = append(r, d)
}
return fs.NewListDirStream(r), 0
}
// Ensure we are implementing the NodeLookuper interface
var _ = (fs.NodeLookuper)((*numberNode)(nil))
// Lookup is part of the NodeLookuper interface
func (n *numberNode) Lookup(ctx context.Context, name string, out *fuse.EntryOut) (*fs.Inode, syscall.Errno) {
i, err := strconv.Atoi(name)
if err != nil {
return nil, syscall.ENOENT
}
if i >= n.num || i <= 1 {
return nil, syscall.ENOENT
}
stable := fs.StableAttr{
Mode: numberToMode(i),
// The child inode is identified by its Inode number.
// If multiple concurrent lookups try to find the same
// inode, they are deduplicated on this key.
Ino: uint64(i),
}
operations := &numberNode{num: i}
// The NewInode call wraps the `operations` object into an Inode.
child := n.NewInode(ctx, operations, stable)
// In case of concurrent lookup requests, it can happen that operations !=
// child.Operations().
return child, 0
}
// ExampleDynamic is a whimsical example of a dynamically discovered
// file system.
func Example_dynamic() {
// This is where we'll mount the FS
mntDir := "/tmp/x"
os.Mkdir(mntDir, 0755)
root := &numberNode{num: 10}
server, err := fs.Mount(mntDir, root, &fs.Options{
MountOptions: fuse.MountOptions{
// Set to true to see how the file system works.
Debug: true,
},
// This adds read permissions to the files and
// directories, which is necessary for doing a chdir
// into the mount.
DefaultPermissions: true,
})
if err != nil {
log.Panic(err)
}
log.Printf("Mounted on %s", mntDir)
log.Printf("Unmount by calling 'fusermount -u %s'", mntDir)
// Wait until unmount before exiting
server.Wait()
}
...@@ -14,37 +14,30 @@ import ( ...@@ -14,37 +14,30 @@ import (
"github.com/hanwen/go-fuse/fuse" "github.com/hanwen/go-fuse/fuse"
) )
// mountLoopback mounts dir under the given mountpoint
func mountLoopback(dir, mntPoint string) (*fuse.Server, error) {
root, err := fs.NewLoopbackRoot(dir)
if err != nil {
return nil, err
}
// Make the root available under mntDir
return fs.Mount(mntPoint, root, &fs.Options{
MountOptions: fuse.MountOptions{Debug: true},
})
}
// An example of creating a loopback file system, and mounting it onto // An example of creating a loopback file system, and mounting it onto
// a directory // a directory
func Example_mountLoopback() { func Example_mountLoopback() {
mntDir, _ := ioutil.TempDir("", "") mntDir, _ := ioutil.TempDir("", "")
home := os.Getenv("HOME") home := os.Getenv("HOME")
// Make $HOME available on a mount dir under /tmp/ . Caution: // Make $HOME available on a mount dir under /tmp/ . Caution:
// write operations are also mirrored. // write operations are also mirrored.
server, err := mountLoopback(mntDir, home) root, err := fs.NewLoopbackRoot(home)
if err != nil {
log.Fatal(err)
}
// Mount the file system
server, err := fs.Mount(mntDir, root, &fs.Options{
MountOptions: fuse.MountOptions{Debug: true},
})
if err != nil { if err != nil {
log.Panic(err) log.Fatal(err)
} }
fmt.Printf("Mounted %s as loopback on %s\n", home, mntDir) fmt.Printf("Mounted %s as loopback on %s\n", home, mntDir)
fmt.Printf("\n\nCAUTION:\nwrite operations on %s will also affect $HOME (%s)\n\n", mntDir, home) fmt.Printf("\n\nCAUTION:\nwrite operations on %s will also affect $HOME (%s)\n\n", mntDir, home)
fmt.Printf("Unmount by calling 'fusermount -u %s'\n", mntDir) fmt.Printf("Unmount by calling 'fusermount -u %s'\n", mntDir)
// Wait until the directory is unmounted // Serve the file system, until unmounted by calling fusermount -u
server.Wait() server.Wait()
} }
...@@ -316,6 +316,15 @@ func (iparent *Inode) setEntry(name string, ichild *Inode) { ...@@ -316,6 +316,15 @@ func (iparent *Inode) setEntry(name string, ichild *Inode) {
// NewPersistentInode returns an Inode whose lifetime is not in // NewPersistentInode returns an Inode whose lifetime is not in
// control of the kernel. // control of the kernel.
//
// When the kernel is short on memory, it will forget cached file
// system information (directory entries and inode metadata). This is
// announced with FORGET messages. There are no guarantees if or when
// this happens. When it happens, these are handled transparently by
// go-fuse: all Inodes created with NewInode are released
// automatically. NewPersistentInode creates inodes that go-fuse keeps
// in memory, even if the kernel is not interested in them. This is
// convenient for building static trees up-front.
func (n *Inode) NewPersistentInode(ctx context.Context, node InodeEmbedder, id StableAttr) *Inode { func (n *Inode) NewPersistentInode(ctx context.Context, node InodeEmbedder, id StableAttr) *Inode {
return n.newInode(ctx, node, id, true) return n.newInode(ctx, node, id, true)
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment