readonly_unix.go 5.15 KB
Newer Older
1 2
// A Go mirror of libfuse's hello.c

3
// +build linux darwin freebsd
4

5 6 7
package readonly

import (
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
8
	"fmt"
9
	"io/ioutil"
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
10 11 12 13 14 15
	"os"
	"os/exec"
	"os/signal"
	"runtime"
	"syscall"
	"time"
16

17 18 19
	fuse "github.com/jbenet/go-ipfs/Godeps/_workspace/src/bazil.org/fuse"
	fs "github.com/jbenet/go-ipfs/Godeps/_workspace/src/bazil.org/fuse/fs"
	proto "github.com/jbenet/go-ipfs/Godeps/_workspace/src/code.google.com/p/goprotobuf/proto"
Jeromy's avatar
Jeromy committed
20

21 22
	core "github.com/jbenet/go-ipfs/core"
	mdag "github.com/jbenet/go-ipfs/merkledag"
23
	uio "github.com/jbenet/go-ipfs/unixfs/io"
24
	ftpb "github.com/jbenet/go-ipfs/unixfs/pb"
25
	u "github.com/jbenet/go-ipfs/util"
26 27
)

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
28
var log = u.Logger("ipfs")
Jeromy's avatar
Jeromy committed
29

Juan Batiz-Benet's avatar
go lint  
Juan Batiz-Benet committed
30
// FileSystem is the readonly Ipfs Fuse Filesystem.
31
type FileSystem struct {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
32
	Ipfs *core.IpfsNode
33 34
}

Juan Batiz-Benet's avatar
go lint  
Juan Batiz-Benet committed
35
// NewFileSystem constructs new fs using given core.IpfsNode instance.
36
func NewFileSystem(ipfs *core.IpfsNode) *FileSystem {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
37
	return &FileSystem{Ipfs: ipfs}
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
38 39
}

Juan Batiz-Benet's avatar
go lint  
Juan Batiz-Benet committed
40
// Root constructs the Root of the filesystem, a Root object.
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
41
func (f FileSystem) Root() (fs.Node, fuse.Error) {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
42
	return &Root{Ipfs: f.Ipfs}, nil
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
43 44
}

Juan Batiz-Benet's avatar
go lint  
Juan Batiz-Benet committed
45
// Root is the root object of the filesystem tree.
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
46
type Root struct {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
47
	Ipfs *core.IpfsNode
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
48 49
}

Juan Batiz-Benet's avatar
go lint  
Juan Batiz-Benet committed
50
// Attr returns file attributes.
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
51 52
func (*Root) Attr() fuse.Attr {
	return fuse.Attr{Mode: os.ModeDir | 0111} // -rw+x
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
53 54
}

Juan Batiz-Benet's avatar
go lint  
Juan Batiz-Benet committed
55
// Lookup performs a lookup under this node.
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
56
func (s *Root) Lookup(name string, intr fs.Intr) (fs.Node, fuse.Error) {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
57
	log.Debugf("Root Lookup: '%s'", name)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
58 59 60 61 62 63 64 65 66 67 68 69 70
	switch name {
	case "mach_kernel", ".hidden", "._.":
		// Just quiet some log noise on OS X.
		return nil, fuse.ENOENT
	}

	nd, err := s.Ipfs.Resolver.ResolvePath(name)
	if err != nil {
		// todo: make this error more versatile.
		return nil, fuse.ENOENT
	}

	return &Node{Ipfs: s.Ipfs, Nd: nd}, nil
71 72
}

Juan Batiz-Benet's avatar
go lint  
Juan Batiz-Benet committed
73
// ReadDir reads a particular directory. Disallowed for root.
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
74
func (*Root) ReadDir(intr fs.Intr) ([]fuse.Dirent, fuse.Error) {
Jeromy's avatar
Jeromy committed
75
	log.Debug("Read Root.")
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
76
	return nil, fuse.EPERM
77 78
}

Juan Batiz-Benet's avatar
go lint  
Juan Batiz-Benet committed
79
// Node is the core object representing a filesystem tree node.
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
80
type Node struct {
Jeromy's avatar
Jeromy committed
81 82
	Ipfs   *core.IpfsNode
	Nd     *mdag.Node
83
	fd     *uio.DagReader
84
	cached *ftpb.Data
Jeromy's avatar
Jeromy committed
85 86 87
}

func (s *Node) loadData() error {
88
	s.cached = new(ftpb.Data)
Jeromy's avatar
Jeromy committed
89
	return proto.Unmarshal(s.Nd.Data, s.cached)
90 91
}

