grade
is a tool for Computer Science students and instructors to test student projects for correctness. Its features include:
- Clone all student repos based on a list of Github IDs
- Build all student repos using
make
- Run all student repos against instructor-provided input files
- Score actual output vs. expected using an instructor-provided rubric.
- Integration with the Digital circuit simulation tool
- Automated upload of results to Canvas
- Requires python3 and pip3
$ sudo apt install python3-pip
- Clone the
autograder
repo$ cd ~ $ git clone git@github.com:/phpeterson-usf/autograder.git $ cd autograder
- Install python modules (mainly
tomlkit
andrequests
)$ pip3 install -r requirements.txt
- Edit
~/.bash_profile
(on Linux or Git Bash on Windows) or~/.zshrc
(on macOS) to include the path tograde
export PATH=~/autograder:$PATH
- Use the
source
command on that file to update your environment$ source ~/.zshrc
- Clone your class's tests repo. Use the right one for your class - these are just examples.
$ cd ~ $ git clone git@github.com:/cs315-21s/tests.git $ git clone git@github.com:/USF-CS631-S21/tests.git
- You can test a project in the current directory like this
note that
$ cd ~/project02-phpeterson-usf $ grade test
grade
will infer the project name based on your current directory. grade
will create a config file in~/.config/grade/config.toml
containing its default settings. You do not need to modify this file unless you need non-default settings- Example settings
grade
assumes that you authenticate to Github usingssh
. If you authenticate to github using http, you could changeconfig.toml
like this[Git] credentials = "https" # default is"ssh"
grade
assumes that test cases are in~/tests/
. If you put the your test case repo in a different directory, you could changeconfig.toml
to point to that directory[Test] tests_path = "~/myclass/tests" # default is "~/tests"
- Add your Github Classroom organization and a list of students to
~/.config/grade/config.toml
[Git] org = "cs315-f22" [Config] students = [ "phpeterson-usf", "gdbenson", ]
- You can clone all of your students' repos to your machine.
grade
will create a directory for each repo in your current working directory$ grade clone -p project02 ./project02-phpeterson-usf ./project02-gdbenson
grade clone
can take a student GitHub username, or several of them$ grade clone -p project02 -s phpeterson-usf gdbenson
grade clone
can accept a date, or date and time, for your project deadlineNotes:$ grade clone -p project02 --date '2021-10-14' $ grade clone -p project02 --date '2021-10-14 08:00:00'
- If no time is given, the script assumes midnight
00:00:00
- The script uses
git rev-list
to find the commit hash of the last commit on the default branch ('main' or 'master') before that date - The script uses
git checkout
to checkout the repo as of that hash. Keep in mind this leaves the repo in a "detached HEAD" state - If you want to checkout the tip of the default branch again, you can use
grade pull
- If no time is given, the script assumes midnight
- After developing test cases for your projects (see below), you can test all of your students' repos in batch. Passing test cases are shown in green with a '+' and failing test cases are shown in red with a '-'
$ grade class -p project02 project02-phpeterson-usf 01- 02+ 5/10 project02-gdbenson 01+ 02+ 10/10
- Each test case can pass or fail. The score is shown as the total earned/total available, based on the
rubric
field in each test case - Optional config: If you want to put
config.toml
into another location (perhaps a per-semester directory), you can use the shell environment variableGRADE_CONFIG_DIR
which causes the grade script to look forconfig.toml
in the named directory. For example,~/.bashrc
might containexport GRADE_CONFIG_DIR=~/cs521-s24
or whatever directory is useful to you.
- Instructors must create their own repo for the test cases for the class projects. Students will clone and pull this repo. We create this repo in the Github Classroom Organization, but that's up to you.
- Test cases for each project are expressed in TOML
- Test case inputs are a list of strings for each command-line flag and value. The keyword
$project
will be substituted for the name of your project.[[tests]] name = "01" input = ["./$project", "-e", "1 + 1"] expected = "2" rubric = 5 [[tests]] name = "02" input = ["./$project", "-e", "10", "-b", "16"] expected = "0x0000000A" rubric = 5
- Note that this usage of
$project
is flexible enough for projects which are run using an interpreter like Java or Python[[tests]] input = ["python3", "$project.py"]
- Test cases can have input files in your
tests
repo using the keyword$project_tests
, which will be substituted for$testspath/$project/
. In this example, substitution gives the input file as$testspath/project02/testinput.txt
[[tests]] name = "03" input = ["./$project", "$project_tests/testinput.txt"]
- Test case output can be recorded from
stdout
or from a file. In this example, the contents of the file04.txt
will be compared to theexpected
field of the test case.If[[tests]] name = "04" output = "04.txt" input = ["./$project", "-o", "04.txt"]
output
is not given,grade
defaults tostdout
- Comparing expected output with actual output is case-insentitive by default. If you want case-sensitive comparison, you can set that in a test case
[[tests]] name = to_upper input = input = ["./$project", "foObAr1"] expected = "FOOBAR1" case_sensitive = true
- Autograder assumes that the student repo will be the working directory for testing. If your project requires a subdirectory within all student repos, you can add that in the
[project]
settings in the test case TOML file[project] subdir = "xv6"
- Autograder will wait for 60 seconds for a program to finish before concluding that the program is in an infinite loop and killing it. If you need to wait longer than 60 seconds, you can change that setting in the
[project]
section of the test case TOML file[project] timeout = 120 # two minutes
- Autograder will collect at most 10,000 bytes of output before concluding that the program is in an infinite loop and killing it.
grade
supports these command-line parameters
-
-d/--date
is the date to use for grade clone -
-e/--exec
provide commands to execute (e.g.git pull; make clean
) -
-g/--github-action
tellsgrade class
to get the test result fromapi.github.com
rather than local testing -
-n/--name
withgrade test
runs one named test case, rather than all of them -
-p/--project
is the name of the project, which is substituted into repo names and test case inputs -
-v/--verbose
shows expected and actual for failing test cases -
-vv/--very-verbose
shows expected and actual for all test casesThe command-line format is
argparse
-style, with no "=". These two commands are equivalent:$ cd ~/project02-jsmith $ grade test -p project02 $ grade test --project project02
- In order to map GitHub usernames to Canvas login IDs, you need to provide a mapping file.
We ask students to fill out a Google Form which produces a Google Sheets spreadsheet. If you
download that spreadsheet to a local CSV file, you can give the mapping configuration in
~/.config/grade/config.toml
[CanvasMapper] map_path = "~/github-to-canvas.csv" # CSV file which maps GitHub username to Canvas SIS Login ID github_col_name = "GitHub" # Name of the CSV column which contains the GitHub username login_col_name = "SIS Login ID" # Name of the CSV column which contains the Canvas SIS Login ID
- After you run
grade class -p lab01
the test results will be stored in a JSON file, e.g../lab01.json
- You can subsequently run
grade upload -p lab01
to upload the results to Canvas - The JSON file contains the aggregate score for the repo (e.g. 80 out of 100 pts) and a submission comment showing which tests passed and failed. The comment will also be uploaded.
- The name you use for the
project
ingrade
must match the name of the assignment in Canvas, case-sensitively. - Since the Canvas REST API for submissions does not know about assignment groups, I recommend you create the assignment in Canvas before running
grade upload
. Otherwise Canvas will create a new assignment outside your structure for assignment groups grade upload
manipulates only the named Canvas assignment. It does not mimic the uploading of CSV files shown in the Canvas web UI.- In order to upload, you must add these attributes to the
[Canvas]
section in~/.config/grade/config.toml
[Canvas] host_name = "canvas.instructure.com" # your institution may have a test instance of Canvas access_token = "xxx" # create an access token in Profile | Settings in Canvas course_name = "Your long course name" # e.g. 'Computer Architecture - 01 (Spring 2022)'
- If Canvas isn't working for you, try the
-v
command-line flag, which will print the results of each Canvas REST API
- Digital has test case components which can test a circuit using pre-defined inputs and outputs. See Digital's documentation for scripted testing examples.
grade
leverages its ability to loop over the student repos, calling Java on the command line to use Digital's test case components. Example test case:[project] build = "none" strip_output = """ Use default settings! """ [[tests]] name = "1-bit-full-adder" input = ["java", "-cp", "$digital", "CLI", "test", "1-bit-full-adder.dig", "-tests", "$project_tests/1bfa_test.dig"] expected = """ 1bfa_test: passed """ rubric = 1
grade
assumes that the path to the Digital JAR file is~/Digital/Digital.jar
. If you need a different setting, you can change it in the[Test]
section of~/.config/grade/config.toml
:[Test] digital_path = "~/myclass/Digital/Digital.jar"
- If you use GitHub Actions to do the testing, autograder can download the results for each student repo and upload the class results to Canvas
- If you use
grade class
with the flag-g/--github-action
, autograder can use the GitHub REST API to download the results into a JSON filegrade class -p project01 -g
- Once we have the class results in a JSON file, you can upload them to Canvas
grade upload -p project01
- To configure autograder to download the results of GitHub Actions, edit
~/.config/grade/config.toml
to include these settings[Github] access_token = "xxx" # create in your Github settings | Developer Settings | Tokens (classic)