-
-
Notifications
You must be signed in to change notification settings - Fork 3k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
add experimental daemon plugin for 9P2000.L support #6612
Changes from 31 commits
2a02a91
ef75786
43a38b6
7a80532
2d99302
75618a5
07534a3
0d28489
7bb358f
6a17332
090f7cd
19d5332
353007e
6e7ea19
28c5cff
81b9a06
864dbfe
35eee21
483845c
3876302
d9fd656
f48f4a8
4bb5712
760d3c6
6e87be8
7ff0d1a
99df410
dc6287d
b8c4a3c
0997454
3ee383a
15e894a
e5ec672
c653fc5
b6d1506
fbf5bca
888c33d
3017586
6507b34
4a471d6
afd9cb1
e8c1a5e
a2a9e8b
0a6110d
1dd3125
7b10b5f
82780a1
4c9a790
fb20a44
5ad516c
02f69b8
eaff0c9
3ab6029
b6dee76
770bbce
4717f73
21b2d2c
62560e4
11c5add
dd3a894
1636e1b
63f6d27
97f36a9
3c56f5f
4c1ad5d
e3db71a
b2a43b8
9cc06ae
e79e371
d1a1170
be83005
fbac898
62cf47e
e8ccc6b
6d92612
294c3ea
8ce5e51
8328485
db0bcba
7bb94a4
e029c7d
8cee086
2a495c7
161d86a
9ef8e6a
545d666
9024b95
a0465f4
2aead06
ec9f8ea
c744a00
65b677f
7ce0e9b
4c6b8ee
6e654db
3501592
07db2b9
984f155
0fa5983
c9faaa7
c7cfd71
c294ac8
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,12 @@ | ||
/* | ||
Package filesystem is an experimental package that implements the go-ipfs daemon plugin interface | ||
and defines the plugin's config structure. The plugin itself exposes file system services over a multiaddr listener. | ||
|
||
By default, we try to expose the IPFS namespace using the 9P2000.L protocol, over a unix domain socket | ||
(located at $IPFS_PATH/filesystem.9P.sock) | ||
|
||
To set the multiaddr listen address, you may use the environment variable $IPFS_FS_ADDR, or set the option in the node's config file | ||
via `ipfs config --json 'Plugins.Plugins.filesystem.Config "Config":{"Service":{"9P":"/ip4/127.0.0.1/tcp/564"}}'` | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Calling this returns There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Not yet. Really, we should change how we do config type checking (see https://github.com/ipfs/go-ipfs/blob/7e7b76259f655c5aa321e622a38619587a5a02a8/repo/fsrepo/fsrepo.go#L589). Basically, instead of manually type checking like that, we should just set the key and see if it serializes correctly. The tricky part will be making sure the error doesn't suck. |
||
To disable this plugin entirely, use: `ipfs config --json Plugins.Plugins.filesystem.Disabled true` | ||
*/ | ||
package filesystem |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,121 @@ | ||
package filesystem | ||
|
||
import ( | ||
"context" | ||
"encoding/json" | ||
"os" | ||
"path/filepath" | ||
"runtime" | ||
|
||
"github.com/djdv/p9/p9" | ||
plugin "github.com/ipfs/go-ipfs/plugin" | ||
fsnodes "github.com/ipfs/go-ipfs/plugin/plugins/filesystem/nodes" | ||
logging "github.com/ipfs/go-log" | ||
coreiface "github.com/ipfs/interface-go-ipfs-core" | ||
"github.com/multiformats/go-multiaddr" | ||
manet "github.com/multiformats/go-multiaddr-net" | ||
) | ||
|
||
var ( | ||
_ plugin.PluginDaemon = (*FileSystemPlugin)(nil) // impl check | ||
|
||
// Plugins is an exported list of plugins that will be loaded by go-ipfs. | ||
Plugins = []plugin.Plugin{ | ||
&FileSystemPlugin{}, //TODO: individually name implementations: &P9{} | ||
} | ||
|
||
logger logging.EventLogger | ||
) | ||
|
||
func init() { | ||
logger = logging.Logger("plugin/filesystem") | ||
} | ||
|
||
type FileSystemPlugin struct { | ||
ctx context.Context | ||
cancel context.CancelFunc | ||
|
||
addr multiaddr.Multiaddr | ||
listener manet.Listener | ||
errorChan chan error | ||
} | ||
|
||
func (*FileSystemPlugin) Name() string { | ||
return PluginName | ||
} | ||
|
||
func (*FileSystemPlugin) Version() string { | ||
return PluginVersion | ||
} | ||
|
||
func (fs *FileSystemPlugin) Init(env *plugin.Environment) error { | ||
djdv marked this conversation as resolved.
Show resolved
Hide resolved
|
||
logger.Info("Initializing 9P resource server...") | ||
if !filepath.IsAbs(env.Repo) { | ||
absRepo, err := filepath.Abs(env.Repo) | ||
if err != nil { | ||
return err | ||
} | ||
env.Repo = absRepo | ||
} | ||
|
||
cfg := &Config{} | ||
if env.Config != nil { | ||
byteRep, err := json.Marshal(env.Config) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Hm. Maybe we should pass in a There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Are you saying That probably makes the most sense. I'm trying to think of a situation where you'd need raw bytes in the plugin config, that can't be fit inside formatted json, but nothing comes to immediate mind that wouldn't be better stored elsewhere. So long as we discard the object once we've loaded and parsed, I'd imagine things wouldn't use any more resources than the current implementation. But I'm saying that without having doing any actual comparisons. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Alternatively, I'm assuming you could have meant There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I went forward with the second one for now but this isn't tested yet. This #6612 (comment) likely needs to be resolved, and I'd like to see some method for Plugins to initiate a |
||
if err != nil { | ||
return err | ||
} | ||
if err = json.Unmarshal(byteRep, cfg); err != nil { | ||
return err | ||
} | ||
} else { | ||
cfg = defaultConfig(env.Repo) | ||
} | ||
|
||
var err error | ||
if envAddr := os.ExpandEnv(EnvAddr); envAddr == "" { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. So this doesn't get lost, I'm still not a fan of using an environment variable like this. #6612 (comment) There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Resolved around here: 21b2d2c#diff-1054eebd17f3591011a5e6aaa3b3bbdfR31 There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
Do the above changes seem better? I initially assumed "this" meant how the expansion was being processed. But I'm wondering now if you meant using env vars to override config file values, or maybe something else. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. @Stebalien ping on this comment ^ |
||
fs.addr, err = multiaddr.NewMultiaddr(cfg.Service[defaultService]) | ||
} else { | ||
fs.addr, err = multiaddr.NewMultiaddr(envAddr) | ||
} | ||
if err != nil { | ||
return err | ||
} | ||
|
||
//TODO [manet]: unix sockets are not removed on process death (on Windows) | ||
// so for now we just try to remove it before listening on it | ||
if runtime.GOOS == "windows" { | ||
removeUnixSockets(fs.addr) | ||
} | ||
|
||
fs.ctx, fs.cancel = context.WithCancel(context.Background()) | ||
fs.errorChan = make(chan error) | ||
djdv marked this conversation as resolved.
Show resolved
Hide resolved
|
||
|
||
logger.Info("9P resource server okay for launch") | ||
return nil | ||
} | ||
|
||
func (fs *FileSystemPlugin) Start(core coreiface.CoreAPI) error { | ||
logger.Info("Starting 9P resource server...") | ||
|
||
var err error | ||
if fs.listener, err = manet.Listen(fs.addr); err != nil { | ||
djdv marked this conversation as resolved.
Show resolved
Hide resolved
|
||
logger.Errorf("9P listen error: %s\n", err) | ||
return err | ||
} | ||
|
||
// construct and run the 9P resource server | ||
s := p9.NewServer(fsnodes.RootAttacher(fs.ctx, core)) | ||
go func() { | ||
fs.errorChan <- s.Serve(manet.NetListener(fs.listener)) | ||
}() | ||
|
||
logger.Infof("9P service is listening on %s\n", fs.listener.Addr()) | ||
return nil | ||
} | ||
|
||
func (fs *FileSystemPlugin) Close() error { | ||
Stebalien marked this conversation as resolved.
Show resolved
Hide resolved
|
||
logger.Info("9P server requested to close") | ||
fs.cancel() | ||
fs.listener.Close() | ||
djdv marked this conversation as resolved.
Show resolved
Hide resolved
|
||
return <-fs.errorChan | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. nit: this should probably be idempotent over multiple close calls (instead of just hanging forever). The usual way to do this is to set a variable and then close a "closed" chan. To read the error, wait for the closed chan to close then read the error from the variable (no lock necessary). There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Maybe, I don't know how we're actually using this. If it's not idempotent, we should document that. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. You're right. However, I also had to add in a bool to signify close intent. I'm wondering if there's a better way to handle this 8cee086#diff-8d243ab8514fa1b77d41df37a81641c9R146 specifically. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I usually use the flag (bool) approach. However, it needs to be atomic. What if we:
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. |
||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
s/9p/9P/
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm with you on this, but the implementation wants it this way. (at least on my machine)
https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git/plain/Documentation/filesystems/9p.txt
Also hello 👋!!
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Sorry I meant the
filesystem.9p.sock
should befilesystem.9P.sock
. Hello! 👋There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ahh, nice catch. I'm probably going to go in the opposite direction, anywhere I have a 'P' is going to be lowered to 'p' to be consistent with the mount argument.
The exceptions may be in docs referencing the protocol, and maybe in the logger.