package ipns import ( "bytes" "fmt" "io/ioutil" "os" "path/filepath" "time" "github.com/jbenet/go-ipfs/Godeps/_workspace/src/bazil.org/fuse" "github.com/jbenet/go-ipfs/Godeps/_workspace/src/bazil.org/fuse/fs" "github.com/jbenet/go-ipfs/Godeps/_workspace/src/code.google.com/p/goprotobuf/proto" logging "github.com/jbenet/go-ipfs/Godeps/_workspace/src/github.com/op/go-logging" "github.com/jbenet/go-ipfs/core" ci "github.com/jbenet/go-ipfs/crypto" imp "github.com/jbenet/go-ipfs/importer" mdag "github.com/jbenet/go-ipfs/merkledag" u "github.com/jbenet/go-ipfs/util" ) var log = logging.MustGetLogger("ipns") // FileSystem is the readwrite IPNS Fuse Filesystem. type FileSystem struct { Ipfs *core.IpfsNode RootNode *Root } // NewFileSystem constructs new fs using given core.IpfsNode instance. func NewIpns(ipfs *core.IpfsNode, ipfspath string) (*FileSystem, error) { root, err := CreateRoot(ipfs, []ci.PrivKey{ipfs.Identity.PrivKey}, ipfspath) if err != nil { return nil, err } return &FileSystem{Ipfs: ipfs, RootNode: root}, nil } func CreateRoot(n *core.IpfsNode, keys []ci.PrivKey, ipfsroot string) (*Root, error) { root := new(Root) root.LocalDirs = make(map[string]*Node) root.Ipfs = n abspath, err := filepath.Abs(ipfsroot) if err != nil { return nil, err } root.IpfsRoot = abspath root.Keys = keys if len(keys) == 0 { log.Warning("No keys given for ipns root creation") } else { k := keys[0] pub := k.GetPublic() hash, err := pub.Hash() if err != nil { log.Error("Read Root Error: %s", err) return nil, err } root.LocalLink = &Link{u.Key(hash).Pretty()} } for _, k := range keys { hash, err := k.GetPublic().Hash() if err != nil { log.Error("failed to hash public key.") continue } name := u.Key(hash).Pretty() nd := new(Node) nd.Ipfs = n nd.key = k nd.repub = NewRepublisher(nd, time.Millisecond*10, time.Second) go nd.repub.Run() pointsTo, err := n.Namesys.Resolve(name) if err != nil { log.Warning("Could not resolve value for local ipns entry, providing empty dir") nd.Nd = &mdag.Node{Data: mdag.FolderPBData()} root.LocalDirs[name] = nd continue } if !u.IsValidHash(pointsTo) { log.Critical("Got back bad data from namesys resolve! [%s]", pointsTo) return nil, nil } node, err := n.Resolver.ResolvePath(pointsTo) if err != nil { log.Warning("Failed to resolve value from ipns entry in ipfs") continue } nd.Nd = node root.LocalDirs[name] = nd } return root, nil } // Root constructs the Root of the filesystem, a Root object. func (f FileSystem) Root() (fs.Node, fuse.Error) { return f.RootNode, nil } // Root is the root object of the filesystem tree. type Root struct { Ipfs *core.IpfsNode Keys []ci.PrivKey // Used for symlinking into ipfs IpfsRoot string LocalDirs map[string]*Node LocalLink *Link } // Attr returns file attributes. func (*Root) Attr() fuse.Attr { return fuse.Attr{Mode: os.ModeDir | 0111} // -rw+x } // Lookup performs a lookup under this node. func (s *Root) Lookup(name string, intr fs.Intr) (fs.Node, fuse.Error) { log.Debug("ipns: Root Lookup: '%s'", name) switch name { case "mach_kernel", ".hidden", "._.": // Just quiet some log noise on OS X. return nil, fuse.ENOENT } if name == "local" { if s.LocalLink == nil { return nil, fuse.ENOENT } return s.LocalLink, nil } nd, ok := s.LocalDirs[name] if ok { return nd, nil } log.Debug("ipns: Falling back to resolution for [%s].", name) resolved, err := s.Ipfs.Namesys.Resolve(name) if err != nil { log.Warning("ipns: namesys resolve error: %s", err) return nil, fuse.ENOENT } return &Link{s.IpfsRoot + "/" + resolved}, nil } // ReadDir reads a particular directory. Disallowed for root. func (r *Root) ReadDir(intr fs.Intr) ([]fuse.Dirent, fuse.Error) { log.Debug("Read Root.") listing := []fuse.Dirent{ fuse.Dirent{ Name: "local", Type: fuse.DT_Link, }, } for _, k := range r.Keys { pub := k.GetPublic() hash, err := pub.Hash() if err != nil { log.Error("Read Root Error: %s", err) continue } ent := fuse.Dirent{ Name: u.Key(hash).Pretty(), Type: fuse.DT_Dir, } listing = append(listing, ent) } return listing, nil } // Node is the core object representing a filesystem tree node. type Node struct { root *Root nsRoot *Node parent *Node repub *Republisher // This nodes name in its parent dir. // NOTE: this strategy wont work well if we allow hard links // (im all for murdering the thought of hard links) name string // Private keys held by nodes at the root of a keyspace // WARNING(security): the PrivKey interface is currently insecure // (holds the raw key). It will be secured later. key ci.PrivKey Ipfs *core.IpfsNode Nd *mdag.Node fd *mdag.DagReader cached *mdag.PBData // For writing writerBuf WriteAtBuf } func (s *Node) loadData() error { s.cached = new(mdag.PBData) return proto.Unmarshal(s.Nd.Data, s.cached) } // Attr returns the attributes of a given node. func (s *Node) Attr() fuse.Attr { if s.cached == nil { s.loadData() } switch s.cached.GetType() { case mdag.PBData_Directory: return fuse.Attr{Mode: os.ModeDir | 0555} case mdag.PBData_File, mdag.PBData_Raw: size, _ := s.Nd.Size() return fuse.Attr{ Mode: 0666, Size: uint64(size), Blocks: uint64(len(s.Nd.Links)), } default: log.Error("Invalid data type.") return fuse.Attr{} } } // Lookup performs a lookup under this node. func (s *Node) Lookup(name string, intr fs.Intr) (fs.Node, fuse.Error) { log.Debug("ipns: node[%s] Lookup '%s'", s.name, name) nd, err := s.Ipfs.Resolver.ResolveLinks(s.Nd, []string{name}) if err != nil { // todo: make this error more versatile. return nil, fuse.ENOENT } return s.makeChild(name, nd), nil } func (n *Node) makeChild(name string, node *mdag.Node) *Node { child := &Node{ Ipfs: n.Ipfs, Nd: node, name: name, nsRoot: n.nsRoot, } if n.nsRoot == nil { child.nsRoot = n } else { child.nsRoot = n.nsRoot } return child } // ReadDir reads the link structure as directory entries func (s *Node) ReadDir(intr fs.Intr) ([]fuse.Dirent, fuse.Error) { log.Debug("Node ReadDir") 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 } // ReadAll reads the object data as file data func (s *Node) ReadAll(intr fs.Intr) ([]byte, fuse.Error) { log.Debug("ipns: ReadAll Node") r, err := mdag.NewDagReader(s.Nd, s.Ipfs.DAG) if err != nil { return nil, err } // this is a terrible function... 'ReadAll'? // what if i have a 6TB file? GG RAM. b, err := ioutil.ReadAll(r) if err != nil { log.Error("[%s] Readall error: %s", s.name, err) return nil, err } if len(b) > 4 { log.Debug("ReadAll trailing bytes: %v", b[len(b)-4:]) } return b, nil } func (n *Node) Write(req *fuse.WriteRequest, resp *fuse.WriteResponse, intr fs.Intr) fuse.Error { log.Debug("ipns: Node Write [%s]: flags = %s, offset = %d, size = %d", n.name, req.Flags.String(), req.Offset, len(req.Data)) if n.writerBuf == nil { n.writerBuf = NewWriterAtFromBytes(nil) } _, err := n.writerBuf.WriteAt(req.Data, req.Offset) if err != nil { return err } return nil } func (n *Node) Flush(req *fuse.FlushRequest, intr fs.Intr) fuse.Error { log.Debug("Got flush request [%s]!", n.name) if n.writerBuf != nil { //TODO: // This operation holds everything in memory, // should be changed to stream the block creation/storage // but for now, since the buf is all in memory anyways... //NOTE: // This should only occur on a file object, if this were to be a // folder, bad things would happen. buf := bytes.NewReader(n.writerBuf.Bytes()) newNode, err := imp.NewDagFromReader(buf) if err != nil { log.Critical("error creating dag from writerBuf: %s", err) return fuse.ENODATA } if n.parent != nil { err := n.parent.update(n.name, newNode) if err != nil { log.Critical("error in updating ipns dag tree: %s", err) // return fuse.ETHISISPRETTYBAD return fuse.ENOSYS } } n.Nd = newNode //TEMP dr, err := mdag.NewDagReader(n.Nd, n.Ipfs.DAG) if err != nil { log.Critical("Verification read failed.") } b, err := ioutil.ReadAll(dr) if err != nil { log.Critical("Verification read failed.") } fmt.Println("VERIFICATION READ") fmt.Printf("READ %d BYTES\n", len(b)) fmt.Println(string(b)) // n.writerBuf = nil n.wasChanged() } return nil } func (n *Node) wasChanged() { root := n.nsRoot if root == nil { root = n } root.repub.Publish <- struct{}{} } func (n *Node) republishRoot() error { log.Debug("Republish root") var root *Node if n.nsRoot != nil { root = n.nsRoot } else { root = n } // Add any nodes that may be new to the DAG service err := n.Ipfs.DAG.AddRecursive(root.Nd) if err != nil { log.Critical("ipns: Dag Add Error: %s", err) return err } n.writerBuf = nil ndkey, err := root.Nd.Key() if err != nil { log.Error("getKey error: %s", err) // return fuse.ETHISREALLYSUCKS return err } log.Debug("Publishing changes!") err = n.Ipfs.Publisher.Publish(root.key, ndkey.Pretty()) if err != nil { log.Error("ipns: Publish Failed: %s", err) return err } return nil } func (n *Node) Fsync(req *fuse.FsyncRequest, intr fs.Intr) fuse.Error { log.Debug("Got fsync request!") return nil } func (n *Node) Mkdir(req *fuse.MkdirRequest, intr fs.Intr) (fs.Node, fuse.Error) { log.Debug("Got mkdir request!") dagnd := &mdag.Node{Data: mdag.FolderPBData()} nnode := n.Nd.Copy() nnode.AddNodeLink(req.Name, dagnd) child := &Node{ Ipfs: n.Ipfs, Nd: dagnd, name: req.Name, } if n.nsRoot == nil { child.nsRoot = n } else { child.nsRoot = n.nsRoot } if n.parent != nil { err := n.parent.update(n.name, nnode) if err != nil { log.Critical("Error updating node: %s", err) // Can we panic, please? return nil, fuse.ENODATA } } n.Nd = nnode n.wasChanged() return child, nil } func (n *Node) Mknod(req *fuse.MknodRequest, intr fs.Intr) (fs.Node, fuse.Error) { log.Debug("Got mknod request!") return nil, nil } func (n *Node) Open(req *fuse.OpenRequest, resp *fuse.OpenResponse, intr fs.Intr) (fs.Handle, fuse.Error) { //log.Debug("[%s] Received open request! flags = %s", n.name, req.Flags.String()) //TODO: check open flags and truncate if necessary return n, nil } func (n *Node) Create(req *fuse.CreateRequest, resp *fuse.CreateResponse, intr fs.Intr) (fs.Node, fs.Handle, fuse.Error) { log.Debug("Got create request: %s", req.Name) // New 'empty' file nd := &mdag.Node{Data: mdag.FilePBData(nil)} child := n.makeChild(req.Name, nd) nnode := n.Nd.Copy() err := nnode.AddNodeLink(req.Name, nd) if err != nil { log.Error("Error adding child to node: %s", err) return nil, nil, fuse.ENOENT } if n.parent != nil { err := n.parent.update(n.name, nnode) if err != nil { log.Critical("Error updating node: %s", err) // Can we panic, please? return nil, nil, fuse.ENODATA } } n.Nd = nnode n.wasChanged() return child, child, nil } func (n *Node) Remove(req *fuse.RemoveRequest, intr fs.Intr) fuse.Error { log.Debug("[%s] Got Remove request: %s", n.name, req.Name) nnode := n.Nd.Copy() err := nnode.RemoveNodeLink(req.Name) if err != nil { log.Error("Remove: No such file.") return fuse.ENOENT } if n.parent != nil { err := n.parent.update(n.name, nnode) if err != nil { log.Critical("Error updating node: %s", err) // Can we panic, please? return fuse.ENODATA } } n.Nd = nnode n.wasChanged() return nil } func (n *Node) Rename(req *fuse.RenameRequest, newDir fs.Node, intr fs.Intr) fuse.Error { log.Debug("Got Rename request '%s' -> '%s'", req.OldName, req.NewName) var mdn *mdag.Node for _, l := range n.Nd.Links { if l.Name == req.OldName { mdn = l.Node } } if mdn == nil { log.Critical("nil Link found on rename!") return fuse.ENOENT } n.Nd.RemoveNodeLink(req.OldName) switch newDir := newDir.(type) { case *Node: err := newDir.Nd.AddNodeLink(req.NewName, mdn) if err != nil { log.Error("Error adding node to new dir on rename: %s", err) return fuse.ENOENT } default: log.Critical("Unknown node type for rename target dir!") return fuse.ENOENT } return nil } func (n *Node) update(name string, newnode *mdag.Node) error { nnode := n.Nd.Copy() err := nnode.RemoveNodeLink(name) if err != nil { return err } nnode.AddNodeLink(name, newnode) if n.parent != nil { err := n.parent.update(n.name, newnode) if err != nil { return err } } n.Nd = nnode return nil }