Skip to content
Projects
Groups
Snippets
Help
This project
Loading...
Sign in / Register
Toggle navigation
E
edx-platform
Overview
Overview
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
edx
edx-platform
Commits
9a631fe4
Commit
9a631fe4
authored
May 29, 2013
by
Ned Batchelder
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
All uses of safe_exec need to get the correct random seed.
parent
92aba633
Hide whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
96 additions
and
13 deletions
+96
-13
common/lib/capa/capa/responsetypes.py
+28
-4
common/lib/capa/capa/tests/test_responsetypes.py
+68
-9
No files found.
common/lib/capa/capa/responsetypes.py
View file @
9a631fe4
...
...
@@ -288,7 +288,13 @@ class LoncapaResponse(object):
}
try
:
safe_exec
.
safe_exec
(
code
,
globals_dict
,
python_path
=
self
.
context
[
'python_path'
],
slug
=
self
.
id
)
safe_exec
.
safe_exec
(
code
,
globals_dict
,
python_path
=
self
.
context
[
'python_path'
],
slug
=
self
.
id
,
random_seed
=
self
.
context
[
'seed'
],
)
except
Exception
as
err
:
msg
=
'Error
%
s in evaluating hint function
%
s'
%
(
err
,
hintfn
)
msg
+=
"
\n
See XML source line
%
s"
%
getattr
(
...
...
@@ -973,7 +979,13 @@ class CustomResponse(LoncapaResponse):
'ans'
:
ans
,
}
globals_dict
.
update
(
kwargs
)
safe_exec
.
safe_exec
(
code
,
globals_dict
,
python_path
=
self
.
context
[
'python_path'
],
slug
=
self
.
id
)
safe_exec
.
safe_exec
(
code
,
globals_dict
,
python_path
=
self
.
context
[
'python_path'
],
slug
=
self
.
id
,
random_seed
=
self
.
context
[
'seed'
],
)
return
globals_dict
[
'cfn_return'
]
return
check_function
...
...
@@ -1090,7 +1102,13 @@ class CustomResponse(LoncapaResponse):
# exec the check function
if
isinstance
(
self
.
code
,
basestring
):
try
:
safe_exec
.
safe_exec
(
self
.
code
,
self
.
context
,
cache
=
self
.
system
.
cache
,
slug
=
self
.
id
)
safe_exec
.
safe_exec
(
self
.
code
,
self
.
context
,
cache
=
self
.
system
.
cache
,
slug
=
self
.
id
,
random_seed
=
self
.
context
[
'seed'
],
)
except
Exception
as
err
:
self
.
_handle_exec_exception
(
err
)
...
...
@@ -1814,7 +1832,13 @@ class SchematicResponse(LoncapaResponse):
]
self
.
context
.
update
({
'submission'
:
submission
})
try
:
safe_exec
.
safe_exec
(
self
.
code
,
self
.
context
,
cache
=
self
.
system
.
cache
,
slug
=
self
.
id
)
safe_exec
.
safe_exec
(
self
.
code
,
self
.
context
,
cache
=
self
.
system
.
cache
,
slug
=
self
.
id
,
random_seed
=
self
.
context
[
'seed'
],
)
except
Exception
as
err
:
msg
=
'Error
%
s in evaluating SchematicResponse'
%
err
raise
ResponseError
(
msg
)
...
...
common/lib/capa/capa/tests/test_responsetypes.py
View file @
9a631fe4
...
...
@@ -640,6 +640,23 @@ class StringResponseTest(ResponseTest):
correct_map
=
problem
.
grade_answers
(
input_dict
)
self
.
assertEquals
(
correct_map
.
get_hint
(
'1_2_1'
),
"Hello??"
)
def
test_hint_function_randomization
(
self
):
# The hint function should get the seed from the problem.
problem
=
self
.
build_problem
(
answer
=
"1"
,
hintfn
=
"gimme_a_random_hint"
,
script
=
textwrap
.
dedent
(
"""
def gimme_a_random_hint(answer_ids, student_answers, new_cmap, old_cmap):
answer = str(random.randint(0, 1e9))
new_cmap.set_hint_and_mode(answer_ids[0], answer, "always")
"""
)
)
correct_map
=
problem
.
grade_answers
({
'1_2_1'
:
'2'
})
hint
=
correct_map
.
get_hint
(
'1_2_1'
)
r
=
random
.
Random
(
problem
.
seed
)
self
.
assertEqual
(
hint
,
str
(
r
.
randint
(
0
,
1e9
)))
class
CodeResponseTest
(
ResponseTest
):
from
response_xml_factory
import
CodeResponseXMLFactory
...
...
@@ -948,7 +965,6 @@ class CustomResponseTest(ResponseTest):
xml_factory_class
=
CustomResponseXMLFactory
def
test_inline_code
(
self
):
# For inline code, we directly modify global context variables
# 'answers' is a list of answers provided to us
# 'correct' is a list we fill in with True/False
...
...
@@ -961,15 +977,14 @@ class CustomResponseTest(ResponseTest):
self
.
assert_grade
(
problem
,
'0'
,
'incorrect'
)
def
test_inline_message
(
self
):
# Inline code can update the global messages list
# to pass messages to the CorrectMap for a particular input
# The code can also set the global overall_message (str)
# to pass a message that applies to the whole response
inline_script
=
textwrap
.
dedent
(
"""
messages[0] = "Test Message"
overall_message = "Overall message"
"""
)
messages[0] = "Test Message"
overall_message = "Overall message"
"""
)
problem
=
self
.
build_problem
(
answer
=
inline_script
)
input_dict
=
{
'1_2_1'
:
'0'
}
...
...
@@ -983,8 +998,19 @@ class CustomResponseTest(ResponseTest):
overall_msg
=
correctmap
.
get_overall_message
()
self
.
assertEqual
(
overall_msg
,
"Overall message"
)
def
test_function_code_single_input
(
self
):
def
test_inline_randomization
(
self
):
# Make sure the seed from the problem gets fed into the script execution.
inline_script
=
"""messages[0] = str(random.randint(0, 1e9))"""
problem
=
self
.
build_problem
(
answer
=
inline_script
)
input_dict
=
{
'1_2_1'
:
'0'
}
correctmap
=
problem
.
grade_answers
(
input_dict
)
input_msg
=
correctmap
.
get_msg
(
'1_2_1'
)
r
=
random
.
Random
(
problem
.
seed
)
self
.
assertEqual
(
input_msg
,
str
(
r
.
randint
(
0
,
1e9
)))
def
test_function_code_single_input
(
self
):
# For function code, we pass in these arguments:
#
# 'expect' is the expect attribute of the <customresponse>
...
...
@@ -1212,6 +1238,29 @@ class CustomResponseTest(ResponseTest):
with
self
.
assertRaises
(
ResponseError
):
problem
.
grade_answers
({
'1_2_1'
:
'42'
})
def
test_setup_randomization
(
self
):
# Ensure that the problem setup script gets the random seed from the problem.
script
=
textwrap
.
dedent
(
"""
num = random.randint(0, 1e9)
"""
)
problem
=
self
.
build_problem
(
script
=
script
)
r
=
random
.
Random
(
problem
.
seed
)
self
.
assertEqual
(
r
.
randint
(
0
,
1e9
),
problem
.
context
[
'num'
])
def
test_check_function_randomization
(
self
):
# The check function should get random-seeded from the problem.
script
=
textwrap
.
dedent
(
"""
def check_func(expect, answer_given):
return {'ok': True, 'msg': str(random.randint(0, 1e9))}
"""
)
problem
=
self
.
build_problem
(
script
=
script
,
cfn
=
"check_func"
,
expect
=
"42"
)
input_dict
=
{
'1_2_1'
:
'42'
}
correct_map
=
problem
.
grade_answers
(
input_dict
)
msg
=
correct_map
.
get_msg
(
'1_2_1'
)
r
=
random
.
Random
(
problem
.
seed
)
self
.
assertEqual
(
msg
,
str
(
r
.
randint
(
0
,
1e9
)))
def
test_module_imports_inline
(
self
):
'''
Check that the correct modules are available to custom
...
...
@@ -1275,7 +1324,6 @@ class SchematicResponseTest(ResponseTest):
xml_factory_class
=
SchematicResponseXMLFactory
def
test_grade
(
self
):
# Most of the schematic-specific work is handled elsewhere
# (in client-side JavaScript)
# The <schematicresponse> is responsible only for executing the
...
...
@@ -1290,7 +1338,7 @@ class SchematicResponseTest(ResponseTest):
# The actual dictionary would contain schematic information
# sent from the JavaScript simulation
submission_dict
=
{
'test'
:
't
est
'
}
submission_dict
=
{
'test'
:
't
he_answer
'
}
input_dict
=
{
'1_2_1'
:
json
.
dumps
(
submission_dict
)}
correct_map
=
problem
.
grade_answers
(
input_dict
)
...
...
@@ -1299,8 +1347,19 @@ class SchematicResponseTest(ResponseTest):
# is what we expect)
self
.
assertEqual
(
correct_map
.
get_correctness
(
'1_2_1'
),
'correct'
)
def
test_script_exception
(
self
):
def
test_check_function_randomization
(
self
):
# The check function should get a random seed from the problem.
script
=
"correct = ['correct' if (submission[0]['num'] == random.randint(0, 1e9)) else 'incorrect']"
problem
=
self
.
build_problem
(
answer
=
script
)
r
=
random
.
Random
(
problem
.
seed
)
submission_dict
=
{
'num'
:
r
.
randint
(
0
,
1e9
)}
input_dict
=
{
'1_2_1'
:
json
.
dumps
(
submission_dict
)}
correct_map
=
problem
.
grade_answers
(
input_dict
)
self
.
assertEqual
(
correct_map
.
get_correctness
(
'1_2_1'
),
'correct'
)
def
test_script_exception
(
self
):
# Construct a script that will raise an exception
script
=
"raise Exception('test')"
problem
=
self
.
build_problem
(
answer
=
script
)
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment