aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorBjörn Linse <bjorn.linse@gmail.com>2019-09-01 23:27:18 +0200
committerGitHub <noreply@github.com>2019-09-01 23:27:18 +0200
commit801fe799ff359cdae66ed14f5e0b7c2c4cec9536 (patch)
tree19ab5a052ba30de80692e46b13af067cb68ff749
parent9df3a676e7f2302929061a6c0edf42e5fb3f3873 (diff)
parentb447bdb68ceda05b3181bec8572ff22ec588ee0d (diff)
downloadrneovim-801fe799ff359cdae66ed14f5e0b7c2c4cec9536.tar.gz
rneovim-801fe799ff359cdae66ed14f5e0b7c2c4cec9536.tar.bz2
rneovim-801fe799ff359cdae66ed14f5e0b7c2c4cec9536.zip
Merge pull request #10382 from abdelhakeem/waituntil
eval: add wait()
-rw-r--r--runtime/doc/eval.txt21
-rw-r--r--runtime/doc/usr_41.txt1
-rw-r--r--src/nvim/eval.c77
-rw-r--r--src/nvim/eval.lua1
-rw-r--r--test/functional/eval/wait_spec.lua78
5 files changed, 178 insertions, 0 deletions
diff --git a/runtime/doc/eval.txt b/runtime/doc/eval.txt
index 897b5df072..f45f99ede2 100644
--- a/runtime/doc/eval.txt
+++ b/runtime/doc/eval.txt
@@ -2427,6 +2427,8 @@ uniq({list} [, {func} [, {dict}]])
values({dict}) List values in {dict}
virtcol({expr}) Number screen column of cursor or mark
visualmode([expr]) String last visual mode used
+wait({timeout}, {condition}[, {interval}])
+ Number Wait until {condition} is satisfied
wildmenumode() Number whether 'wildmenu' mode is active
win_findbuf({bufnr}) List find windows containing {bufnr}
win_getid([{win} [, {tab}]]) Number get |window-ID| for {win} in {tab}
@@ -8858,6 +8860,25 @@ visualmode([expr]) *visualmode()*
a non-empty String, then the Visual mode will be cleared and
the old value is returned. See |non-zero-arg|.
+wait({timeout}, {condition}[, {interval}]) *wait()*
+ Wait until {condition} is satisfied, where {condition} is a
+ |Funcref| or a |string| containing an expression.
+
+ {timeout} is the maximum number of milliseconds to wait,
+ -1 means forever.
+
+ By default, the condition is evaluated on user and internal
+ events. If {interval} is given, the condition is evaluated
+ every {interval} milliseconds in addition. This can be useful
+ to guarantee that the function returns when the condition is
+ satisfied even if the editor is idle.
+
+ Returns one of the following:
+ * 0 if the condition was satisfied before the timeout
+ * -1 if the timeout was exceeded
+ * -2 if the function was interrupted
+ * -3 if an error occurred
+
wildmenumode() *wildmenumode()*
Returns |TRUE| when the wildmenu is active and |FALSE|
otherwise. See 'wildmenu' and 'wildmode'.
diff --git a/runtime/doc/usr_41.txt b/runtime/doc/usr_41.txt
index b26b7cb646..2a2885b4d3 100644
--- a/runtime/doc/usr_41.txt
+++ b/runtime/doc/usr_41.txt
@@ -959,6 +959,7 @@ Timers: *timer-functions*
timer_stop() stop a timer
timer_stopall() stop all timers
timer_info() get information about timers
+ wait() wait for a condition
Tags: *tag-functions*
taglist() get list of matching tags
diff --git a/src/nvim/eval.c b/src/nvim/eval.c
index 488790970e..79e67f6d7b 100644
--- a/src/nvim/eval.c
+++ b/src/nvim/eval.c
@@ -10892,6 +10892,83 @@ static void f_getwininfo(typval_T *argvars, typval_T *rettv, FunPtr fptr)
}
}
+// Dummy timer callback. Used by f_wait().
+static void dummy_timer_due_cb(TimeWatcher *tw, void *data)
+{
+}
+
+// Dummy timer close callback. Used by f_wait().
+static void dummy_timer_close_cb(TimeWatcher *tw, void *data)
+{
+ xfree(tw);
+}
+
+/// "wait(timeout, condition[, interval])" function
+static void f_wait(typval_T *argvars, typval_T *rettv, FunPtr fptr)
+{
+ rettv->v_type = VAR_NUMBER;
+ rettv->vval.v_number = -1;
+
+ if (argvars[0].v_type != VAR_NUMBER) {
+ EMSG2(_(e_invargval), "1");
+ return;
+ }
+
+ int timeout = argvars[0].vval.v_number;
+ typval_T expr = argvars[1];
+
+ int interval = -1;
+ typval_T *tv_interval = &argvars[2];
+
+ TimeWatcher *tw = NULL;
+
+ if (tv_interval->v_type == VAR_NUMBER) {
+ interval = tv_interval->vval.v_number;
+ if (interval <= 0) {
+ EMSG2(_(e_invargval), "3");
+ return;
+ }
+ // Start dummy timer
+ tw = xmalloc(sizeof(TimeWatcher));
+ time_watcher_init(&main_loop, tw, NULL);
+ tw->events = main_loop.events;
+ tw->blockable = true;
+ time_watcher_start(tw, dummy_timer_due_cb, interval, interval);
+ } else if (tv_interval->v_type != VAR_UNKNOWN) {
+ EMSG2(_(e_invargval), "3");
+ return;
+ }
+
+ typval_T argv = TV_INITIAL_VALUE;
+ typval_T exprval = TV_INITIAL_VALUE;
+ bool error = false;
+ int save_called_emsg = called_emsg;
+ called_emsg = false;
+
+ LOOP_PROCESS_EVENTS_UNTIL(&main_loop, main_loop.events, timeout,
+ eval_expr_typval(&expr, &argv, 0, &exprval) != OK
+ || tv_get_number_chk(&exprval, &error)
+ || called_emsg || error || got_int);
+
+ if (called_emsg || error) {
+ rettv->vval.v_number = -3;
+ } else if (got_int) {
+ got_int = false;
+ vgetc();
+ rettv->vval.v_number = -2;
+ } else if (tv_get_number_chk(&exprval, &error)) {
+ rettv->vval.v_number = 0;
+ }
+
+ called_emsg = save_called_emsg;
+
+ // Stop dummy timer
+ if (tw) {
+ time_watcher_stop(tw);
+ time_watcher_close(tw, dummy_timer_close_cb);
+ }
+}
+
// "win_screenpos()" function
static void f_win_screenpos(typval_T *argvars, typval_T *rettv, FunPtr fptr)
{
diff --git a/src/nvim/eval.lua b/src/nvim/eval.lua
index ebdf3f5489..0ad9ef5dac 100644
--- a/src/nvim/eval.lua
+++ b/src/nvim/eval.lua
@@ -366,6 +366,7 @@ return {
values={args=1},
virtcol={args=1},
visualmode={args={0, 1}},
+ wait={args={2,3}},
wildmenumode={},
win_findbuf={args=1},
win_getid={args={0,2}},
diff --git a/test/functional/eval/wait_spec.lua b/test/functional/eval/wait_spec.lua
new file mode 100644
index 0000000000..5bb27131f5
--- /dev/null
+++ b/test/functional/eval/wait_spec.lua
@@ -0,0 +1,78 @@
+local helpers = require('test.functional.helpers')(after_each)
+local call = helpers.call
+local clear = helpers.clear
+local command = helpers.command
+local eval = helpers.eval
+local eq = helpers.eq
+local expect_err = helpers.expect_err
+local feed = helpers.feed
+local feed_command = helpers.feed_command
+local next_msg = helpers.next_msg
+local nvim = helpers.nvim
+local source = helpers.source
+
+before_each(function()
+ clear()
+ local channel = nvim('get_api_info')[1]
+ nvim('set_var', 'channel', channel)
+end)
+
+describe('wait()', function()
+ it('waits and returns 0 when condition is satisfied', function()
+ source([[
+ let g:_awake = 0
+ call timer_start(100, { -> nvim_command('let g:_awake = 1') })
+ ]])
+ eq(0, eval('g:_awake'))
+ eq(0, eval('wait(1500, { -> g:_awake })'))
+ eq(1, eval('g:_awake'))
+
+ eq(0, eval('wait(0, 1)'))
+ end)
+
+ it('returns -1 on timeout', function()
+ eq(-1, eval('wait(0, 0)'))
+ eq(-1, eval('wait(50, 0)'))
+ end)
+
+ it('returns -2 when interrupted', function()
+ feed_command('call rpcnotify(g:channel, "ready") | '..
+ 'call rpcnotify(g:channel, "wait", wait(-1, 0))')
+ eq({'notification', 'ready', {}}, next_msg())
+ feed('<c-c>')
+ eq({'notification', 'wait', {-2}}, next_msg())
+ end)
+
+ it('returns -3 on error', function()
+ command('silent! let ret = wait(-1, "error")')
+ eq(-3, eval('ret'))
+ command('let ret = 0 | silent! let ret = wait(-1, { -> error })')
+ eq(-3, eval('ret'))
+ end)
+
+ it('evaluates the condition on given interval', function()
+ source([[
+ function Count()
+ let g:counter += 1
+ return g:counter
+ endfunction
+ ]])
+
+ nvim('set_var', 'counter', 0)
+ eq(-1, call('wait', 20, 'Count() >= 5'))
+
+ nvim('set_var', 'counter', 0)
+ eq(0, call('wait', 1000, 'Count() >= 5', 5))
+ eq(5, nvim('get_var', 'counter'))
+ end)
+
+ it('errors out on invalid timeout value', function()
+ expect_err('Invalid value for argument', call, 'wait', '', 1)
+ end)
+
+ it('errors out on invalid interval', function()
+ expect_err('Invalid value for argument', call, 'wait', 0, 1, -1)
+ expect_err('Invalid value for argument', call, 'wait', 0, 1, 0)
+ expect_err('Invalid value for argument', call, 'wait', 0, 1, '')
+ end)
+end)