[njs] Shell: completions are refactored and moved out of the core.

Dmitry Volyntsev xeioex at nginx.com
Tue Mar 19 05:56:58 UTC 2024


details:   https://hg.nginx.org/njs/rev/fe94552843d7
branches:  
changeset: 2301:fe94552843d7
user:      Dmitry Volyntsev <xeioex at nginx.com>
date:      Mon Mar 18 22:15:48 2024 -0700
description:
Shell: completions are refactored and moved out of the core.

diffstat:

 external/njs_shell.c    |  275 +++++++++++++++++++++++++++++++-------------
 src/njs.h               |    2 -
 src/njs_builtin.c       |  294 +-----------------------------------------------
 test/shell_test_njs.exp |   26 +--
 4 files changed, 201 insertions(+), 396 deletions(-)

diffs (737 lines):

diff -r 1aa2bb15c966 -r fe94552843d7 external/njs_shell.c
--- a/external/njs_shell.c	Fri Mar 15 23:14:39 2024 -0700
+++ b/external/njs_shell.c	Mon Mar 18 22:15:48 2024 -0700
@@ -83,15 +83,7 @@ typedef enum {
 
 typedef struct {
     size_t                  index;
-    size_t                  length;
-    njs_arr_t               *completions;
     njs_arr_t               *suffix_completions;
-    njs_rbtree_node_t       *node;
-
-    enum {
-       NJS_COMPLETION_SUFFIX = 0,
-       NJS_COMPLETION_GLOBAL
-    }                       phase;
 } njs_completion_t;
 
 
@@ -157,7 +149,6 @@ struct njs_engine_s {
             njs_vm_t            *vm;
 
             njs_opaque_value_t  value;
-            njs_completion_t    completion;
         }                       njs;
 #if (NJS_HAVE_QUICKJS)
         struct {
@@ -174,9 +165,11 @@ struct njs_engine_s {
     njs_int_t                 (*process_events)(njs_engine_t *engine);
     njs_int_t                 (*destroy)(njs_engine_t *engine);
     njs_int_t                 (*output)(njs_engine_t *engine, njs_int_t ret);
+    njs_arr_t                *(*complete)(njs_engine_t *engine, njs_str_t *ex);
 
     unsigned                    type;
     njs_mp_t                    *pool;
+    njs_completion_t            completion;
 };
 
 typedef struct {
@@ -1361,11 +1354,6 @@ njs_engine_njs_init(njs_engine_t *engine
         return NJS_ERROR;
     }
 
-    engine->u.njs.completion.completions = njs_vm_completions(vm, NULL);
-    if (engine->u.njs.completion.completions == NULL) {
-        return NJS_ERROR;
-    }
-
     if (opts->unhandled_rejection) {
         njs_vm_set_rejection_tracker(vm, njs_rejection_tracker,
                                      njs_vm_external_ptr(vm));
@@ -1455,6 +1443,179 @@ njs_engine_njs_output(njs_engine_t *engi
 }
 
 
+static njs_arr_t *
+njs_object_completions(njs_vm_t *vm, njs_value_t *object, njs_str_t *expression)
+{
+    u_char              *prefix;
+    size_t              len, prefix_len;
+    int64_t             k, n, length;
+    njs_int_t           ret;
+    njs_arr_t           *array;
+    njs_str_t           *completion, key;
+    njs_value_t         *keys;
+    njs_opaque_value_t  *start, retval, prototype;
+
+    prefix = expression->start + expression->length;
+
+    while (prefix > expression->start && *prefix != '.') {
+        prefix--;
+    }
+
+    if (prefix != expression->start) {
+        prefix++;
+    }
+
+    prefix_len = prefix - expression->start;
+    len = expression->length - prefix_len;
+
+    array = njs_arr_create(njs_vm_memory_pool(vm), 8, sizeof(njs_str_t));
+    if (njs_slow_path(array == NULL)) {
+        goto fail;
+    }
+
+    while (!njs_value_is_null(object)) {
+        keys = njs_vm_value_enumerate(vm, object, NJS_ENUM_KEYS
+                                      | NJS_ENUM_STRING,
+                                      njs_value_arg(&retval));
+        if (njs_slow_path(keys == NULL)) {
+            goto fail;
+        }
+
+        (void) njs_vm_array_length(vm, keys, &length);
+
+        start = (njs_opaque_value_t *) njs_vm_array_start(vm, keys);
+        if (start == NULL) {
+            goto fail;
+        }
+
+
+        for (n = 0; n < length; n++) {
+            ret = njs_vm_value_to_string(vm, &key, njs_value_arg(start));
+            if (njs_slow_path(ret != NJS_OK)) {
+                goto fail;
+            }
+
+            start++;
+
+            if (len > key.length || njs_strncmp(key.start, prefix, len) != 0) {
+                continue;
+            }
+
+            for (k = 0; k < array->items; k++) {
+                completion = njs_arr_item(array, k);
+
+                if ((completion->length - prefix_len - 1) == key.length
+                    && njs_strncmp(&completion->start[prefix_len],
+                                   key.start, key.length)
+                       == 0)
+                {
+                    break;
+                }
+            }
+
+            if (k != array->items) {
+                continue;
+            }
+
+            completion = njs_arr_add(array);
+            if (njs_slow_path(completion == NULL)) {
+                goto fail;
+            }
+
+            completion->length = prefix_len + key.length + 1;
+            completion->start = njs_mp_alloc(njs_vm_memory_pool(vm),
+                                             completion->length);
+            if (njs_slow_path(completion->start == NULL)) {
+                goto fail;
+            }
+
+
+            njs_sprintf(completion->start,
+                        completion->start + completion->length,
+                        "%*s%V%Z", prefix_len, expression->start, &key);
+        }
+
+        ret = njs_vm_prototype(vm, object, njs_value_arg(&prototype));
+        if (njs_slow_path(ret != NJS_OK)) {
+            goto fail;
+        }
+
+        object = njs_value_arg(&prototype);
+    }
+
+    return array;
+
+fail:
+
+    if (array != NULL) {
+        njs_arr_destroy(array);
+    }
+
+    return NULL;
+}
+
+
+static njs_arr_t *
+njs_engine_njs_complete(njs_engine_t *engine, njs_str_t *expression)
+{
+    u_char              *p, *start, *end;
+    njs_vm_t            *vm;
+    njs_int_t           ret;
+    njs_bool_t          global;
+    njs_opaque_value_t  value, key, retval;
+
+    vm = engine->u.njs.vm;
+
+    p = expression->start;
+    end = p + expression->length;
+
+    global = 1;
+    (void) njs_vm_global(vm, njs_value_arg(&value));
+
+    while (p < end && *p != '.') { p++; }
+
+    if (p == end) {
+        goto done;
+    }
+
+    p = expression->start;
+
+    for ( ;; ) {
+
+        start = (*p == '.' && p < end) ? ++p: p;
+
+        if (p == end) {
+            break;
+        }
+
+        while (p < end && *p != '.') { p++; }
+
+        ret = njs_vm_value_string_set(vm, njs_value_arg(&key), start,
+                                      p - start);
+        if (njs_slow_path(ret != NJS_OK)) {
+            return NULL;
+        }
+
+        ret = njs_value_property(vm, njs_value_arg(&value), njs_value_arg(&key),
+                                 njs_value_arg(&retval));
+        if (njs_slow_path(ret != NJS_OK)) {
+            if (ret == NJS_DECLINED && !global) {
+                goto done;
+            }
+
+            return NULL;
+        }
+
+        global = 0;
+        njs_value_assign(&value, &retval);
+    }
+
+done:
+
+    return njs_object_completions(vm, njs_value_arg(&value), expression);
+}
+
+
 static njs_int_t
 njs_engine_njs_process_events(njs_engine_t *engine)
 {
@@ -2984,6 +3145,7 @@ njs_create_engine(njs_opts_t *opts)
         engine->process_events = njs_engine_njs_process_events;
         engine->destroy = njs_engine_njs_destroy;
         engine->output = njs_engine_njs_output;
+        engine->complete = njs_engine_njs_complete;
         break;
 
 #ifdef NJS_HAVE_QUICKJS
@@ -3413,91 +3575,38 @@ njs_editline_init(void)
 }
 
 
-/* editline frees the buffer every time. */
-#define njs_editline(s) strndup((char *) (s)->start, (s)->length)
-
-#define njs_completion(c, i) &(((njs_str_t *) (c)->start)[i])
-
-#define njs_next_phase(c)                                                   \
-    (c)->index = 0;                                                         \
-    (c)->phase++;                                                           \
-    goto next;
-
 static char *
 njs_completion_generator(const char *text, int state)
 {
     njs_str_t         expression, *suffix;
-    njs_vm_t          *vm;
+    njs_engine_t      *engine;
     njs_completion_t  *cmpl;
 
-    if (njs_console.engine == NULL
-        || njs_console.engine->type != NJS_ENGINE_NJS)
-    {
+    engine = njs_console.engine;
+    if (engine->complete == NULL) {
         return NULL;
     }
 
-    vm = njs_console.engine->u.njs.vm;
-    cmpl = &njs_console.engine->u.njs.completion;
+    cmpl = &engine->completion;
 
     if (state == 0) {
-        cmpl->phase = NJS_COMPLETION_SUFFIX;
         cmpl->index = 0;
-        cmpl->length = njs_strlen(text);
-        cmpl->suffix_completions = NULL;
-    }
-
-next:
-
-    switch (cmpl->phase) {
-    case NJS_COMPLETION_SUFFIX:
-        if (cmpl->length == 0) {
-            njs_next_phase(cmpl);
-        }
-
+        expression.start = (u_char *) text;
+        expression.length = njs_strlen(text);
+
+        cmpl->suffix_completions = engine->complete(engine, &expression);
         if (cmpl->suffix_completions == NULL) {
-            expression.start = (u_char *) text;
-            expression.length = cmpl->length;
-
-            cmpl->suffix_completions = njs_vm_completions(vm, &expression);
-            if (cmpl->suffix_completions == NULL) {
-                njs_next_phase(cmpl);
-            }
-        }
-
-        for ( ;; ) {
-            if (cmpl->index >= cmpl->suffix_completions->items) {
-                njs_next_phase(cmpl);
-            }
-
-            suffix = njs_completion(cmpl->suffix_completions, cmpl->index++);
-
-            return njs_editline(suffix);
-        }
-
-    case NJS_COMPLETION_GLOBAL:
-        if (cmpl->suffix_completions != NULL) {
-            /* No global completions if suffixes were found. */
-            njs_next_phase(cmpl);
-        }
-
-        for ( ;; ) {
-            if (cmpl->index >= cmpl->completions->items) {
-                break;
-            }
-
-            suffix = njs_completion(cmpl->completions, cmpl->index++);
-
-            if (suffix->start[0] == '.' || suffix->length < cmpl->length) {
-                continue;
-            }
-
-            if (njs_strncmp(text, suffix->start, cmpl->length) == 0) {
-                return njs_editline(suffix);
-            }
+            return NULL;
         }
     }
 
-    return NULL;
+    if (cmpl->index == cmpl->suffix_completions->items) {
+        return NULL;
+    }
+
+    suffix = njs_arr_item(cmpl->suffix_completions, cmpl->index++);
+
+    return strndup((char *) suffix->start, suffix->length);
 }
 
 #endif
diff -r 1aa2bb15c966 -r fe94552843d7 src/njs.h
--- a/src/njs.h	Fri Mar 15 23:14:39 2024 -0700
+++ b/src/njs.h	Mon Mar 18 22:15:48 2024 -0700
@@ -316,8 +316,6 @@ NJS_EXPORT njs_int_t njs_vm_pending(njs_
 NJS_EXPORT void njs_vm_set_rejection_tracker(njs_vm_t *vm,
         njs_rejection_tracker_t rejection_tracker, void *opaque);
 
-NJS_EXPORT void *njs_vm_completions(njs_vm_t *vm, njs_str_t *expression);
-
 /*
  * Runs the specified function with provided arguments.
  *  NJS_OK successful run.
diff -r 1aa2bb15c966 -r fe94552843d7 src/njs_builtin.c
--- a/src/njs_builtin.c	Fri Mar 15 23:14:39 2024 -0700
+++ b/src/njs_builtin.c	Mon Mar 18 22:15:48 2024 -0700
@@ -25,12 +25,7 @@ typedef struct {
 static njs_int_t njs_global_this_prop_handler(njs_vm_t *vm,
     njs_object_prop_t *self, njs_value_t *global, njs_value_t *setval,
     njs_value_t *retval);
-static njs_arr_t *njs_vm_expression_completions(njs_vm_t *vm,
-    njs_str_t *expression);
-static njs_arr_t *njs_vm_global_var_completions(njs_vm_t *vm,
-    njs_str_t *expression);
-static njs_arr_t *njs_object_completions(njs_vm_t *vm, njs_value_t *object,
-    njs_str_t *expression);
+
 static njs_int_t njs_env_hash_init(njs_vm_t *vm, njs_lvlhsh_t *hash,
     char **environment);
 
@@ -415,293 +410,6 @@ njs_builtin_traverse(njs_vm_t *vm, njs_t
 }
 
 
-static njs_arr_t *
-njs_builtin_completions(njs_vm_t *vm)
-{
-    njs_arr_t                *array;
-    njs_str_t                *completion;
-    njs_int_t                ret;
-    njs_lvlhsh_each_t        lhe;
-    njs_builtin_traverse_t   ctx;
-    const njs_object_prop_t  *prop;
-
-    array = njs_arr_create(vm->mem_pool, 64, sizeof(njs_str_t));
-    if (njs_slow_path(array == NULL)) {
-        return NULL;
-    }
-
-    ret = njs_lexer_keywords(array);
-    if (njs_slow_path(ret != NJS_OK)) {
-        return NULL;
-    }
-
-    /* Global object completions. */
-
-    ctx.type = NJS_BUILTIN_TRAVERSE_KEYS;
-    njs_lvlhsh_init(&ctx.keys);
-
-    ret = njs_object_traverse(vm, njs_object(&vm->global_value), &ctx,
-                              njs_builtin_traverse);
-    if (njs_slow_path(ret != NJS_OK)) {
-        return NULL;
-    }
-
-    njs_lvlhsh_each_init(&lhe, &njs_object_hash_proto);
-
-    for ( ;; ) {
-        prop = njs_lvlhsh_each(&ctx.keys, &lhe);
-
-        if (prop == NULL) {
-            break;
-        }
-
-        completion = njs_arr_add(array);
-        if (njs_slow_path(completion == NULL)) {
-            return NULL;
-        }
-
-        njs_string_get(&prop->name, completion);
-    }
-
-    return array;
-}
-
-
-void *
-njs_vm_completions(njs_vm_t *vm, njs_str_t *expression)
-{
-    u_char  *p, *end;
-
-    if (expression == NULL) {
-        return njs_builtin_completions(vm);
-    }
-
-    p = expression->start;
-    end = p + expression->length;
-
-    while (p < end && *p != '.') { p++; }
-
-    if (p == end) {
-        return njs_vm_global_var_completions(vm, expression);
-    }
-
-    return njs_vm_expression_completions(vm, expression);
-}
-
-
-static njs_arr_t *
-njs_vm_global_var_completions(njs_vm_t *vm, njs_str_t *expression)
-{
-    njs_str_t                *completion;
-    njs_arr_t                *array;
-    njs_rbtree_t             *variables;
-    njs_rbtree_node_t        *node;
-    njs_variable_node_t      *vnode;
-    const njs_lexer_entry_t  *lex_entry;
-
-    variables = (vm->global_scope != NULL) ? &vm->global_scope->variables
-                                           : NULL;
-    if (njs_slow_path(variables == NULL)) {
-        return NULL;
-    }
-
-    array = njs_arr_create(vm->mem_pool, 8, sizeof(njs_str_t));
-    if (njs_slow_path(array == NULL)) {
-        return NULL;
-    }
-
-    node = njs_rbtree_min(variables);
-
-    while (njs_rbtree_is_there_successor(variables, node)) {
-        vnode = (njs_variable_node_t *) node;
-
-        node = njs_rbtree_node_successor(variables, node);
-
-        lex_entry = njs_lexer_entry(vnode->key);
-        if (lex_entry == NULL) {
-            continue;
-        }
-
-        if (lex_entry->name.length >= expression->length
-            && njs_strncmp(expression->start, lex_entry->name.start,
-                           expression->length) == 0)
-        {
-            completion = njs_arr_add(array);
-            if (njs_slow_path(completion == NULL)) {
-                return NULL;
-            }
-
-            *completion = lex_entry->name;
-        }
-    }
-
-    return array;
-}
-
-
-static njs_arr_t *
-njs_vm_expression_completions(njs_vm_t *vm, njs_str_t *expression)
-{
-    u_char               *p, *end;
-    njs_int_t            ret;
-    njs_value_t          *value;
-    njs_variable_t       *var;
-    njs_rbtree_node_t    *node;
-    njs_object_prop_t    *prop;
-    njs_lvlhsh_query_t   lhq;
-    njs_variable_node_t  var_node;
-
-    p = expression->start;
-    end = p + expression->length;
-
-    lhq.key.start = p;
-
-    while (p < end && *p != '.') { p++; }
-
-    lhq.proto = &njs_lexer_hash_proto;
-    lhq.key.length = p - lhq.key.start;
-    lhq.key_hash = njs_djb_hash(lhq.key.start, lhq.key.length);
-
-    ret = njs_lvlhsh_find(&vm->shared->keywords_hash, &lhq);
-    if (njs_slow_path(ret != NJS_OK)) {
-        return NULL;
-    }
-
-    var_node.key = (uintptr_t) lhq.value;
-
-    node = njs_rbtree_find(&vm->global_scope->variables, &var_node.node);
-    if (njs_slow_path(node == NULL)) {
-        return NULL;
-    }
-
-    var = ((njs_variable_node_t *) node)->variable;
-    value = njs_scope_value(vm, var->index);
-
-    if (!njs_is_object(value)) {
-        return NULL;
-    }
-
-    lhq.proto = &njs_object_hash_proto;
-
-    for ( ;; ) {
-
-        if (p == end) {
-            break;
-        }
-
-        lhq.key.start = ++p;
-
-        while (p < end && *p != '.') { p++; }
-
-        lhq.key.length = p - lhq.key.start;
-        lhq.key_hash = njs_djb_hash(lhq.key.start, lhq.key.length);
-
-        ret = njs_lvlhsh_find(njs_object_hash(value), &lhq);
-        if (njs_slow_path(ret != NJS_OK)) {
-            if (ret == NJS_DECLINED) {
-                break;
-            }
-
-            return NULL;
-        }
-
-        prop = lhq.value;
-
-        if (njs_is_accessor_descriptor(prop) ||
-            !njs_is_object(njs_prop_value(prop)))
-        {
-            return NULL;
-        }
-
-        value = njs_prop_value(prop);
-    }
-
-    return njs_object_completions(vm, value, expression);
-}
-
-
-static njs_arr_t *
-njs_object_completions(njs_vm_t *vm, njs_value_t *object, njs_str_t *expression)
-{
-    u_char            *prefix;
-    double            num;
-    size_t            len;
-    njs_arr_t         *array;
-    njs_str_t         *completion, key;
-    njs_uint_t        n;
-    njs_array_t       *keys;
-    njs_value_type_t  type;
-
-    prefix = expression->start + expression->length;
-
-    while (prefix > expression->start && *prefix != '.') {
-        prefix--;
-    }
-
-    prefix++;
-    len = expression->length - (prefix - expression->start);
-
-    array = NULL;
-    type = object->type;
-
-    if (type == NJS_ARRAY || type == NJS_TYPED_ARRAY) {
-        object->type = NJS_OBJECT;
-    }
-
-    keys = njs_value_enumerate(vm, object, NJS_ENUM_KEYS | NJS_ENUM_STRING);
-    if (njs_slow_path(keys == NULL)) {
-        goto done;
-    }
-
-    array = njs_arr_create(vm->mem_pool, 8, sizeof(njs_str_t));
-    if (njs_slow_path(array == NULL)) {
-        goto done;
-    }
-
-    for (n = 0; n < keys->length; n++) {
-        njs_string_get(&keys->start[n], &key);
-
-        if (len != 0
-            && njs_strncmp(key.start, prefix, njs_min(len, key.length)) != 0)
-        {
-            continue;
-        }
-
-        num = njs_key_to_index(&keys->start[n]);
-
-        if (!njs_key_is_integer_index(num, &keys->start[n])) {
-            completion = njs_arr_add(array);
-            if (njs_slow_path(completion == NULL)) {
-                njs_arr_destroy(array);
-                array = NULL;
-                goto done;
-            }
-
-            completion->length = (prefix - expression->start) + key.length + 1;
-            completion->start = njs_mp_alloc(vm->mem_pool, completion->length);
-            if (njs_slow_path(completion->start == NULL)) {
-                njs_arr_destroy(array);
-                array = NULL;
-                goto done;
-            }
-
-            njs_sprintf(completion->start,
-                        completion->start + completion->length,
-                        "%*s%V%Z", prefix - expression->start,
-                        expression->start, &key);
-        }
-    }
-
-done:
-
-    if (type == NJS_ARRAY || type == NJS_TYPED_ARRAY) {
-        object->type = type;
-    }
-
-    return array;
-}
-
-
 typedef struct {
     njs_str_t               name;
     njs_function_native_t   native;
diff -r 1aa2bb15c966 -r fe94552843d7 test/shell_test_njs.exp
--- a/test/shell_test_njs.exp	Fri Mar 15 23:14:39 2024 -0700
+++ b/test/shell_test_njs.exp	Mon Mar 18 22:15:48 2024 -0700
@@ -46,23 +46,15 @@ njs_test {
      "a *= 2\r\n2\r\n>> "}
 }
 
-# Global completions, no
-# Disabled: readline does not support it in callback mode
-# njs_test {
-#    {"\t\tn"
-#     "\a\r\nDisplay all*possibilities? (y or n)*>> "}
-# }
+# Global completions
 
-# Global completions, yes
 njs_test {
-    {"\t\ty"
-     "\a\r\nDisplay all*possibilities? (y or n)*Array"}
+    {"\t\t"
+     "$262*"}
 }
 
 # Global completions, single partial match
 
-# \a* is WORKAROUND for libedit-20170329.3.1-r3
-# which inserts '\rESC[6G' after '\a'.
 njs_test {
     {"O\t"
      "O\a*bject"}
@@ -87,13 +79,13 @@ njs_test {
 njs_test {
     {"Type\t"
      "Type\a*Error"}
-    {"\t\t"
-     "TypeError.length"}
+    {".\t\t"
+     "TypeError.__proto__"}
 }
 
 njs_test {
     {"TypeError.\t\t"
-     "TypeError.length*TypeError.prototype"}
+     "TypeError.__proto__*TypeError.prototype"}
 }
 
 njs_test {
@@ -106,8 +98,8 @@ njs_test {
 njs_test {
     {"JS\t"
      "JS\a*ON"}
-    {"\t\t"
-     "JSON.parse*JSON.stringify"}
+    {".\t\t"
+     "JSON.__proto__*JSON.stringify"}
 }
 
 # Global completions, no matches
@@ -134,8 +126,6 @@ njs_test {
      "AA*AAA*"}
 }
 
-# z*z is WORKAROUND for libedit-20170329.3.1-r3
-# which inserts bogus '\a' between 'z'
 njs_test {
     {"var zz = 1\r\n"
      "var zz = 1\r\nundefined\r\n>> "}


More information about the nginx-devel mailing list