]> granicus.if.org Git - vim/commitdiff
patch 8.2.2620: Vim9: Using #{ for a dictionary gives strange errors v8.2.2620
authorBram Moolenaar <Bram@vim.org>
Thu, 18 Mar 2021 20:37:55 +0000 (21:37 +0100)
committerBram Moolenaar <Bram@vim.org>
Thu, 18 Mar 2021 20:37:55 +0000 (21:37 +0100)
Problem:    Vim9: Using #{ for a dictionary gives strange errors.
Solution:   Give an error when using #{ for a comment after a command.

src/errors.h
src/proto/vim9script.pro
src/testdir/test_vim9_expr.vim
src/testdir/test_vim9_script.vim
src/version.c
src/vim9compile.c
src/vim9script.c

index c1dc547fbe8195f81c8d1bf020c15298231125c3..ed55304bc4053acc9274f1f0d3d9c8872646b816 100644 (file)
@@ -375,3 +375,5 @@ EXTERN char e_argument_already_declared_in_script_str[]
        INIT(= N_("E1168: Argument already declared in the script: %s"));
 EXTERN char e_import_as_name_not_supported_here[]
        INIT(= N_("E1169: 'import * as {name}' not supported here"));
+EXTERN char e_cannot_use_hash_curly_to_start_comment[]
+       INIT(= N_("E1170: 'Cannot use #{ to start a comment"));
index c80618f750632f21c92fafbe6f0a2cff854a5aa8..a7d821c0d837a95bb494b27cb8c801e9be4e3a23 100644 (file)
@@ -2,6 +2,7 @@
 int in_vim9script(void);
 void ex_vim9script(exarg_T *eap);
 int not_in_vim9(exarg_T *eap);
+int vim9_bad_comment(char_u *p);
 int vim9_comment_start(char_u *p);
 void ex_export(exarg_T *eap);
 void free_imports_and_script_vars(int sid);
index 9829d88007ff2705f8f1cdb6b015fdc5b75401ab..10468410feaf57059c0b5c84fcaec85b1594d4b3 100644 (file)
@@ -2159,8 +2159,10 @@ def Test_expr7_dict()
   CheckDefAndScriptSuccess(lines)
  
   # legacy syntax doesn't work
-  CheckDefFailure(["var x = #{key: 8}"], 'E1097:', 3)
-  CheckDefFailure(["var x = 'a' .. #{a: 1}"], 'E1097:', 3)
+  CheckDefFailure(["var x = #{key: 8}"], 'E1170:', 1)
+  CheckDefFailure(["var x = 'a' #{a: 1}"], 'E1170:', 1)
+  CheckDefFailure(["var x = 'a' .. #{a: 1}"], 'E1170:', 1)
+  CheckDefFailure(["var x = true ? #{a: 1}"], 'E1170:', 1)
 
   CheckDefFailure(["var x = {a:8}"], 'E1069:', 1)
   CheckDefFailure(["var x = {a : 8}"], 'E1068:', 1)
index 38d0b0abcd0833b0f31d41c738086fbe54254121..49ecbea650ad5ae9e15e87e01bc3f24779af1861 100644 (file)
@@ -2452,7 +2452,7 @@ def Test_while_loop()
   assert_equal('1_3_', result)
 
   var s = ''
-  while s == 'x' #{comment}
+  while s == 'x' # {comment}
   endwhile
 enddef
 
index 74eec468a5aca60681c454d3f52c8d9ff68241b7..dfc0f033cab130cd57af8a9ddf7a0ba0a2a146f9 100644 (file)
@@ -750,6 +750,8 @@ static char *(features[]) =
 
 static int included_patches[] =
 {   /* Add new patch number below this line */
+/**/
+    2620,
 /**/
     2619,
 /**/
index 458b4a1a34cee08730b02bd4e26fc6a05baf33f5..238ce5d1fac14bf1319c6d7422238ad55f2c8a71 100644 (file)
@@ -1546,7 +1546,7 @@ generate_FUNCREF(cctx_T *cctx, ufunc_T *ufunc)
     isn->isn_arg.funcref.fr_func = ufunc->uf_dfunc_idx;
     cctx->ctx_has_closure = 1;
 
-    // if the referenced function is a closure, it may use items further up in
+    // If the referenced function is a closure, it may use items further up in
     // the nested context, including this one.
     if (ufunc->uf_flags & FC_CLOSURE)
        cctx->ctx_ufunc->uf_flags |= FC_CLOSURE;
@@ -2401,6 +2401,8 @@ peek_next_line_from_context(cctx_T *cctx)
        if (line != NULL)
        {
            p = skipwhite(line);
+           if (vim9_bad_comment(p))
+               return NULL;
            if (*p != NUL && !vim9_comment_start(p))
                return p;
        }
@@ -2465,6 +2467,8 @@ next_line_from_context(cctx_T *cctx, int skip_comment)
 may_get_next_line(char_u *whitep, char_u **arg, cctx_T *cctx)
 {
     *arg = skipwhite(whitep);
+    if (vim9_bad_comment(*arg))
+       return FAIL;
     if (**arg == NUL || (VIM_ISWHITE(*whitep) && vim9_comment_start(*arg)))
     {
        char_u *next = next_line_from_context(cctx, TRUE);
@@ -4277,10 +4281,13 @@ compile_expr7(
 
        if (!eval_isnamec1(**arg))
        {
-           if (ends_excmd(*skipwhite(*arg)))
-               semsg(_(e_empty_expression_str), *arg);
-           else
-               semsg(_(e_name_expected_str), *arg);
+           if (!vim9_bad_comment(*arg))
+           {
+               if (ends_excmd(*skipwhite(*arg)))
+                   semsg(_(e_empty_expression_str), *arg);
+               else
+                   semsg(_(e_name_expected_str), *arg);
+           }
            return FAIL;
        }
 
@@ -8297,6 +8304,8 @@ compile_def_function(
            semsg(_(e_trailing_arg), line);
            goto erret;
        }
+       else if (line != NULL && vim9_bad_comment(skipwhite(line)))
+           goto erret;
        else
        {
            line = next_line_from_context(&cctx, FALSE);
index 05977b641c879aa72397c77b693b9517119348cf..309d4bdaf6c9672f0ef01ba3ea5ae020726bd90f 100644 (file)
@@ -113,12 +113,29 @@ not_in_vim9(exarg_T *eap)
 }
 
 /*
- * Return TRUE if "p" points at a "#".  Does not check for white space.
+ * Give an error message if "p" points at "#{" and return TRUE.
+ * This avoids that using a legacy style #{} dictionary leads to difficult to
+ * understand errors.
+ */
+    int
+vim9_bad_comment(char_u *p)
+{
+    if (p[0] == '#' && p[1] == '{')
+    {
+       emsg(_(e_cannot_use_hash_curly_to_start_comment));
+       return TRUE;
+    }
+    return FALSE;
+}
+
+/*
+ * Return TRUE if "p" points at a "#" not followed by '{'.
+ * Does not check for white space.
  */
     int
 vim9_comment_start(char_u *p)
 {
-    return p[0] == '#';
+    return p[0] == '#' && p[1] != '{';
 }
 
 #if defined(FEAT_EVAL) || defined(PROTO)