[njs] Shell: added completions for QuickJS engine.

Dmitry Volyntsev xeioex at nginx.com
Tue Mar 19 05:57:00 UTC 2024


details:   https://hg.nginx.org/njs/rev/85313a068147
branches:  
changeset: 2302:85313a068147
user:      Dmitry Volyntsev <xeioex at nginx.com>
date:      Mon Mar 18 22:24:57 2024 -0700
description:
Shell: added completions for QuickJS engine.

diffstat:

 external/njs_shell.c    |  202 +++++++++++++++++++++++++++++++++++++++++++++++-
 test/shell_test.exp     |  147 ++++++++++++++++++++++++++++++++++
 test/shell_test_njs.exp |  148 -----------------------------------
 3 files changed, 345 insertions(+), 152 deletions(-)

diffs (541 lines):

diff -r fe94552843d7 -r 85313a068147 external/njs_shell.c
--- a/external/njs_shell.c	Mon Mar 18 22:15:48 2024 -0700
+++ b/external/njs_shell.c	Mon Mar 18 22:24:57 2024 -0700
@@ -3106,6 +3106,203 @@ njs_engine_qjs_output(njs_engine_t *engi
     return NJS_OK;
 }
 
+
+static njs_arr_t *
+njs_qjs_object_completions(njs_engine_t *engine, JSContext *ctx,
+    JSValueConst object, njs_str_t *expression)
+{
+    u_char          *prefix;
+    size_t          len, prefix_len;
+    JSValue         prototype;
+    uint32_t        k, n, length;
+    njs_int_t       ret;
+    njs_arr_t       *array;
+    njs_str_t       *completion, key;
+    JSPropertyEnum  *ptab;
+
+    prefix = expression->start + expression->length;
+
+    while (prefix > expression->start && *prefix != '.') {
+        prefix--;
+    }
+
+    if (prefix != expression->start) {
+        prefix++;
+    }
+
+    ptab = NULL;
+    key.start = NULL;
+    prefix_len = prefix - expression->start;
+    len = expression->length - prefix_len;
+
+    array = njs_arr_create(engine->pool, 8, sizeof(njs_str_t));
+    if (njs_slow_path(array == NULL)) {
+        goto fail;
+    }
+
+    while (!JS_IsNull(object)) {
+        ret = JS_GetOwnPropertyNames(ctx, &ptab, &length, object,
+                                     JS_GPN_STRING_MASK);
+        if (ret < 0) {
+            goto fail;
+        }
+
+        for (n = 0; n < length; n++) {
+            key.start = (u_char *) JS_AtomToCString(ctx, ptab[n].atom);
+            JS_FreeAtom(ctx, ptab[n].atom);
+            if (njs_slow_path(key.start == NULL)) {
+                goto fail;
+            }
+
+            key.length = njs_strlen(key.start);
+
+            if (len > key.length || njs_strncmp(key.start, prefix, len) != 0) {
+                goto next;
+            }
+
+            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)
+                {
+                    goto next;
+                }
+            }
+
+            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(engine->pool, 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);
+
+next:
+
+            JS_FreeCString(ctx, (const char *) key.start);
+        }
+
+        js_free_rt(JS_GetRuntime(ctx), ptab);
+
+        prototype = JS_GetPrototype(ctx, object);
+        if (JS_IsException(prototype)) {
+            goto fail;
+        }
+
+        JS_FreeValue(ctx, object);
+        object = prototype;
+    }
+
+    return array;
+
+fail:
+
+    if (array != NULL) {
+        njs_arr_destroy(array);
+    }
+
+    if (key.start != NULL) {
+        JS_FreeCString(ctx, (const char *) key.start);
+    }
+
+    if (ptab != NULL) {
+        js_free_rt(JS_GetRuntime(ctx), ptab);
+    }
+
+    JS_FreeValue(ctx, object);
+
+    return NULL;
+}
+
+
+static njs_arr_t *
+njs_engine_qjs_complete(njs_engine_t *engine, njs_str_t *expression)
+{
+    u_char      *p, *start, *end;
+    JSAtom      key;
+    JSValue     value, retval;
+    njs_arr_t   *arr;
+    JSContext   *ctx;
+    njs_bool_t  global;
+
+    ctx = engine->u.qjs.ctx;
+
+    p = expression->start;
+    end = p + expression->length;
+
+    global = 1;
+    value = JS_GetGlobalObject(ctx);
+
+    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++; }
+
+        key = JS_NewAtomLen(ctx, (char *) start, p - start);
+        if (key == JS_ATOM_NULL) {
+            goto fail;
+        }
+
+        retval = JS_GetProperty(ctx, value, key);
+
+        JS_FreeAtom(ctx, key);
+
+        if (JS_IsUndefined(retval)) {
+            if (global) {
+                goto fail;
+            }
+
+            goto done;
+        }
+
+        if (JS_IsException(retval)) {
+            goto fail;
+        }
+
+        JS_FreeValue(ctx, value);
+        value = retval;
+        global = 0;
+    }
+
+done:
+
+    arr = njs_qjs_object_completions(engine, ctx, JS_DupValue(ctx, value),
+                                     expression);
+
+    JS_FreeValue(ctx, value);
+
+    return arr;
+
+fail:
+
+    JS_FreeValue(ctx, value);
+
+    return NULL;
+}
+
 #endif
 
 
