2015-11-29 19:36 GMT+03:00 Bram Moolenaar <[email protected]>:
>
> Patch 7.4.944
> Problem: Writing tests for Vim script is hard.
> Solution: Add assertEqual(), assertFalse() and assertTrue() functions.
> Add
>
Why `assert{upper case letter}`? I know exactly no functions that use this
naming convention.
> the v:errors variable. Add the runtest script. Add a first new
> style test script.
> Files: src/eval.c, src/vim.h, src/misc2.c, src/testdir/Makefile,
> src/testdir/runtest.vim, src/testdir/test_assert.vim,
> runtime/doc/eval.txt
>
>
> *** ../vim-7.4.943/src/eval.c 2015-09-29 16:53:18.200480733 +0200
> --- src/eval.c 2015-11-29 16:45:12.155720718 +0100
> ***************
> *** 368,373 ****
> --- 368,374 ----
> {VV_NAME("option_new", VAR_STRING), VV_RO},
> {VV_NAME("option_old", VAR_STRING), VV_RO},
> {VV_NAME("option_type", VAR_STRING), VV_RO},
> + {VV_NAME("errors", VAR_LIST), 0},
> };
>
> /* shorthand */
> ***************
> *** 472,477 ****
> --- 473,481 ----
> static void f_argidx __ARGS((typval_T *argvars, typval_T *rettv));
> static void f_arglistid __ARGS((typval_T *argvars, typval_T *rettv));
> static void f_argv __ARGS((typval_T *argvars, typval_T *rettv));
> + static void f_assertEqual __ARGS((typval_T *argvars, typval_T *rettv));
> + static void f_assertFalse __ARGS((typval_T *argvars, typval_T *rettv));
> + static void f_assertTrue __ARGS((typval_T *argvars, typval_T *rettv));
> #ifdef FEAT_FLOAT
> static void f_asin __ARGS((typval_T *argvars, typval_T *rettv));
> static void f_atan __ARGS((typval_T *argvars, typval_T *rettv));
> ***************
> *** 8068,8073 ****
> --- 8072,8080 ----
> {"argidx", 0, 0, f_argidx},
> {"arglistid", 0, 2, f_arglistid},
> {"argv", 0, 1, f_argv},
> + {"assertEqual", 2, 3, f_assertEqual},
> + {"assertFalse", 1, 2, f_assertFalse},
> + {"assertTrue", 1, 2, f_assertTrue},
> #ifdef FEAT_FLOAT
> {"asin", 1, 1, f_asin}, /* WJMc */
> {"atan", 1, 1, f_atan},
> ***************
> *** 9124,9129 ****
> --- 9131,9243 ----
> alist_name(&ARGLIST[idx]),
> -1);
> }
>
> + static void assertError __ARGS((garray_T *gap));
> + static void prepareForAssertError __ARGS((garray_T*gap));
> + static void assertBool __ARGS((typval_T *argvars, int isTrue));
> +
> + /*
> + * Add an assert error to v:errors.
> + */
> + static void
> + assertError(gap)
> + garray_T *gap;
> + {
> + struct vimvar *vp = &vimvars[VV_ERRORS];
> +
> + if (vp->vv_type != VAR_LIST || vimvars[VV_ERRORS].vv_list == NULL)
> + /* Make sure v:errors is a list. */
> + set_vim_var_list(VV_ERRORS, list_alloc());
> + list_append_string(vimvars[VV_ERRORS].vv_list, gap->ga_data,
> gap->ga_len);
> + }
> +
> + static void
> + prepareForAssertError(gap)
> + garray_T *gap;
> + {
> + char buf[NUMBUFLEN];
> +
> + ga_init2(gap, 1, 100);
> + ga_concat(gap, sourcing_name);
> + sprintf(buf, " line %ld", (long)sourcing_lnum);
> + ga_concat(gap, (char_u *)buf);
> + }
> +
> + /*
> + * "assertEqual(expected, actual[, msg])" function
> + */
> + static void
> + f_assertEqual(argvars, rettv)
> + typval_T *argvars;
> + typval_T *rettv UNUSED;
> + {
> + garray_T ga;
> + char_u *tofree;
> + char_u numbuf[NUMBUFLEN];
> +
> + if (!tv_equal(&argvars[0], &argvars[1], FALSE, FALSE))
> + {
> + prepareForAssertError(&ga);
> + ga_concat(&ga, (char_u *)": Expected ");
> + ga_concat(&ga, tv2string(&argvars[0], &tofree, numbuf, 0));
> + vim_free(tofree);
> + ga_concat(&ga, (char_u *)" but got ");
> + ga_concat(&ga, tv2string(&argvars[1], &tofree, numbuf, 0));
> + vim_free(tofree);
> + assertError(&ga);
> + ga_clear(&ga);
> + }
> + }
> +
> + static void
> + assertBool(argvars, isTrue)
> + typval_T *argvars;
> + int isTrue;
> + {
> + int error = FALSE;
> + garray_T ga;
> + char_u *tofree;
> + char_u numbuf[NUMBUFLEN];
> +
> + if (argvars[0].v_type != VAR_NUMBER
> + || (get_tv_number_chk(&argvars[0], &error) == 0) == isTrue
> + || error)
> + {
> + prepareForAssertError(&ga);
> + ga_concat(&ga, (char_u *)": Expected ");
> + if (isTrue)
> + ga_concat(&ga, (char_u *)"True ");
> + else
> + ga_concat(&ga, (char_u *)"False ");
> + ga_concat(&ga, (char_u *)"but got ");
> + ga_concat(&ga, tv2string(&argvars[0], &tofree, numbuf, 0));
> + vim_free(tofree);
> + assertError(&ga);
> + ga_clear(&ga);
> + }
> + }
> +
> + /*
> + * "assertFalse(actual[, msg])" function
> + */
> + static void
> + f_assertFalse(argvars, rettv)
> + typval_T *argvars;
> + typval_T *rettv UNUSED;
> + {
> + assertBool(argvars, FALSE);
> + }
> +
> + /*
> + * "assertTrue(actual[, msg])" function
> + */
> + static void
> + f_assertTrue(argvars, rettv)
> + typval_T *argvars;
> + typval_T *rettv UNUSED;
> + {
> + assertBool(argvars, TRUE);
> + }
> +
> #ifdef FEAT_FLOAT
> /*
> * "asin()" function
> *** ../vim-7.4.943/src/vim.h 2015-07-17 17:38:00.567399623 +0200
> --- src/vim.h 2015-11-29 14:53:31.936054759 +0100
> ***************
> *** 1902,1908 ****
> #define VV_OPTION_NEW 59
> #define VV_OPTION_OLD 60
> #define VV_OPTION_TYPE 61
> ! #define VV_LEN 62 /* number of v: vars */
>
> #ifdef FEAT_CLIPBOARD
>
> --- 1902,1909 ----
> #define VV_OPTION_NEW 59
> #define VV_OPTION_OLD 60
> #define VV_OPTION_TYPE 61
> ! #define VV_ERRORS 62
> ! #define VV_LEN 63 /* number of v: vars */
>
> #ifdef FEAT_CLIPBOARD
>
> *** ../vim-7.4.943/src/misc2.c 2015-11-10 19:04:18.729542221 +0100
> --- src/misc2.c 2015-11-29 15:59:56.977106043 +0100
> ***************
> *** 2092,2097 ****
> --- 2092,2098 ----
>
> /*
> * Concatenate a string to a growarray which contains characters.
> + * When "s" is NULL does not do anything.
> * Note: Does NOT copy the NUL at the end!
> */
> void
> ***************
> *** 2099,2106 ****
> garray_T *gap;
> char_u *s;
> {
> ! int len = (int)STRLEN(s);
>
> if (ga_grow(gap, len) == OK)
> {
> mch_memmove((char *)gap->ga_data + gap->ga_len, s, (size_t)len);
> --- 2100,2110 ----
> garray_T *gap;
> char_u *s;
> {
> ! int len;
>
> + if (s == NULL)
> + return;
> + len = (int)STRLEN(s);
> if (ga_grow(gap, len) == OK)
> {
> mch_memmove((char *)gap->ga_data + gap->ga_len, s, (size_t)len);
> *** ../vim-7.4.943/src/testdir/Makefile 2015-11-28 14:29:20.344223803 +0100
> --- src/testdir/Makefile 2015-11-29 16:08:44.823408011 +0100
> ***************
> *** 68,82 ****
> test_utf8.out \
> test_writefile.out
>
> SCRIPTS_GUI = test16.out
>
> SCRIPTS_BENCH = bench_re_freeze.out
>
> ! .SUFFIXES: .in .out
>
> ! nongui: nolog $(SCRIPTS) report
>
> ! gui: nolog $(SCRIPTS) $(SCRIPTS_GUI) report
>
> benchmark: $(SCRIPTS_BENCH)
>
> --- 68,84 ----
> test_utf8.out \
> test_writefile.out
>
> + NEW_TESTS = test_assert.res
> +
> SCRIPTS_GUI = test16.out
>
> SCRIPTS_BENCH = bench_re_freeze.out
>
> ! .SUFFIXES: .in .out .res .vim
>
> ! nongui: nolog $(SCRIPTS) newtests report
>
> ! gui: nolog $(SCRIPTS) $(SCRIPTS_GUI) newtests report
>
> benchmark: $(SCRIPTS_BENCH)
>
> ***************
> *** 95,101 ****
> RUN_VIM = VIMRUNTIME=$(SCRIPTSOURCE); export VIMRUNTIME; $(VALGRIND)
> $(VIMPROG) -u unix.vim -U NONE --noplugin -s dotest.in
>
> clean:
> ! -rm -rf *.out *.failed *.rej *.orig test.log $(RM_ON_RUN)
> $(RM_ON_START) valgrind.*
>
> test1.out: test1.in
> -rm -rf $*.failed $(RM_ON_RUN) $(RM_ON_START) wrongtermsize
> --- 97,103 ----
> RUN_VIM = VIMRUNTIME=$(SCRIPTSOURCE); export VIMRUNTIME; $(VALGRIND)
> $(VIMPROG) -u unix.vim -U NONE --noplugin -s dotest.in
>
> clean:
> ! -rm -rf *.out *.failed *.res *.rej *.orig test.log $(RM_ON_RUN)
> $(RM_ON_START) valgrind.*
>
> test1.out: test1.in
> -rm -rf $*.failed $(RM_ON_RUN) $(RM_ON_START) wrongtermsize
> ***************
> *** 157,159 ****
> --- 159,172 ----
>
> nolog:
> -rm -f test.log
> +
> +
> + # New style of tests uses Vim script with assert calls. These are easier
> + # to write and a lot easier to read and debug.
> + # Limitation: Only works with the +eval feature.
> + RUN_VIMTEST = VIMRUNTIME=$(SCRIPTSOURCE); export VIMRUNTIME; $(VALGRIND)
> $(VIMPROG) -u unix.vim -U NONE --noplugin
> +
> + newtests: $(NEW_TESTS)
> +
> + .vim.res:
> + $(RUN_VIMTEST) -u runtest.vim $*.vim
> *** ../vim-7.4.943/src/testdir/runtest.vim 2015-11-29
> 17:31:53.669293134 +0100
> --- src/testdir/runtest.vim 2015-11-29 17:27:48.991967016 +0100
> ***************
> *** 0 ****
> --- 1,97 ----
> + " This script is sourced while editing the .vim file with the tests.
> + " When the script is successful the .res file will be created.
> + " Errors are appended to the test.log file.
> + "
> + " The test script may contain anything, only functions that start with
> + " "Test_" are special. These will be invoked and should contain assert
> + " functions. See test_assert.vim for an example.
> + "
> + " It is possible to source other files that contain "Test_" functions.
> This
> + " can speed up testing, since Vim does not need to restart. But be
> careful
> + " that the tests do not interfere with each other.
> + "
> + " If an error cannot be detected properly with an assert function add the
> + " error to the v:errors list:
> + " call add(v:errors, 'test foo failed: Cannot find xyz')
> + "
> + " If preparation for each Test_ function is needed, define a SetUp
> function.
> + " It will be called before each Test_ function.
> + "
> + " If cleanup after each Test_ function is needed, define a TearDown
> function.
> + " It will be called after each Test_ function.
> +
> + " Without the +eval feature we can't run these tests, bail out.
> + if 0
> + quit!
> + endif
> +
> + " Check that the screen size is at least 24 x 80 characters.
> + if &lines < 24 || &columns < 80
> + let error = 'Screen size too small! Tests require at least 24 lines
> with 80 characters'
> + echoerr error
> + split test.log
> + $put =error
> + w
> + cquit
> + endif
> +
> + " Source the test script. First grab the file name, in case the script
> + " navigates away.
> + let testname = expand('%')
> + source %
> +
> + " Locate Test_ functions and execute them.
> + redir @q
> + function /^Test_
> + redir END
> + let tests = split(substitute(@q, 'function \(\k*()\)', '\1', 'g'))
> +
> + let done = 0
> + let fail = 0
> + let errors = []
> + for test in tests
> + if exists("*SetUp")
> + call SetUp()
> + endif
> +
> + let done += 1
> + try
> + exe 'call ' . test
> + catch
> + let fail += 1
> + call add(v:errors, 'Caught exception in ' . test . ': ' .
> v:exception . ' @ ' . v:throwpoint)
> + endtry
> +
> + if len(v:errors) > 0
> + let fail += 1
> + call add(errors, 'Found errors in ' . test . ':')
> + call extend(errors, v:errors)
> + let v:errors = []
> + endif
> +
> + if exists("*TearDown")
> + call TearDown()
> + endif
> + endfor
> +
> + if fail == 0
> + " Success, create the .res file so that make knows it's done.
> + split %:r.res
> + write
> + endif
> +
> + if len(errors) > 0
> + " Append errors to test.log
> + split test.log
> + call append(line('$'), '')
> + call append(line('$'), 'From ' . testname . ':')
> + call append(line('$'), errors)
> + write
> + endif
> +
> + echo 'Executed ' . done . (done > 1 ? ' tests': ' test')
> + if fail > 0
> + echo fail . ' FAILED'
> + endif
> +
> + qall!
> *** ../vim-7.4.943/src/testdir/test_assert.vim 2015-11-29
> 17:31:53.677293047 +0100
> --- src/testdir/test_assert.vim 2015-11-29 17:29:43.674713619 +0100
> ***************
> *** 0 ****
> --- 1,19 ----
> + " Test that the methods used for testing work.
> +
> + func Test_assertFalse()
> + call assertFalse(0)
> + endfunc
> +
> + func Test_assertTrue()
> + call assertTrue(1)
> + call assertTrue(123)
> + endfunc
> +
> + func Test_assertEqual()
> + let s = 'foo'
> + call assertEqual('foo', s)
> + let n = 4
> + call assertEqual(4, n)
> + let l = [1, 2, 3]
> + call assertEqual([1, 2, 3], l)
> + endfunc
> *** ../vim-7.4.943/runtime/doc/eval.txt 2015-08-11 14:26:03.586931222 +0200
> --- runtime/doc/eval.txt 2015-11-29 16:45:20.891626037 +0100
> ***************
> *** 1379,1384 ****
> --- 1379,1393 ----
> : ... handle error
> < "errmsg" also works, for backwards compatibility.
>
> + *v:errors* *errors-variable*
> + v:errors Errors found by assert functions, such as |assertTrue()|.
> + This is a list of strings.
> + The assert functions append an item when an assert fails.
> + To remove old results make it empty: >
> + :let v:errors = []
> + < If v:errors is set to anything but a list it is made an
> empty
> + list by the assert function.
> +
> *v:exception* *exception-variable*
> v:exception The value of the exception most recently caught and not
> finished. See also |v:throwpoint| and |throw-variables|.
> ***************
> *** 1735,1740 ****
> --- 1746,1754 ----
> Number argument list id
> argv( {nr}) String {nr} entry of the argument list
> argv( ) List the argument list
> + assertEqual( {exp}, {act}) none assert that {exp} equals {act}
> + assertFalse( {actual}) none assert that {actual} is
> false
> + assertTrue( {actual}) none assert that {actual} is true
> asin( {expr}) Float arc sine of {expr}
> atan( {expr}) Float arc tangent of {expr}
> atan2( {expr}, {expr}) Float arc tangent of {expr1} /
> {expr2}
> ***************
> *** 2152,2157 ****
> --- 2166,2196 ----
> < Without the {nr} argument a |List| with the whole
> |arglist| is
> returned.
>
> + *assertEqual()*
> + assertEqual({expected}, {actual})
> + When {expected} and {actual} are not equal an error
> message is
> + added to |v:errors|.
> + There is no automatic conversion, the String "4" is
> different
> + from the Number 4. And the number 4 is different from the
> + Float 4.0. The value of 'ignorecase' is not used here,
> case
>
It would be helpful to disambiguate what happens with containers here. I
guess this function is like `assertTrue(type(expected) == type(actual) &&
expected ==# actual)`, am I right?
> + always matters.
> + Example: >
> + assertEqual('foo', 'bar')
> + < Will result in a string to be added to |v:errors|:
> + test.vim line 12: Expected 'foo' but got 'bar' ~
> +
> + assertFalse({actual}) *assertFalse()*
> + When {actual} is not false an error message is added to
> + |v:errors|, like with |assertEqual()|..
> + A value is false when it is zero. When "{actual}" is not a
> + number the assert fails.
> +
> + assertTrue({actual}) *assertTrue()*
> + When {actual} is not true an error message is added to
> + |v:errors|, like with |assertEqual()|..
> + A value is true when it is a non-zeron number. When
> {actual}
> + is not a number the assert fails.
> +
> asin({expr}) *asin()*
> Return the arc sine of {expr} measured in radians, as a
> |Float|
> in the range of [-pi/2, pi/2].
> *** ../vim-7.4.943/src/version.c 2015-11-28 14:29:20.348223759 +0100
> --- src/version.c 2015-11-29 17:30:53.717948204 +0100
> ***************
> *** 743,744 ****
> --- 743,746 ----
> { /* Add new patch number below this line */
> + /**/
> + 944,
> /**/
>
> --
> You are not really successful until someone claims he sat
> beside you in school.
>
> /// Bram Moolenaar -- [email protected] -- http://www.Moolenaar.net
> \\\
> /// sponsor Vim, vote for features -- http://www.Vim.org/sponsor/
> \\\
> \\\ an exciting new programming language -- http://www.Zimbu.org
> ///
> \\\ help me help AIDS victims -- http://ICCF-Holland.org
> ///
>
> --
> --
> You received this message from the "vim_dev" maillist.
> Do not top-post! Type your reply below the text you are replying to.
> For more information, visit http://www.vim.org/maillist.php
>
> ---
> You received this message because you are subscribed to the Google Groups
> "vim_dev" group.
> To unsubscribe from this group and stop receiving emails from it, send an
> email to [email protected].
> For more options, visit https://groups.google.com/d/optout.
>
--
--
You received this message from the "vim_dev" maillist.
Do not top-post! Type your reply below the text you are replying to.
For more information, visit http://www.vim.org/maillist.php
---
You received this message because you are subscribed to the Google Groups
"vim_dev" group.
To unsubscribe from this group and stop receiving emails from it, send an email
to [email protected].
For more options, visit https://groups.google.com/d/optout.