Skip to content

Commit

Permalink
Batch without input (#9)
Browse files Browse the repository at this point in the history
* add BatchNoInput task type

* better name

---------

Co-authored-by: Kevin Atienza <[email protected]>
  • Loading branch information
2 people authored and verngutz committed Jun 8, 2024
1 parent e92d452 commit e2e0d73
Show file tree
Hide file tree
Showing 2 changed files with 361 additions and 0 deletions.
360 changes: 360 additions & 0 deletions cms/grading/tasktypes/BatchNoInput.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,360 @@
#!/usr/bin/env python3

# Contest Management System - http://cms-dev.github.io/
# Copyright © 2010-2015 Giovanni Mascellani <[email protected]>
# Copyright © 2010-2018 Stefano Maggiolo <[email protected]>
# Copyright © 2010-2012 Matteo Boscariol <[email protected]>
# Copyright © 2012-2014 Luca Wehrstedt <[email protected]>
# Copyright © 2017 Myungwoo Chun <[email protected]>
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU Affero General Public License as
# published by the Free Software Foundation, either version 3 of the
# License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU Affero General Public License for more details.
#
# You should have received a copy of the GNU Affero General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.

import logging
import os

from cms.db import Executable
from cms.grading.ParameterTypes import ParameterTypeCollection, \
ParameterTypeChoice, ParameterTypeString
from cms.grading.languagemanager import LANGUAGES, get_language
from cms.grading.steps import compilation_step, evaluation_step, \
human_evaluation_message
from . import TaskType, \
check_executables_number, check_files_number, check_manager_present, \
create_sandbox, delete_sandbox, eval_output, is_manager_for_compilation


logger = logging.getLogger(__name__)


# Dummy function to mark translatable string.
def N_(message):
return message


class BatchNoInput(TaskType):
"""Task type class for a unique standalone submission source, with
comparator (or not), and input is not given to the user!
Parameters needs to be a list of three elements.
The first element is 'grader' or 'alone': in the first
case, the source file is to be compiled with a provided piece of
software ('grader'); in the other by itself.
The second element is a 2-tuple of the input file name and output file
name. The input file may be '' to denote stdin, and similarly the
output filename may be '' to denote stdout.
The third element is 'diff' or 'comparator' and says whether the
output is compared with a simple diff algorithm or using a
comparator.
Note: the first element is used only in the compilation step; the
others only in the evaluation step.
A comparator can read argv[1], argv[2], argv[3] (respectively,
input, correct output and user output) and should write the
outcome to stdout and the text to stderr.
"""
# Codename of the checker, if it is used.
CHECKER_CODENAME = "checker"
# Basename of the grader, used in the manager filename and as the main
# class in languages that require us to specify it.
GRADER_BASENAME = "grader"
# Default input and output filenames when not provided as parameters.
DEFAULT_INPUT_FILENAME = "input.txt"
DEFAULT_OUTPUT_FILENAME = "output.txt"

# Constants used in the parameter definition.
OUTPUT_EVAL_DIFF = "diff"
OUTPUT_EVAL_CHECKER = "comparator"
COMPILATION_ALONE = "alone"
COMPILATION_GRADER = "grader"

# Other constants to specify the task type behaviour and parameters.
ALLOW_PARTIAL_SUBMISSION = False

_COMPILATION = ParameterTypeChoice(
"Compilation",
"compilation",
"",
{COMPILATION_ALONE: "Submissions are self-sufficient",
COMPILATION_GRADER: "Submissions are compiled with a grader"})

_USE_FILE = ParameterTypeCollection(
"I/O (blank for stdin/stdout)",
"io",
"",
[
ParameterTypeString("Input file", "inputfile", ""),
ParameterTypeString("Output file", "outputfile", ""),
])

_EVALUATION = ParameterTypeChoice(
"Output evaluation",
"output_eval",
"",
{OUTPUT_EVAL_DIFF: "Outputs compared with white diff",
OUTPUT_EVAL_CHECKER: "Outputs are compared by a comparator"})

ACCEPTED_PARAMETERS = [_COMPILATION, _USE_FILE, _EVALUATION]

@property
def name(self):
"""See TaskType.name."""
# TODO add some details if a grader/comparator is used, etc...
return "Batch (no input)"

def __init__(self, parameters):
super().__init__(parameters)

# Data in the parameters.
self.compilation = self.parameters[0]
self.input_filename, self.output_filename = self.parameters[1]
self.output_eval = self.parameters[2]

# Actual input and output are the files used to store input and
# where the output is checked, regardless of using redirects or not.
self._actual_input = self.input_filename
self._actual_output = self.output_filename
if len(self.input_filename) == 0:
self._actual_input = self.DEFAULT_INPUT_FILENAME
if len(self.output_filename) == 0:
self._actual_output = self.DEFAULT_OUTPUT_FILENAME

def get_compilation_commands(self, submission_format):
"""See TaskType.get_compilation_commands."""
codenames_to_compile = []
if self._uses_grader():
codenames_to_compile.append(self.GRADER_BASENAME + ".%l")
codenames_to_compile.extend(submission_format)
res = dict()
for language in LANGUAGES:
source_ext = language.source_extension
executable_filename = self._executable_filename(submission_format,
language)
res[language.name] = language.get_compilation_commands(
[codename.replace(".%l", source_ext)
for codename in codenames_to_compile],
executable_filename)
return res

def get_user_managers(self):
"""See TaskType.get_user_managers."""
# In case the task uses a grader, we let the user provide their own
# grader (which is usually a simplified grader provided by the admins).
if self._uses_grader():
return [self.GRADER_BASENAME + ".%l"]
else:
return []

def get_auto_managers(self):
"""See TaskType.get_auto_managers."""
return []

def _uses_grader(self):
return self.compilation == self.COMPILATION_GRADER

def _uses_checker(self):
return self.output_eval == self.OUTPUT_EVAL_CHECKER

@staticmethod
def _executable_filename(codenames, language):
"""Return the chosen executable name computed from the codenames.
codenames ([str]): submission format or codename of submitted files,
may contain %l.
language (Language): the programming language of the submission.
return (str): a deterministic executable name.
"""
name = "_".join(sorted(codename.replace(".%l", "")
for codename in codenames))
return name + language.executable_extension

def compile(self, job, file_cacher):
"""See TaskType.compile."""
language = get_language(job.language)
source_ext = language.source_extension

if not check_files_number(job, 1, or_more=True):
return

# Create the list of filenames to be passed to the compiler. If we use
# a grader, it needs to be in first position in the command line, and
# we check that it exists.
filenames_to_compile = []
filenames_and_digests_to_get = {}
# The grader, that must have been provided (copy and add to
# compilation).
if self._uses_grader():
grader_filename = self.GRADER_BASENAME + source_ext
if not check_manager_present(job, grader_filename):
return
filenames_to_compile.append(grader_filename)
filenames_and_digests_to_get[grader_filename] = \
job.managers[grader_filename].digest
# User's submitted file(s) (copy and add to compilation).
for codename, file_ in job.files.items():
filename = codename.replace(".%l", source_ext)
filenames_to_compile.append(filename)
filenames_and_digests_to_get[filename] = file_.digest
# Any other useful manager (just copy).
for filename, manager in job.managers.items():
if is_manager_for_compilation(filename, language):
filenames_and_digests_to_get[filename] = manager.digest

# Prepare the compilation command.
executable_filename = self._executable_filename(job.files.keys(),
language)
commands = language.get_compilation_commands(
filenames_to_compile, executable_filename)

# Create the sandbox.
sandbox = create_sandbox(file_cacher, name="compile")
job.sandboxes.append(sandbox.get_root_path())

# Copy required files in the sandbox (includes the grader if present).
for filename, digest in filenames_and_digests_to_get.items():
sandbox.create_file_from_storage(filename, digest)

# Run the compilation.
box_success, compilation_success, text, stats = \
compilation_step(sandbox, commands)

# Retrieve the compiled executables.
job.success = box_success
job.compilation_success = compilation_success
job.text = text
job.plus = stats
if box_success and compilation_success:
digest = sandbox.get_file_to_storage(
executable_filename,
"Executable %s for %s" % (executable_filename, job.info))
job.executables[executable_filename] = \
Executable(executable_filename, digest)

# Cleanup.
delete_sandbox(sandbox, job.success, job.keep_sandbox)

def evaluate(self, job, file_cacher):
"""See TaskType.evaluate."""
if not check_executables_number(job, 1):
return

# Prepare the execution
executable_filename = next(iter(job.executables.keys()))
language = get_language(job.language)
main = self.GRADER_BASENAME if self._uses_grader() \
else os.path.splitext(executable_filename)[0]
commands = language.get_evaluation_commands(
executable_filename, main=main)
executables_to_get = {
executable_filename: job.executables[executable_filename].digest
}
files_to_get = {
self._actual_input: job.input
}

# Check which redirect we need to perform, and in case we don't
# manage the output via redirect, the submission needs to be able
# to write on it.
files_allowing_write = []
stdin_redirect = None
stdout_redirect = None
if len(self.input_filename) == 0:
stdin_redirect = self._actual_input
if len(self.output_filename) == 0:
stdout_redirect = self._actual_output
else:
files_allowing_write.append(self._actual_output)

# Create the sandbox
sandbox = create_sandbox(file_cacher, name="evaluate")
job.sandboxes.append(sandbox.get_root_path())

# Put the required files into the sandbox
for filename, digest in executables_to_get.items():
sandbox.create_file_from_storage(filename, digest, executable=True)
for filename, digest in files_to_get.items():
sandbox.create_file_from_storage(filename, digest)

# Actually performs the execution
box_success, evaluation_success, stats = evaluation_step(
sandbox,
commands,
job.time_limit,
job.memory_limit,
writable_files=files_allowing_write,
stdin_redirect=None,#stdin_redirect,
stdout_redirect=stdout_redirect,
multiprocess=job.multithreaded_sandbox)

outcome = None
text = None

# Error in the sandbox: nothing to do!
if not box_success:
pass

# Contestant's error: the marks won't be good
elif not evaluation_success:
outcome = 0.0
text = human_evaluation_message(stats)
if job.get_output:
job.user_output = None

# Otherwise, advance to checking the solution
else:

# Check that the output file was created
if not sandbox.file_exists(self._actual_output):
outcome = 0.0
text = [N_("Evaluation didn't produce file %s"),
self._actual_output]
if job.get_output:
job.user_output = None

else:
# If asked so, put the output file into the storage.
if job.get_output:
job.user_output = sandbox.get_file_to_storage(
self._actual_output,
"Output file in job %s" % job.info,
trunc_len=100 * 1024)

# If just asked to execute, fill text and set dummy outcome.
if job.only_execution:
outcome = 0.0
text = [N_("Execution completed successfully")]

# Otherwise evaluate the output file.
else:
box_success, outcome, text = eval_output(
file_cacher, job,
self.CHECKER_CODENAME
if self._uses_checker() else None,
user_output_path=sandbox.relative_path(
self._actual_output),
user_output_filename=self.output_filename)

# Fill in the job with the results.
job.success = box_success
job.outcome = str(outcome) if outcome is not None else None
job.text = text
job.plus = stats

delete_sandbox(sandbox, job.success, job.keep_sandbox)
1 change: 1 addition & 0 deletions setup.py
Original file line number Diff line number Diff line change
Expand Up @@ -171,6 +171,7 @@ def run(self):
],
"cms.grading.tasktypes": [
"Batch=cms.grading.tasktypes.Batch:Batch",
"BatchNoInput=cms.grading.tasktypes.BatchNoInput:BatchNoInput",
"Communication=cms.grading.tasktypes.Communication:Communication",
"OutputOnly=cms.grading.tasktypes.OutputOnly:OutputOnly",
"TwoSteps=cms.grading.tasktypes.TwoSteps:TwoSteps",
Expand Down

0 comments on commit e2e0d73

Please sign in to comment.