@@ -3163,6 +3360,7 @@ njs_create_engine(njs_opts_t *opts)
         engine->process_events = njs_engine_qjs_process_events;
         engine->destroy = njs_engine_qjs_destroy;
         engine->output = njs_engine_qjs_output;
+        engine->complete = njs_engine_qjs_complete;
         break;
 #endif
 
@@ -3583,10 +3781,6 @@ njs_completion_generator(const char *tex
     njs_completion_t  *cmpl;
 
     engine = njs_console.engine;
-    if (engine->complete == NULL) {
-        return NULL;
-    }
-
     cmpl = &engine->completion;
 
     if (state == 0) {
diff -r fe94552843d7 -r 85313a068147 test/shell_test.exp
--- a/test/shell_test.exp	Mon Mar 18 22:15:48 2024 -0700
+++ b/test/shell_test.exp	Mon Mar 18 22:24:57 2024 -0700
@@ -58,6 +58,153 @@ njs_test {
      "2\r\n>> "}
 }
 
+# Global completions
+
+njs_test {
+    {"\t\t"
+     "$262*"}
+}
+
+# Global completions, single partial match
+
+njs_test {
+    {"O\t"
+     "O\a*bject"}
+}
+
+njs_test {
+    {"Mat\t"
+     "Mat\a*h"}
+}
+
+ njs_test {
+     {"conso\t"
+     "conso\a*le"}
+ }
+
+# Global completions, multiple partial match
+njs_test {
+    {"cons\t\t"
+     "console*const"}
+}
+
+njs_test {
+    {"Type\t"
+     "Type\a*Error"}
+    {".\t\t"
+     "TypeError.__proto__"}
+}
+
+njs_test {
+    {"TypeError.\t\t"
+     "TypeError.__proto__*TypeError.prototype"}
+}
+
+njs_test {
+    {"Object.ge\t"
+     "Object.ge\a*t"}
+    {"\t\t"
+     "Object.getOwnPropertyDescriptor*Object.getPrototypeOf"}
+}
+
+njs_test {
+    {"JS\t"
+     "JS\a*ON"}
+    {".\t\t"
+     "JSON.__*JSON.stringify"}
+}
+
+njs_test {
+    {"decodeURI\t\t"
+     "decodeURI*decodeURIComponent"}
+    {"C\t"
+     "decodeURIC\a*omponent"}
+}
+
+# Global completions, no matches
+njs_test {
+    {"1.\t\t"
+     "1."}
+}
+
+njs_test {
+    {"1..\t\t"
+     "1.."}
+}
+
+njs_test {
+    {"'abc'.\t\t"
+     "'abc'."}
+}
+
+# Global completions, global vars
+njs_test {
+    {"var AA = 1; var AAA = 2\r\n"
+     "var AA = 1; var AAA = 2\r\nundefined\r\n>> "}
+    {"AA\t\t"
+     "AA*AAA*"}
+}
+
+njs_test {
+    {"var zz = 1\r\n"
+     "var zz = 1\r\nundefined\r\n>> "}
+    {"1 + z\t\r\n"
+     "1 + z*z*\r\n2"}
+}
+
+njs_test {
+    {"unknown_var\t\t"
+     "unknown_var"}
+}
+
+njs_test {
+    {"unknown_var.\t\t"
+     "unknown_var."}
+}
+
+# An object's level completions
+njs_test {
+    {"var o = {zz:1, zb:2}\r\n"
+     "var o = {zz:1, zb:2}\r\nundefined\r\n>> "}
+    {"o.z\t\t"
+     "o.zb*o.zz"}
+}
+
+njs_test {
+    {"var d = new Date()\r\n"
+     "var d = new Date()\r\nundefined\r\n>> "}
+    {"d.to\t\t"
+     "d.toDateString*d.toLocaleDateString*d.toString"}
+}
+
+njs_test {
+    {"var o = {a:new Date()}\r\n"
+     "var o = {a:new Date()}\r\nundefined\r\n>> "}
+    {"o.a.to\t\t"
+     "o.a.toDateString*o.a.toLocaleDateString*o.a.toString"}
+}
+
+njs_test {
+    {"var o = {a:1,b:2,333:'t'}\r\n"
+     "var o = {a:1,b:2,333:'t'}\r\nundefined\r\n>> "}
+    {"o.3\t\t"
+     "o.3"}
+}
+
+njs_test {
+    {"var a = Array(5000000); a.aab = 1; a.aac = 2\r\n"
+     "var a = Array(5000000); a.aab = 1; a.aac = 2\r\n2\r\n>> "}
+    {"a.\t\t"
+     "a.aab*"}
+}
+
+njs_test {
+    {"var a = new Uint8Array([5,6,7,8,8]); a.aab = 1; a.aac = 2\r\n"
+     "var a = new Uint8Array(\\\[5,6,7,8,8]); a.aab = 1; a.aac = 2\r\n2\r\n>> "}
+    {"a.\t\t"
+     "a.aab*"}
+}
+
 # console object
 njs_test {
     {"console[Symbol.toStringTag]\r\n"
diff -r fe94552843d7 -r 85313a068147 test/shell_test_njs.exp
--- a/test/shell_test_njs.exp	Mon Mar 18 22:15:48 2024 -0700
+++ b/test/shell_test_njs.exp	Mon Mar 18 22:24:57 2024 -0700
@@ -38,154 +38,6 @@ njs_test {
      "njs.version\r\n\*\.\*\.\*"}
 }
 
-# simple multi line interaction
-njs_test {
-    {"var a = 1\r\n"
-     "var a = 1\r\nundefined\r\n>> "}
-    {"a *= 2\r\n"
-     "a *= 2\r\n2\r\n>> "}
-}
-
-# Global completions
-
-njs_test {
-    {"\t\t"
-     "$262*"}
-}
-
-# Global completions, single partial match
-
-njs_test {
-    {"O\t"
-     "O\a*bject"}
-}
-
-njs_test {
-    {"Ma\t"
-     "Ma\a*th"}
-}
-
- njs_test {
-     {"conso\t"
-     "conso\a*le"}
- }
-
-# Global completions, multiple partial match
-njs_test {
-    {"cons\t\t"
-     "console*const"}
-}
-
-njs_test {
-    {"Type\t"
-     "Type\a*Error"}
-    {".\t\t"
-     "TypeError.__proto__"}
-}
-
-njs_test {
-    {"TypeError.\t\t"
-     "TypeError.__proto__*TypeError.prototype"}
-}
-
-njs_test {
-    {"Object.g\t"
-     "Object.g\a*et"}
-    {"\t\t"
-     "Object.getOwnPropertyDescriptor*Object.getPrototypeOf"}
-}
-
-njs_test {
-    {"JS\t"
-     "JS\a*ON"}
-    {".\t\t"
-     "JSON.__proto__*JSON.stringify"}
-}
-
-# Global completions, no matches
-njs_test {
-    {"1.\t\t"
-     "1."}
-}
-
-njs_test {
-    {"1..\t\t"
-     "1.."}
-}
-
-njs_test {
-    {"'abc'.\t\t"
-     "'abc'."}
-}
-
-# Global completions, global vars
-njs_test {
-    {"var AA = 1; var AAA = 2\r\n"
-     "var AA = 1; var AAA = 2\r\nundefined\r\n>> "}
-    {"AA\t\t"
-     "AA*AAA*"}
-}
-
-njs_test {
-    {"var zz = 1\r\n"
-     "var zz = 1\r\nundefined\r\n>> "}
-    {"1 + z\t\r\n"
-     "1 + z*z*\r\n2"}
-}
-
-njs_test {
-    {"unknown_var\t\t"
-     "unknown_var"}
-}
-
-njs_test {
-    {"unknown_var.\t\t"
-     "unknown_var."}
-}
-
-# An object's level completions
-njs_test {
-    {"var o = {zz:1, zb:2}\r\n"
-     "var o = {zz:1, zb:2}\r\nundefined\r\n>> "}
-    {"o.z\t\t"
-     "o.zb*o.zz"}
-}
-
-njs_test {
-    {"var d = new Date()\r\n"
-     "var d = new Date()\r\nundefined\r\n>> "}
-    {"d.to\t\t"
-     "d.toDateString*d.toLocaleDateString*d.toString"}
-}
-
-njs_test {
-    {"var o = {a:new Date()}\r\n"
-     "var o = {a:new Date()}\r\nundefined\r\n>> "}
-    {"o.a.to\t\t"
-     "o.a.toDateString*o.a.toLocaleDateString*o.a.toString"}
-}
-
-njs_test {
-    {"var o = {a:1,b:2,333:'t'}\r\n"
-     "var o = {a:1,b:2,333:'t'}\r\nundefined\r\n>> "}
-    {"o.3\t\t"
-     "o.3"}
-}
-
-njs_test {
-    {"var a = Array(5000000); a.aab = 1; a.aac = 2\r\n"
-     "var a = Array(5000000); a.aab = 1; a.aac = 2\r\n2\r\n>> "}
-    {"a.\t\t"
-     "a.aab*"}
-}
-
-njs_test {
-    {"var a = new Uint8Array([5,6,7,8,8]); a.aab = 1; a.aac = 2\r\n"
-     "var a = new Uint8Array(\\\[5,6,7,8,8]); a.aab = 1; a.aac = 2\r\n2\r\n>> "}
-    {"a.\t\t"
-     "a.aab*"}
-}
-
 # console dump
 njs_test {
     {"console.dump()\r\n"


More information about the nginx-devel mailing list