Skip to content

Commit 7e7b687

Browse files
authored
Add basic remote module support (#107)
See: neovim/neovim#27949 This includes a helper method for defining remote modules as well as an acceptance spec to demonstrate their usage. I chose to implement a new DSL class just for remote modules because the existing plugin DSL is far too complicated for simple RPC handling. As remote plugins are phased out, I expect to phase out and eventually deprecate the existing plugin DSL.
1 parent a6d93c4 commit 7e7b687

File tree

7 files changed

+139
-27
lines changed

7 files changed

+139
-27
lines changed

README.md

+11-27
Original file line numberDiff line numberDiff line change
@@ -42,37 +42,21 @@ client = Neovim.attach_unix("/tmp/nvim.sock")
4242

4343
Refer to the [`Neovim` docs](https://www.rubydoc.info/github/neovim/neovim-ruby/main/Neovim) for other ways to connect to `nvim`, and the [`Neovim::Client` docs](https://www.rubydoc.info/github/neovim/neovim-ruby/main/Neovim/Client) for a summary of the client interface.
4444

45-
### Plugins
45+
### Remote Modules
4646

47-
Plugins are Ruby files loaded from the `$VIMRUNTIME/rplugin/ruby/` directory. Here's an example plugin:
47+
Remote modules allow users to define custom handlers in Ruby. To implement a remote module:
4848

49-
```ruby
50-
# ~/.config/nvim/rplugin/ruby/example_plugin.rb
51-
52-
Neovim.plugin do |plug|
53-
# Define a command called "SetLine" which sets the contents of the current
54-
# line. This command is executed asynchronously, so the return value is
55-
# ignored.
56-
plug.command(:SetLine, nargs: 1) do |nvim, str|
57-
nvim.current.line = str
58-
end
59-
60-
# Define a function called "Sum" which adds two numbers. This function is
61-
# executed synchronously, so the result of the block will be returned to nvim.
62-
plug.function(:Sum, nargs: 2, sync: true) do |nvim, x, y|
63-
x + y
64-
end
65-
66-
# Define an autocmd for the BufEnter event on Ruby files.
67-
plug.autocmd(:BufEnter, pattern: "*.rb") do |nvim|
68-
nvim.command("echom 'Ruby file, eh?'")
69-
end
70-
end
71-
```
49+
- Define your handlers in a plain Ruby script that imports `neovim`
50+
- Spawn the script from lua using `jobstart`
51+
- Define commands in lua using `nvim_create_user_command` that route to the job's channel ID
52+
53+
For usage examples, see:
7254

73-
When you add or update a plugin, you will need to call `:UpdateRemotePlugins` to update the remote plugin manifest. See `:help remote-plugin-manifest` for more information.
55+
- [`example_remote_module.rb`](spec/acceptance/runtime/example_remote_module.rb)
56+
- [`example_remote_module.lua`](spec/acceptance/runtime/plugin/example_remote_module.lua)
57+
- [`remote_module_spec.vim`](spec/acceptance/remote_module_spec.vim)
7458

75-
Refer to the [`Neovim::Plugin::DSL` docs](https://www.rubydoc.info/github/neovim/neovim-ruby/main/Neovim/Plugin/DSL) for a more complete overview of the `Neovim.plugin` DSL.
59+
*Note*: Remote modules are a replacement for the deprecated "remote plugin" architecture. See https://github.com/neovim/neovim/issues/27949 for details.
7660

7761
### Vim Plugin Support
7862

lib/neovim.rb

+9
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@
44
require "neovim/event_loop"
55
require "neovim/executable"
66
require "neovim/logging"
7+
require "neovim/remote_module"
78
require "neovim/version"
89

910
# The main entrypoint to the +Neovim+ gem. It allows you to connect to a
@@ -83,6 +84,14 @@ def self.attach_child(argv=[executable.path])
8384
attach(EventLoop.child(argv))
8485
end
8586

87+
# Start a remote module process with handlers defined in the config block.
88+
# Blocks indefinitely to handle messages.
89+
#
90+
# @see RemoteModule::DSL
91+
def self.start_remote(&block)
92+
RemoteModule.from_config_block(&block).start
93+
end
94+
8695
# Placeholder method for exposing the remote plugin DSL. This gets
8796
# temporarily overwritten in +Host::Loader#load+.
8897
#

lib/neovim/remote_module.rb

+42
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,42 @@
1+
require "neovim/client"
2+
require "neovim/event_loop"
3+
require "neovim/logging"
4+
require "neovim/remote_module/dsl"
5+
require "neovim/session"
6+
7+
module Neovim
8+
class RemoteModule
9+
include Logging
10+
11+
def self.from_config_block(&block)
12+
new(DSL::new(&block).handlers)
13+
end
14+
15+
def initialize(handlers)
16+
@handlers = handlers
17+
end
18+
19+
def start
20+
event_loop = EventLoop.stdio
21+
session = Session.new(event_loop)
22+
client = nil
23+
24+
session.run do |message|
25+
case message
26+
when Message::Request
27+
begin
28+
client ||= Client.from_event_loop(event_loop, session)
29+
args = message.arguments.flatten(1)
30+
31+
@handlers[message.method_name].call(client, *args).tap do |rv|
32+
session.respond(message.id, rv, nil) if message.sync?
33+
end
34+
rescue => e
35+
log_exception(:error, e, __method__)
36+
session.respond(message.id, nil, e.message) if message.sync?
37+
end
38+
end
39+
end
40+
end
41+
end
42+
end

lib/neovim/remote_module/dsl.rb

+30
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,30 @@
1+
module Neovim
2+
class RemoteModule
3+
# The DSL exposed in +Neovim.start_remote+ blocks.
4+
#
5+
# @api public
6+
class DSL < BasicObject
7+
attr_reader :handlers
8+
9+
def initialize(&block)
10+
@handlers = ::Hash.new do |h, name|
11+
h[name] = ::Proc.new do |_, *|
12+
raise NotImplementedError, "undefined handler #{name.inspect}"
13+
end
14+
end
15+
16+
block&.call(self)
17+
end
18+
19+
# Define an RPC handler for use in remote modules.
20+
#
21+
# @param name [String] The handler name.
22+
# @param block [Proc] The body of the handler.
23+
def register_handler(name, &block)
24+
@handlers[name.to_s] = ::Proc.new do |client, *args|
25+
block.call(client, *args)
26+
end
27+
end
28+
end
29+
end
30+
end
+13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
let s:suite = themis#suite("Remote module")
2+
let s:expect = themis#helper("expect")
3+
4+
call themis#helper('command').with(s:)
5+
6+
function! s:suite.defines_commands() abort
7+
RbSetVar set_from_rb_mod foobar
8+
call s:expect(g:set_from_rb_mod).to_equal('foobar')
9+
endfunction
10+
11+
function! s:suite.propagates_errors() abort
12+
Throws /oops/ :RbWillRaise
13+
endfunction
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
require "neovim"
2+
3+
Neovim.start_remote do |mod|
4+
mod.register_handler("rb_set_var") do |nvim, name, val|
5+
nvim.set_var(name, val.to_s)
6+
end
7+
8+
mod.register_handler("rb_will_raise") do |nvim|
9+
raise "oops"
10+
end
11+
end
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
local chan
2+
3+
local function ensure_job()
4+
if chan then
5+
return chan
6+
end
7+
8+
chan = vim.fn.jobstart({
9+
'ruby',
10+
'-I', 'lib',
11+
'spec/acceptance/runtime/example_remote_module.rb',
12+
}, { rpc = true })
13+
14+
return chan
15+
end
16+
17+
vim.api.nvim_create_user_command('RbSetVar', function(args)
18+
vim.fn.rpcrequest(ensure_job(), 'rb_set_var', args.fargs)
19+
end, { nargs = '*' })
20+
21+
vim.api.nvim_create_user_command('RbWillRaise', function(args)
22+
vim.fn.rpcrequest(ensure_job(), 'rb_will_raise', args.fargs)
23+
end, { nargs = 0 })

0 commit comments

Comments
 (0)