Juan Batiz-Benet's avatar
go lint  
Juan Batiz-Benet committed
92
// Attr returns the attributes of a given node.
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
93
func (s *Node) Attr() fuse.Attr {
Jeromy's avatar
Jeromy committed
94
	log.Debug("Node attr.")
Jeromy's avatar
Jeromy committed
95 96 97 98
	if s.cached == nil {
		s.loadData()
	}
	switch s.cached.GetType() {
99
	case ftpb.Data_Directory:
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
100
		return fuse.Attr{Mode: os.ModeDir | 0555}
101
	case ftpb.Data_File, ftpb.Data_Raw:
Jeromy's avatar
Jeromy committed
102
		size, _ := s.Nd.Size()
Jeromy's avatar
Jeromy committed
103 104 105 106 107
		return fuse.Attr{
			Mode:   0444,
			Size:   uint64(size),
			Blocks: uint64(len(s.Nd.Links)),
		}
Jeromy's avatar
Jeromy committed
108
	default:
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
109
		log.Error("Invalid data type.")
Jeromy's avatar
Jeromy committed
110
		return fuse.Attr{}
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
111
	}
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
112
}
113

Juan Batiz-Benet's avatar
go lint  
Juan Batiz-Benet committed
114
// Lookup performs a lookup under this node.
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
115
func (s *Node) Lookup(name string, intr fs.Intr) (fs.Node, fuse.Error) {
116
	log.Debugf("Lookup '%s'", name)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
117 118 119 120 121
	nd, err := s.Ipfs.Resolver.ResolveLinks(s.Nd, []string{name})
	if err != nil {
		// todo: make this error more versatile.
		return nil, fuse.ENOENT
	}
122

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
123
	return &Node{Ipfs: s.Ipfs, Nd: nd}, nil
124 125
}

Juan Batiz-Benet's avatar
go lint  
Juan Batiz-Benet committed
126
// ReadDir reads the link structure as directory entries
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
127
func (s *Node) ReadDir(intr fs.Intr) ([]fuse.Dirent, fuse.Error) {
Jeromy's avatar
Jeromy committed
128
	log.Debug("Node ReadDir")
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
129 130 131 132 133 134 135 136 137 138 139 140 141
	entries := make([]fuse.Dirent, len(s.Nd.Links))
	for i, link := range s.Nd.Links {
		n := link.Name
		if len(n) == 0 {
			n = link.Hash.B58String()
		}
		entries[i] = fuse.Dirent{Name: n, Type: fuse.DT_File}
	}

	if len(entries) > 0 {
		return entries, nil
	}
	return nil, fuse.ENOENT
142 143
}

Juan Batiz-Benet's avatar
go lint  
Juan Batiz-Benet committed
144
// ReadAll reads the object data as file data
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
145
func (s *Node) ReadAll(intr fs.Intr) ([]byte, fuse.Error) {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
146
	log.Debug("Read node.")
147
	r, err := uio.NewDagReader(s.Nd, s.Ipfs.DAG)
148 149 150 151 152 153
	if err != nil {
		return nil, err
	}
	// this is a terrible function... 'ReadAll'?
	// what if i have a 6TB file? GG RAM.
	return ioutil.ReadAll(r)
154 155
}

Juan Batiz-Benet's avatar
go lint  
Juan Batiz-Benet committed
156 157
// Mount mounts an IpfsNode instance at a particular path. It
// serves until the process receives exit signals (to Unmount).
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
158 159 160 161 162 163 164
func Mount(ipfs *core.IpfsNode, fpath string) error {

	sigc := make(chan os.Signal, 1)
	signal.Notify(sigc, syscall.SIGHUP, syscall.SIGINT,
		syscall.SIGTERM, syscall.SIGQUIT)

	go func() {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
165
		defer ipfs.Network.Close()
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
166
		<-sigc
167 168 169 170 171 172 173
		for {
			err := Unmount(fpath)
			if err == nil {
				return
			}
			time.Sleep(time.Millisecond * 10)
		}
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193
	}()

	c, err := fuse.Mount(fpath)
	if err != nil {
		return err
	}
	defer c.Close()

	err = fs.Serve(c, FileSystem{Ipfs: ipfs})
	if err != nil {
		return err
	}

	// check if the mount process has an error to report
	<-c.Ready
	if err := c.MountError; err != nil {
		return err
	}
	return nil
}
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
194

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
195 196 197
// Unmount attempts to unmount the provided FUSE mount point, forcibly
// if necessary.
func Unmount(point string) error {
Jeromy's avatar
Jeromy committed
198
	log.Info("Unmounting %s...", point)
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224

	var cmd *exec.Cmd
	switch runtime.GOOS {
	case "darwin":
		cmd = exec.Command("diskutil", "umount", "force", point)
	case "linux":
		cmd = exec.Command("fusermount", "-u", point)
	default:
		return fmt.Errorf("unmount: unimplemented")
	}

	errc := make(chan error, 1)
	go func() {
		if err := exec.Command("umount", point).Run(); err == nil {
			errc <- err
		}
		// retry to unmount with the fallback cmd
		errc <- cmd.Run()
	}()

	select {
	case <-time.After(1 * time.Second):
		return fmt.Errorf("umount timeout")
	case err := <-errc:
		return err
	}
225
}