// +build linux darwin freebsd package commands import ( "fmt" "time" cmds "github.com/jbenet/go-ipfs/commands" "github.com/jbenet/go-ipfs/config" core "github.com/jbenet/go-ipfs/core" ipns "github.com/jbenet/go-ipfs/fuse/ipns" rofs "github.com/jbenet/go-ipfs/fuse/readonly" ) // amount of time to wait for mount errors const mountTimeout = time.Second var mountCmd = &cmds.Command{ Arguments: []cmds.Argument{ cmds.Argument{Name: "os-path", Type: cmds.ArgString, Required: false, Variadic: false}, }, Options: []cmds.Option{ cmds.Option{[]string{"f"}, cmds.String}, cmds.Option{[]string{"n"}, cmds.String}, }, Help: `ipfs mount - Mount an ipfs read-only mountpoint. Mount ipfs at a read-only mountpoint on the OS. All ipfs objects will be accessible under that directory. Note that the root will not be listable, as it is virtual. Accessing known paths directly. `, Run: func(res cmds.Response, req cmds.Request) { ctx := req.Context() // error if we aren't running node in online mode if ctx.Node.Network == nil { res.SetError(errNotOnline, cmds.ErrNormal) return } if err := platformFuseChecks(); err != nil { res.SetError(err, cmds.ErrNormal) return } // update fsdir with flag. fsdir := ctx.Config.Mounts.IPFS opt, _ := req.Option("f") if val, ok := opt.(string); ok && val != "" { fsdir = val } fsdone := mountIpfs(ctx.Node, fsdir) // get default mount points nsdir := ctx.Config.Mounts.IPNS opt, _ = req.Option("f") if val, ok := opt.(string); ok && val != "" { nsdir = val } nsdone := mountIpns(ctx.Node, nsdir, fsdir) // wait until mounts return an error (or timeout if successful) var err error select { case err = <-fsdone: case err = <-nsdone: // mounted successfully, we timed out with no errors case <-time.After(mountTimeout): output := ctx.Config.Mounts res.SetOutput(&output) return } res.SetError(err, cmds.ErrNormal) }, Type: &config.Mounts{}, Marshallers: map[cmds.EncodingType]cmds.Marshaller{ cmds.Text: func(res cmds.Response) ([]byte, error) { v := res.Output().(*config.Mounts) s := fmt.Sprintf("IPFS mounted at: %s\n", v.IPFS) s += fmt.Sprintf("IPNS mounted at: %s\n", v.IPNS) return []byte(s), nil }, }, } func mountIpfs(node *core.IpfsNode, fsdir string) <-chan error { done := make(chan error) log.Info("Mounting IPFS at ", fsdir) go func() { err := rofs.Mount(node, fsdir) done <- err close(done) }() return done } func mountIpns(node *core.IpfsNode, nsdir, fsdir string) <-chan error { if nsdir == "" { return nil } done := make(chan error) log.Info("Mounting IPNS at ", nsdir) go func() { err := ipns.Mount(node, nsdir, fsdir) done <- err close(done) }() return done } var platformFuseChecks = func() error { return nil }