execute test files concurrently
This commit is contained in:
@ -1,17 +1,30 @@
|
|||||||
|
require 'concurrent/future'
|
||||||
|
|
||||||
module SubmissionScoring
|
module SubmissionScoring
|
||||||
def execute_test_files(submission)
|
def collect_test_results(submission)
|
||||||
submission.collect_files.select(&:teacher_defined_test?).map do |file|
|
submission.collect_files.select(&:teacher_defined_test?).map do |file|
|
||||||
output = @docker_client.execute_test_command(submission, file.name_with_extension)
|
future = Concurrent::Future.execute do
|
||||||
output.merge!(@assessor.assess(output))
|
assessor = Assessor.new(execution_environment: submission.execution_environment)
|
||||||
output.merge!(filename: file.name_with_extension, message: output[:score] == Assessor::MAXIMUM_SCORE ? I18n.t('exercises.implement.default_feedback') : file.feedback_message, weight: file.weight)
|
output = execute_test_file(file, submission)
|
||||||
|
output.merge!(assessor.assess(output))
|
||||||
|
output.merge!(filename: file.name_with_extension, message: feedback_message(file, output[:score]), weight: file.weight)
|
||||||
|
end
|
||||||
|
future.value
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
private :execute_test_files
|
private :collect_test_results
|
||||||
|
|
||||||
|
def execute_test_file(file, submission)
|
||||||
|
DockerClient.new(execution_environment: file.context.execution_environment, user: current_user).execute_test_command(submission, file.name_with_extension)
|
||||||
|
end
|
||||||
|
private :execute_test_file
|
||||||
|
|
||||||
|
def feedback_message(file, score)
|
||||||
|
score == Assessor::MAXIMUM_SCORE ? I18n.t('exercises.implement.default_feedback') : file.feedback_message
|
||||||
|
end
|
||||||
|
|
||||||
def score_submission(submission)
|
def score_submission(submission)
|
||||||
@assessor = Assessor.new(execution_environment: submission.execution_environment)
|
outputs = collect_test_results(submission)
|
||||||
@docker_client = DockerClient.new(execution_environment: submission.execution_environment, user: current_user)
|
|
||||||
outputs = execute_test_files(submission)
|
|
||||||
score = outputs.map { |output| output[:score] * output[:weight] }.reduce(:+)
|
score = outputs.map { |output| output[:score] * output[:weight] }.reduce(:+)
|
||||||
submission.update(score: score)
|
submission.update(score: score)
|
||||||
outputs
|
outputs
|
||||||
|
@ -5,35 +5,29 @@ class Controller < AnonymousController
|
|||||||
end
|
end
|
||||||
|
|
||||||
describe SubmissionScoring do
|
describe SubmissionScoring do
|
||||||
before(:all) do
|
|
||||||
@submission = FactoryGirl.create(:submission, cause: 'submit')
|
|
||||||
end
|
|
||||||
|
|
||||||
let(:controller) { Controller.new }
|
let(:controller) { Controller.new }
|
||||||
|
before(:all) { @submission = FactoryGirl.create(:submission, cause: 'submit') }
|
||||||
before(:each) { controller.instance_variable_set(:@current_user, FactoryGirl.create(:external_user)) }
|
before(:each) { controller.instance_variable_set(:@current_user, FactoryGirl.create(:external_user)) }
|
||||||
|
|
||||||
describe '#score_submission' do
|
describe '#collect_test_results' do
|
||||||
let(:score_submission) { proc { controller.score_submission(@submission) } }
|
after(:each) { controller.send(:collect_test_results, @submission) }
|
||||||
before(:each) { score_submission.call }
|
|
||||||
|
|
||||||
it 'assigns @assessor' do
|
it 'executes every teacher-defined test file' do
|
||||||
expect(controller.instance_variable_get(:@assessor)).to be_an(Assessor)
|
|
||||||
end
|
|
||||||
|
|
||||||
it 'assigns @docker_client' do
|
|
||||||
expect(controller.instance_variable_get(:@docker_client)).to be_a(DockerClient)
|
|
||||||
end
|
|
||||||
|
|
||||||
it 'executes the teacher-defined test cases' do
|
|
||||||
@submission.collect_files.select(&:teacher_defined_test?).each do |file|
|
@submission.collect_files.select(&:teacher_defined_test?).each do |file|
|
||||||
expect_any_instance_of(DockerClient).to receive(:execute_test_command).with(@submission, file.name_with_extension).and_return({})
|
expect(controller).to receive(:execute_test_file).with(file, @submission).and_return({})
|
||||||
end
|
end
|
||||||
score_submission.call
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
describe '#score_submission' do
|
||||||
|
after(:each) { controller.score_submission(@submission) }
|
||||||
|
|
||||||
|
it 'collects the test results' do
|
||||||
|
expect(controller).to receive(:collect_test_results).and_return([])
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'updates the submission' do
|
it 'assigns a score to the submissions' do
|
||||||
expect(@submission).to receive(:update).with(score: anything)
|
expect(@submission).to receive(:update).with(score: anything)
|
||||||
score_submission.call
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
@ -210,7 +210,7 @@ describe ExercisesController do
|
|||||||
|
|
||||||
before(:each) do
|
before(:each) do
|
||||||
allow_any_instance_of(Submission).to receive(:normalized_score).and_return(1)
|
allow_any_instance_of(Submission).to receive(:normalized_score).and_return(1)
|
||||||
expect(controller).to receive(:execute_test_files).and_return([{score: 1, weight: 1}])
|
expect(controller).to receive(:collect_test_results).and_return([{score: 1, weight: 1}])
|
||||||
expect(controller).to receive(:score_submission).and_call_original
|
expect(controller).to receive(:score_submission).and_call_original
|
||||||
end
|
end
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user