def test_comp( state, typestr, comptype, index, iter_vars_names, not_called_msg, insufficient_ifs_msg, incorrect_iter_vars_msg, comp_iter, ifs, key=None, body=None, value=None, rep=None, ): MSG_INCORRECT_ITER_VARS = "您是否使用了正确的迭代器(iterator)变量?" MSG_INCORRECT_NUM_ITER_VARS = "你是否使用了{{num_vars}}迭代器变量(iterator variables)?" MSG_INSUFFICIENT_IFS = "Have you used {{sol_len}} ifs?" # make sure other messages are set to default if None if insufficient_ifs_msg is None: insufficient_ifs_msg = MSG_INSUFFICIENT_IFS # get comprehension child = check_node(state, comptype, index - 1, typestr, missing_msg=not_called_msg) # test comprehension iter and its variable names (or number of variables) if comp_iter: multi(check_part(child, "iter", "iterable part"), comp_iter) # test iterator variables default_msg = (MSG_INCORRECT_ITER_VARS if iter_vars_names else MSG_INCORRECT_NUM_ITER_VARS) has_context(child, incorrect_iter_vars_msg or default_msg, iter_vars_names) # test the main expressions. if body: multi(check_part(child, "body", "body"), body) # list and gen comp if key: multi(check_part(child, "key", "key part"), key) # dict comp if value: multi(check_part(child, "value", "value part"), value) # "" # test a list of ifs. each entry corresponds to a filter in the comprehension. for i, if_test in enumerate(ifs or []): # test that ifs are same length has_equal_part_len(child, "ifs", insufficient_ifs_msg) # test individual ifs multi( check_part_index(child, "ifs", i, utils.get_ord(i + 1) + " if"), if_test, )
def test_while_loop(state, index=1, test=None, body=None, orelse=None): """Test parts of the while loop. This test function will allow you to extract parts of a specific while loop and perform a set of tests specifically on these parts. A while loop generally consists of two parts: the condition test, :code:`test`, which is the condition that is tested each loop, and the :code:`body`. A for while can have a else part as well, :code:`orelse`, but this is almost never used.:: a = 10 while a < 5: print(a) a -= 1 Has :code:`a < 5` as the condition test and `print(i)` as the body. Args: index (int): index of the function call to be checked. Defaults to 1. test: this argument holds the part of code that will be ran to check the condition test of the while loop. It should be passed as a lambda expression or a function definition. The functions that are ran should be other pythonwhat test functions, and they will be tested specifically on only the condition test of the while loop. body: this argument holds the part of code that will be ran to check the body of the while loop. It should be passed as a lambda expression or a function definition. The functions that are ran should be other pythonwhat test functions, and they will be tested specifically on only the body of the while loop. orelse: this argument holds the part of code that will be ran to check the else part of the while loop. It should be passed as a lambda expression or a function definition. The functions that are ran should be other pythonwhat test functions, and they will be tested specifically on only the else part of the while loop. :Example: Student code:: a = 10 while a < 5: print(a) a -= 1 Solution code:: a = 20 while a < 5: print(a) a -= 1 SCT:: test_while_loop(1, test = test_expression_result({"a": 5}), body = test_expression_output({"a": 5})) This SCT will evaluate to True as condition test will have thes same result in student and solution code and `test_exression_output()` will pass on the body code. """ state = check_node(state, "whiles", index - 1, "{{ordinal}} while loop") multi(check_part(state, "test", "condition"), test) multi(check_part(state, "body", "body"), body) multi(check_part(state, "orelse", "else part"), orelse)
def test_for_loop(state, index=1, for_iter=None, body=None, orelse=None): """Test parts of the for loop. This test function will allow you to extract parts of a specific for loop and perform a set of tests specifically on these parts. A for loop consists of two parts: the sequence, `for_iter`, which is the values over which are looped, and the `body`. A for loop can have a else part as well, `orelse`, but this is almost never used.:: for i in range(10): print(i) Has :code:`range(10)` as the sequence and :code:`print(i)` as the body. Args: index (int): index of the function call to be checked. Defaults to 1. for_iter: this argument holds the part of code that will be ran to check the sequence of the for loop. It should be passed as a lambda expression or a function. The functions that are ran should be other pythonwhat test functions, and they will be tested specifically on only the sequence part of the for loop. body: this argument holds the part of code that will be ran to check the body of the for loop. It should be passed as a lambda expression or a function. The functions that are ran should be other pythonwhat test functions, and they will be tested specifically on only the body of the for loop. orelse: this argument holds the part of code that will be ran to check the else part of the for loop. It should be passed as a lambda expression or a function. The functions that are ran should be other pythonwhat test functions, and they will be tested specifically on only the else part of the for loop. :Example: Student code:: for i in range(10): print(i) Solution code:: for n in range(10): print(n) SCT:: test_for_loop(1, for_iter = test_function("range"), body = test_expression_output(context_val = [5]) This SCT will evaluate to True as the function :code:`range` is used in the sequence and the function :code:`test_exression_output()` will pass on the body code. """ state = check_node(state, "for_loops", index - 1, "{{ordinal}} for loop") multi(check_part(state, "iter", "sequence part"), for_iter) multi(check_part(state, "body", "body"), body) multi(check_part(state, "orelse", "else part"), orelse)
def test_function_definition( state, name, arg_names=True, arg_defaults=True, body=None, results=None, outputs=None, errors=None, not_called_msg=None, nb_args_msg=None, other_args_msg=None, arg_names_msg=None, arg_defaults_msg=None, wrong_result_msg=None, wrong_output_msg=None, no_error_msg=None, wrong_error_msg=None, ): """Test a function definition. This function helps you test a function definition. Generally four things can be tested: 1) The argument names of the function (including if the correct defaults are used) 2) The body of the functions (does it output correctly, are the correct functions used) 3) The return value with a certain input 4) The output value with a certain input 5) Whether certain inputs generate an error and what type of error Custom feedback messages can be set for all these parts, default messages are generated automatically if none are set. Args: name (str): the name of the function definition to be tested. arg_names (bool): if True, the argument names will be tested, if False they won't be tested. Defaults to True. arg_defaults (bool): if True, the default values of the arguments will be tested, if False they won't be tested. Defaults to True. body: this arguments holds the part of the code that will be ran to check the body of the function definition. It should be passed as a lambda expression or a function. The functions that are ran should be other pythonwhat test functions, and they will be tested specifically on only the body of the for loop. Defaults to None. results (list(list)): a list of lists representing arguments that should be passed to the defined function. These arguments are passed to the function in the student environment and the solution environment, the results (what's returned) are compared. outputs (list(list)): a list of lists representing arguments that should be passed to the defined function. These arguments are passed to the function in the student environment and the solution environment, the outpus are compared. errors (list(list)): a list of lists representing arguments that should be passed to the defined function. These arguments are passed to the function in the student environment and the solution environment, the errors they generate are compared. not_called_msg (str): message if the function is not defined. nb_args_msg (str): message if the number of arguments do not matched. arg_names_msg (str): message if the argument names do not match. arg_defaults_msg (str): message if the argument default values do not match. wrong_result_msg (str): message if one of the tested function calls' result did not match. wrong_output_msg (str): message if one of the tested functions calls' output did not match. no_error_msg (str): message if one of the tested function calls' result did not generate an error. wrong_error_msg (str): message if the error that one of the tested function calls generated did not match. :Example: Student code:: def shout( word, times = 3): shout_word = not_word + '???' print( shout_word ) return word * times Solution code:: def shout( word = 'help', times = 3 ): shout_word = word + '!!!' print( shout_word ) return word * times SCT:: test_function_definition('shout') # fail test_function_definition('shout', arg_defaults = False) # pass test_function_definition('shout', arg_defaults = False, # fail outputs = [('help')]) test_function_definition('shout', arg_defaults = False, # pass results = [('help', 2)]) test_function_definition('shout', args_defaults = False # pass body = test_function('print', args = []])) """ # what the function will be referred to as child = check_node(state, "function_defs", name, "definition of `{{index}}()`") test_args(child, arg_names, arg_defaults, nb_args_msg, arg_names_msg, arg_defaults_msg) multi(check_part(child, "body", ""), body) # Test function calls ----------------------------------------------------- for el in results or []: el = fix_format(el) call( child, el, "value", incorrect_msg=wrong_result_msg, error_msg=wrong_result_msg, argstr="`{}{}`".format(name, stringify(el)), ) for el in outputs or []: el = fix_format(el) call( child, el, "output", incorrect_msg=wrong_output_msg, error_msg=wrong_output_msg, argstr="`{}{}`".format(name, stringify(el)), ) for el in errors or []: el = fix_format(el) call( child, el, "error", incorrect_msg=wrong_error_msg, error_msg=no_error_msg, argstr="`{}{}`".format(name, stringify(el)), )
def test_if_else(state, index=1, test=None, body=None, orelse=None): """Test parts of the if statement. This test function will allow you to extract parts of a specific if statement and perform a set of tests specifically on these parts. A for loop consists of three potential parts: the condition test, :code:`test`, which specifies the condition of the if statement, the :code:`body`, which is what's executed if the condition is True and a else part, :code:`orelse`, which will be executed if the condition is not True.:: if 5 == 3: print("success") else: print("fail") Has :code:`5 == 3` as the condition test, :code:`print("success")` as the body and :code:`print("fail")` as the else part. Args: index (int): index of the function call to be checked. Defaults to 1. test: this argument holds the part of code that will be ran to check the condition test of the if statement. It should be passed as a lambda expression or a function definition. The functions that are ran should be other pythonwhat test functions, and they will be tested specifically on only the condition test of the if statement. body: this argument holds the part of code that will be ran to check the body of the if statement. It should be passed as a lambda expression or a function definition. The functions that are ran should be other pythonwhat test functions, and they will be tested specifically on only the body of the if statement. orelse: this argument holds the part of code that will be ran to check the else part of the if statement. It should be passed as a lambda expression or a function definition. The functions that are ran should be other pythonwhat test functions, and they will be tested specifically on only the else part of the if statement. :Example: Student code:: a = 12 if a > 3: print('test %d' % a) Solution code:: a = 4 if a > 3: print('test %d' % a) SCT:: test_if_else(1, body = test_expression_output( extra_env = { 'a': 5 } incorrect_msg = "Print out the correct things")) This SCT will pass as :code:`test_expression_output()` is ran on the body of the if statement and it will output the same thing in the solution as in the student code. """ state = check_node(state, "if_elses", index - 1, typestr="{{ordinal}} if expression") multi(check_part(state, "test", "condition"), test) multi(check_part(state, "body", "body"), body) multi(check_part(state, "orelse", "else part"), orelse)