mirror of
https://github.com/neovim/neovim.git
synced 2025-02-25 18:55:25 -06:00
Merge pull request #17691 from bfredl/serverconnect
feat(ui): connect to remote ui (beginning of ui client)
This commit is contained in:
commit
36ca585d2f
@ -30,6 +30,7 @@
|
|||||||
#include "nvim/api/vimscript.h"
|
#include "nvim/api/vimscript.h"
|
||||||
#include "nvim/api/win_config.h"
|
#include "nvim/api/win_config.h"
|
||||||
#include "nvim/api/window.h"
|
#include "nvim/api/window.h"
|
||||||
|
#include "nvim/ui_client.h"
|
||||||
|
|
||||||
static Map(String, MsgpackRpcRequestHandler) methods = MAP_INIT;
|
static Map(String, MsgpackRpcRequestHandler) methods = MAP_INIT;
|
||||||
|
|
||||||
@ -38,6 +39,13 @@ static void msgpack_rpc_add_method_handler(String method, MsgpackRpcRequestHandl
|
|||||||
map_put(String, MsgpackRpcRequestHandler)(&methods, method, handler);
|
map_put(String, MsgpackRpcRequestHandler)(&methods, method, handler);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void msgpack_rpc_add_redraw(void)
|
||||||
|
{
|
||||||
|
msgpack_rpc_add_method_handler(STATIC_CSTR_AS_STRING("redraw"),
|
||||||
|
(MsgpackRpcRequestHandler) { .fn = ui_client_handle_redraw,
|
||||||
|
.fast = true });
|
||||||
|
}
|
||||||
|
|
||||||
/// @param name API method name
|
/// @param name API method name
|
||||||
/// @param name_len name size (includes terminating NUL)
|
/// @param name_len name size (includes terminating NUL)
|
||||||
MsgpackRpcRequestHandler msgpack_rpc_get_handler_for(const char *name, size_t name_len,
|
MsgpackRpcRequestHandler msgpack_rpc_get_handler_for(const char *name, size_t name_len,
|
||||||
|
@ -1997,9 +1997,8 @@ Array nvim_get_proc_children(Integer pid, Error *err)
|
|||||||
DLOG("fallback to vim._os_proc_children()");
|
DLOG("fallback to vim._os_proc_children()");
|
||||||
Array a = ARRAY_DICT_INIT;
|
Array a = ARRAY_DICT_INIT;
|
||||||
ADD(a, INTEGER_OBJ(pid));
|
ADD(a, INTEGER_OBJ(pid));
|
||||||
String s = cstr_to_string("return vim._os_proc_children(...)");
|
String s = STATIC_CSTR_AS_STRING("return vim._os_proc_children(...)");
|
||||||
Object o = nlua_exec(s, a, err);
|
Object o = nlua_exec(s, a, err);
|
||||||
api_free_string(s);
|
|
||||||
api_free_array(a);
|
api_free_array(a);
|
||||||
if (o.type == kObjectTypeArray) {
|
if (o.type == kObjectTypeArray) {
|
||||||
rvobj = o.data.array;
|
rvobj = o.data.array;
|
||||||
|
@ -342,6 +342,9 @@ EXTERN sctx_T current_sctx INIT(= { 0 COMMA 0 COMMA 0 });
|
|||||||
// ID of the current channel making a client API call
|
// ID of the current channel making a client API call
|
||||||
EXTERN uint64_t current_channel_id INIT(= 0);
|
EXTERN uint64_t current_channel_id INIT(= 0);
|
||||||
|
|
||||||
|
// ID of the client channel. Used by ui client
|
||||||
|
EXTERN uint64_t ui_client_channel_id INIT(= 0);
|
||||||
|
|
||||||
EXTERN bool did_source_packages INIT(= false);
|
EXTERN bool did_source_packages INIT(= false);
|
||||||
|
|
||||||
// Scope information for the code that indirectly triggered the current
|
// Scope information for the code that indirectly triggered the current
|
||||||
|
@ -29,6 +29,7 @@
|
|||||||
#include "nvim/if_cscope.h"
|
#include "nvim/if_cscope.h"
|
||||||
#include "nvim/lua/executor.h"
|
#include "nvim/lua/executor.h"
|
||||||
#include "nvim/main.h"
|
#include "nvim/main.h"
|
||||||
|
#include "nvim/ui_client.h"
|
||||||
#include "nvim/vim.h"
|
#include "nvim/vim.h"
|
||||||
#ifdef HAVE_LOCALE_H
|
#ifdef HAVE_LOCALE_H
|
||||||
# include <locale.h>
|
# include <locale.h>
|
||||||
@ -269,8 +270,7 @@ int main(int argc, char **argv)
|
|||||||
|
|
||||||
server_init(params.listen_addr);
|
server_init(params.listen_addr);
|
||||||
if (params.remote) {
|
if (params.remote) {
|
||||||
handle_remote_client(¶ms, params.remote,
|
remote_request(¶ms, params.remote, params.server_addr, argc, argv);
|
||||||
params.server_addr, argc, argv);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (GARGCOUNT > 0) {
|
if (GARGCOUNT > 0) {
|
||||||
@ -807,31 +807,54 @@ static void init_locale(void)
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/// Handle remote subcommands
|
|
||||||
static void handle_remote_client(mparm_T *params, int remote_args,
|
static uint64_t server_connect(char *server_addr, const char **errmsg)
|
||||||
char *server_addr, int argc, char **argv)
|
|
||||||
{
|
{
|
||||||
Object rvobj = OBJECT_INIT;
|
if (server_addr == NULL) {
|
||||||
rvobj.data.dictionary = (Dictionary)ARRAY_DICT_INIT;
|
*errmsg = "no address specified";
|
||||||
rvobj.type = kObjectTypeDictionary;
|
return 0;
|
||||||
|
}
|
||||||
CallbackReader on_data = CALLBACK_READER_INIT;
|
CallbackReader on_data = CALLBACK_READER_INIT;
|
||||||
const char *connect_error = NULL;
|
const char *error = NULL;
|
||||||
uint64_t rc_id = 0;
|
bool is_tcp = strrchr(server_addr, ':') ? true : false;
|
||||||
if (server_addr != NULL) {
|
// connected to channel
|
||||||
rc_id = channel_connect(false, server_addr, true, on_data, 50, &connect_error);
|
uint64_t chan = channel_connect(is_tcp, server_addr, true, on_data, 50, &error);
|
||||||
|
if (error) {
|
||||||
|
*errmsg = error;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
return chan;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Handle remote subcommands
|
||||||
|
static void remote_request(mparm_T *params, int remote_args,
|
||||||
|
char *server_addr, int argc, char **argv)
|
||||||
|
{
|
||||||
|
const char *connect_error = NULL;
|
||||||
|
uint64_t chan = server_connect(server_addr, &connect_error);
|
||||||
|
Object rvobj = OBJECT_INIT;
|
||||||
|
|
||||||
|
if (strequal(argv[remote_args], "--remote-ui-test")) {
|
||||||
|
if (!chan) {
|
||||||
|
emsg(connect_error);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
ui_client_init(chan);
|
||||||
|
ui_client_execute(chan);
|
||||||
|
abort(); // unreachable
|
||||||
}
|
}
|
||||||
|
|
||||||
int t_argc = remote_args;
|
|
||||||
Array args = ARRAY_DICT_INIT;
|
Array args = ARRAY_DICT_INIT;
|
||||||
String arg_s;
|
String arg_s;
|
||||||
for (; t_argc < argc; t_argc++) {
|
for (int t_argc = remote_args; t_argc < argc; t_argc++) {
|
||||||
arg_s = cstr_to_string(argv[t_argc]);
|
arg_s = cstr_to_string(argv[t_argc]);
|
||||||
ADD(args, STRING_OBJ(arg_s));
|
ADD(args, STRING_OBJ(arg_s));
|
||||||
}
|
}
|
||||||
|
|
||||||
Error err = ERROR_INIT;
|
Error err = ERROR_INIT;
|
||||||
Array a = ARRAY_DICT_INIT;
|
Array a = ARRAY_DICT_INIT;
|
||||||
ADD(a, INTEGER_OBJ((int)rc_id));
|
ADD(a, INTEGER_OBJ((int)chan));
|
||||||
ADD(a, CSTR_TO_OBJ(server_addr));
|
ADD(a, CSTR_TO_OBJ(server_addr));
|
||||||
ADD(a, CSTR_TO_OBJ(connect_error));
|
ADD(a, CSTR_TO_OBJ(connect_error));
|
||||||
ADD(a, ARRAY_OBJ(args));
|
ADD(a, ARRAY_OBJ(args));
|
||||||
@ -892,6 +915,7 @@ static void handle_remote_client(mparm_T *params, int remote_args,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/// Decides whether text (as opposed to commands) will be read from stdin.
|
/// Decides whether text (as opposed to commands) will be read from stdin.
|
||||||
/// @see EDIT_STDIN
|
/// @see EDIT_STDIN
|
||||||
static bool edit_stdin(bool explicit, mparm_T *parmp)
|
static bool edit_stdin(bool explicit, mparm_T *parmp)
|
||||||
|
@ -547,6 +547,11 @@ void rpc_close(Channel *channel)
|
|||||||
channel->rpc.closed = true;
|
channel->rpc.closed = true;
|
||||||
channel_decref(channel);
|
channel_decref(channel);
|
||||||
|
|
||||||
|
if (channel->id == ui_client_channel_id) {
|
||||||
|
// TODO(bfredl): handle this in ui_client, where os_exit() is safe
|
||||||
|
exit(0);
|
||||||
|
}
|
||||||
|
|
||||||
if (channel->streamtype == kChannelStreamStdio) {
|
if (channel->streamtype == kChannelStreamStdio) {
|
||||||
multiqueue_put(main_loop.fast_events, exit_event, 0);
|
multiqueue_put(main_loop.fast_events, exit_event, 0);
|
||||||
}
|
}
|
||||||
|
70
src/nvim/ui_client.c
Normal file
70
src/nvim/ui_client.c
Normal file
@ -0,0 +1,70 @@
|
|||||||
|
// This is an open source non-commercial project. Dear PVS-Studio, please check
|
||||||
|
// it. PVS-Studio Static Code Analyzer for C, C++ and C#: http://www.viva64.com
|
||||||
|
|
||||||
|
#include <stdbool.h>
|
||||||
|
#include <stdint.h>
|
||||||
|
#include <assert.h>
|
||||||
|
|
||||||
|
#include "nvim/vim.h"
|
||||||
|
#include "nvim/ui_client.h"
|
||||||
|
#include "nvim/api/private/helpers.h"
|
||||||
|
#include "nvim/msgpack_rpc/channel.h"
|
||||||
|
#include "nvim/api/private/dispatch.h"
|
||||||
|
#include "nvim/ui.h"
|
||||||
|
|
||||||
|
void ui_client_init(uint64_t chan)
|
||||||
|
{
|
||||||
|
Array args = ARRAY_DICT_INIT;
|
||||||
|
int width = 80;
|
||||||
|
int height = 25;
|
||||||
|
Dictionary opts = ARRAY_DICT_INIT;
|
||||||
|
|
||||||
|
PUT(opts, "rgb", BOOLEAN_OBJ(true));
|
||||||
|
PUT(opts, "ext_linegrid", BOOLEAN_OBJ(true));
|
||||||
|
PUT(opts, "ext_termcolors", BOOLEAN_OBJ(true));
|
||||||
|
|
||||||
|
// TODO(bfredl): use the size of the client UI
|
||||||
|
ADD(args, INTEGER_OBJ((int)width));
|
||||||
|
ADD(args, INTEGER_OBJ((int)height));
|
||||||
|
ADD(args, DICTIONARY_OBJ(opts));
|
||||||
|
|
||||||
|
rpc_send_event(chan, "nvim_ui_attach", args);
|
||||||
|
msgpack_rpc_add_redraw(); // GAME!
|
||||||
|
ui_client_channel_id = chan;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Handler for "redraw" events sent by the NVIM server
|
||||||
|
///
|
||||||
|
/// This is just a stub. The mentioned functionality will be implemented.
|
||||||
|
///
|
||||||
|
/// This function will be called by handle_request (in msgpack_rpc/channle.c)
|
||||||
|
/// The individual ui_events sent by the server are individually handled
|
||||||
|
/// by their respective handlers defined in ui_events_redraw.generated.h
|
||||||
|
///
|
||||||
|
/// @note The "flush" event is called only once and only after handling all
|
||||||
|
/// the other events
|
||||||
|
/// @param channel_id: The id of the rpc channel
|
||||||
|
/// @param uidata: The dense array containing the ui_events sent by the server
|
||||||
|
/// @param[out] err Error details, if any
|
||||||
|
Object ui_client_handle_redraw(uint64_t channel_id, Array args, Error *error)
|
||||||
|
{
|
||||||
|
for (size_t i = 0; i < args.size; i++) {
|
||||||
|
Array call = args.items[i].data.array;
|
||||||
|
char *method_name = call.items[0].data.string.data;
|
||||||
|
|
||||||
|
fprintf(stderr, "%s: %zu\n", method_name, call.size-1);
|
||||||
|
}
|
||||||
|
return NIL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// run the main thread in ui client mode
|
||||||
|
///
|
||||||
|
/// This is just a stub. the full version will handle input, resizing, etc
|
||||||
|
void ui_client_execute(uint64_t chan)
|
||||||
|
{
|
||||||
|
while (true) {
|
||||||
|
loop_poll_events(&main_loop, -1);
|
||||||
|
}
|
||||||
|
|
||||||
|
getout(0);
|
||||||
|
}
|
9
src/nvim/ui_client.h
Normal file
9
src/nvim/ui_client.h
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
#ifndef NVIM_UI_CLIENT_H
|
||||||
|
#define NVIM_UI_CLIENT_H
|
||||||
|
|
||||||
|
#include "nvim/api/private/defs.h"
|
||||||
|
|
||||||
|
#ifdef INCLUDE_GENERATED_DECLARATIONS
|
||||||
|
#include "ui_client.h.generated.h"
|
||||||
|
#endif
|
||||||
|
#endif // NVIM_UI_CLIENT_H
|
Loading…
Reference in New Issue
Block a user