Skip to main content

A Python ToolBox for CourseGrading platform.

Project description

PyGrading

CourseGrading(希冀)计算机专业课一体化在线平台开发用Python工具包。包含通用评测内核创建以及HTML标签生成工具。

Official Site GitHub stars Pypi package GitHub issues GitHub license

What is itInstallChange LogGetting StartAPITutorialsFAQCG Site

Made by Charles Zhang • :globe_with_meridians: https://github.com/PhenomingZ

What is it

▴ Back to top

希冀平台 是一个国内最具专业深度、安全可扩展的计算机类课程一体化支撑平台,是一个定位于全面支撑计算机、人工智能和大数据专业建设的大型综合教学实验平台,而非一个只能支撑若干门课程的实验系统。

通用评测 是一个通用的自动评测框架,基于该框架可以定制开发任何自己需要的自动评测内核。

PyGrading工具包 目前该工具包包含以下功能:

  1. 支持CourseGrading平台通用评测内核快速构建;
  2. 支持适用于通用评测题和虚拟桌面环境的评测结果JSON串的快速生成;
  3. 支持HTML标签文本内容的快速生成,绝对好用的HTML生成工具;

希望使用本工具能够提高大家的工作效率,祝各位开发顺利!

Install

▴ Back to top

使用pip可以轻松安装PyGrading:

pip install pygrading

也可以下载项目文件后,切换到setup.py所在的目录,执行以下命令来安装:

python setup.py install

也可使用下面的Dockerfile来构建一个装有PyGrading的通用评测环境:

FROM jupyter/base-notebook

LABEL maintainer="Charles Zhang <694556046@qq.com>"

RUN pip install pygrading

切换至Dockerfile所在的目录,构建镜像命令为:

docker build -t cg/pygrading_env .

该镜像包含基本的Jupyter环境,可直接用于创建Jupyter调试环境。

启动镜像命令为:

docker run -it --name pygrading -p 8888:8888 cg/pygrading_env

8888端口为Jupyter服务所需,如果被占用可以映射为其他端口

PyGrading的运行环境要求 Python >= 3.7,不支持Python2。

Change Log

▴ Back to top

v0.2.3 Change Log (2020.03.03)

  1. 修复了pygrading.general_test.compiler模块中c/c++编译功能的问题,将编译选项option移动至生成编译命令的最后,添加了flag字段位于源文件字段之前,用与设定编译版本标志如-std=c++11
以往版本更新日志(点击以展开...)

v0.2.2 Change Log (2020.03.03)

  1. 发现Python3.6以下版本可能会出现包导入错误,暂时仅支持Python3.7以上版本。

v0.2.1 Change Log (2020.02.09)

  1. 添加了构建通用评测环境的Dockerfile
  2. 增加了__version__变量,方便查看程序包版本:
import pygrading.general_test as gg

# 获取版本信息字符串
print(gg.__version__)

# 直接打印版本信息
gg.version()

v0.2.0 Change Log (2020.02.04)

  1. 使用文档施工完成;
  2. 修复了postwork函数为None时prework函数不工作的问题;
  3. 读写文件功能增加了读写选项;
  4. HTML模块增加了<br>标签的支持。

v0.1.2 Change Log (2020.02.01)
pygrading.heml模块中添加了自定义标签方法custom()并支持形如<input>标签的不成组标签。

v0.1.0 Change Log (2020.01.29)
通用评测内核功能完成,HTML构建功能初步搭建完成。

Getting Start

▴ Back to top

通用评测内核构建

1. 设计逻辑

PyGrading采用三段式的设计逻辑,将每一次评测任务分为三个阶段,分别完成“评测任务预处理”、“评测用例执行”、“评测结果处理”,如下图所示:

评测任务预处理 通常包括读取配置文件信息、读取评测用例信息、编译学生提交的文件等任务。

评测用例执行 PyGrading会自动迭代执行评测用例列表中的每个评测用例,而具体的评测规则可以用一个函数快速指定。

评测结果处理 通常包括评测结果汇总、生成评测报告、输出评测结果JSON串等任务。

在本文档这一部分接下来的内容中,将以一个普通编程题为例,创建一个评测内核并输出结果JSON串,样例题目如下:

题目名称:判断回文数
问题描述:判断一个整数是否是回文数。回文数是指正序(从左向右)和倒序(从右向左)读都是一样的整数。
示例 1:
  输入:121
  输出:True

示例 2:
  输入:-121
  输出:False
  说明:从左向右为“-121”,从右向左为“121-”,故不是回文数。

本例使用的学生提交代码如下:

def isPalindrome(num: int) -> bool:
    num = abs(num)
    num_str = str(num)
    return num_str == num_str[::-1]

x = eval(input())
print(isPalindrome(x))

本例使用的测试用例如下:

input output
121 True
10 False
-12 False
331133 True
-121 False

样例学生代码在执行最后一组测试用例时会输出错误答案,本示例所有代码均可在./example/GettingStart目录下找到。

2. 导入程序包

PyGrading安装完成之后,推荐在您的代码中使用如下方式导入通用评测相关模块:

import pygrading.general_test as gg

如果您需要在评测结果中显示html内容,推荐如下方式导入html相关模块:

from pygrading.html import *

3. 创建评测任务预处理函数

首先创建用于评测任务预处理的prework()函数,完成配置文件和评测用例的读取,配置文件使用JSON格式,内容如下:

{
    "testcase_num": "5",
    "testcase_dir": "./example/GettingStart/testdata",
    "submit_path": "./example/GettingStart/submit/main.py"
}

PyGrading推荐按照如下目录结构构建评测数据,学生提交的代码将会被挂载到submit目录中,测试数据的输入输出存放于testdata目录中。PyGrading提供了函数可用于直接读取以这种目录结构创建的评测用例:

.
├── config.json
├── submit
│   └── main.py
└── testdata
    ├── input
    │   ├── input1.txt
    │   ├── input2.txt
    │   ├── input3.txt
    │   ├── input4.txt
    │   └── input5.txt
    └── output
        ├── output1.txt
        ├── output2.txt
        ├── output3.txt
        ├── output4.txt
        └── output5.txt

创建prework()函数的代码如下,job为PyGrading创建的任务实例:

def prework(job):
    # 读取配置文件
    config = gg.load_config("./example/GettingStart/config.json")

    # 创建测试用例实例
    testcases = gg.create_std_testcase(config["testcase_dir"], config["testcase_num"])

    job.set_config(config)
    job.set_testcases(testcases)

4. 创建评测用例执行函数

接下来创建创建用于执行测试用例的run()函数,该函数接收单组测试用例,并返回一个可包含任意内容的字典,所有评测用例返回的内容最终会汇总到一个列表中传递给评测结果处理函数。

使用gg.utils.bash()执行Shell命令时,会返回当前命令的执行时间,可用做代码性能评判依据。

创建run()函数的代码如下,job为PyGrading创建的任务实例,testcase为字典类型,包含单个测试用例信息:

def run(job, testcase):
    # 读取当前任务的配置信息
    configuration = job.get_config()

    # 创建和执行评测用Shell命令
    cmd = ["cat", testcase.input_src, "|", "python3 " + configuration["submit_path"]]
    status, output, time = gg.utils.bash(" ".join(cmd))

    # 初始化返回的字典对象
    result = {"name": testcase.name, "time": time}

    # 读取评测用例答案
    answer = gg.utils.readfile(testcase.output_src)

    # 将执行结果写回返回对象
    result["output"] = output
    result["answer"] = answer

    # 比较评测答案和实际输出将评测结果写回返回对象
    if gg.utils.compare_str(output, answer):
        result["verdict"] = "Accept"
        result["score"] = testcase.score
    else:
        result["verdict"] = "Wrong Answer"
        result["score"] = 0

    return result

5. 创建评测结果处理函数

之后创建评测结果处理函数postwork(),并使用pygrading.html中的相关工具创建带有HTML标签的字符串。

创建postwork()函数的代码如下,job为PyGrading创建的任务实例:

def postwork(job):
    # 设定结果verdict
    job.verdict(str(font(color="green").set_text("Accept")))

    # 设定结果score
    job.score(job.get_total_score())

    # 设定结果rank
    job.rank({"rank": str(job.get_total_time())})

    # 创建HTML标签
    detail = table(
        tr(
            th(),
            th().set_text("Verdict"),
            th().set_text("Output"),
            th().set_text("Answer")
        ), border="0"
    )
    for i in job.get_summary():
        if i["verdict"] == "Runtime Error":
            ver = font(color="red").set_text("Runtime Error")
            job.verdict(str(ver))
        elif i["verdict"] == "Wrong Answer":
            ver = font(color="red").set_text("Wrong Answer")
            job.verdict(str(ver))

        row = tr(
            td(align="center").set_text(str2html(i["name"])),
            td(align="center").set_text(str2html(i["verdict"])),
            td(align="center").set_text(str2html(i["output"])),
            td(align="center").set_text(str2html(i["answer"]))
        )
        detail << row

    # 将HTML标签转换为字符串,设定为结果detail
    job.detail(str(detail))

6. 启动任务

完成三个阶段的函数编写后,将三个函数作为参数传入gg.job()函数,生成一个任务实例:

# 创建任务实例
new_job = gg.job(prework=prework, run=run, postwork=postwork)

# 任务启动
new_job.start()

# 打印结果
new_job.print()

程序执行后输出结果如下:

{"verdict": "<font color='red'>Wrong Answer</font>", "score": "80", "rank": {"rank": "238"}, "HTML": "enable", "detail": "<table border='1'><tr><th></th><th>Verdict</th><th>Output</th><th>Answer</th></tr><tr><td align='center'>TestCase1<br></td><td align='center'>Accept<br></td><td align='center'>True<br></td><td align='center'>True<br></td></tr><tr><td align='center'>TestCase2<br></td><td align='center'>Accept<br></td><td align='center'>False<br></td><td align='center'>False<br></td></tr><tr><td align='center'>TestCase3<br></td><td align='center'>Accept<br></td><td align='center'>False<br></td><td align='center'>False<br></td></tr><tr><td align='center'>TestCase4<br></td><td align='center'>Accept<br></td><td align='center'>True<br></td><td align='center'>True<br></td></tr><tr><td align='center'>TestCase5<br></td><td align='center'>Wrong Answer<br></td><td align='center'>True<br></td><td align='center'>False<br></td></tr></table>"}

在CG平台中显示效果如下:

Wrong Answer

Verdict Output Answer
TestCase1
Accept
True
True
TestCase2
Accept
False
False
TestCase3
Accept
False
False
TestCase4
Accept
True
True
TestCase5
Wrong Answer
True
False

一个简单的通用评测内核开发完成!

PyGrading API

▴ Back to top

在本节中,将会列出当前版本(v0.2.1)全部的接口与方法,详细使用方法请参考Tutorials部分。

pygrading.general_test

该包推荐导入方式:

import pygrading.general_test as gg

包含有以下方法和类:

1. gg.load_config(source: str)

读取含有配置信息的JSON文件,返回字典类型。

详细信息(点击以展开...)

Arguments:

Arguments Type Default Description
source String Required 配置文件的文件路径

Returns:

Type Description Example
Dict 以字典形式返回配置信息 {'testcase_num': '3','testcase_dir': 'example/testdata','submit_path': 'example/submit/*'}

2. gg.create_std_testcase(testcase_dir: str, testcase_num: int)

以推荐的方式创建TestCases对象实例。

详细信息(点击以展开...)

以推荐的方式构建评测用例目录,即可使用本方法直接创建一个TestCases对象实例。

推荐的目录构建方式如下:

testdata
├── input
│   ├── input1.txt
│   ├── input2.txt
│   └── input3.txt
└── output
    ├── output1.txt
    ├── output2.txt
    └── output3.txt

testdata目录为评测用例所在的根目录,评测用例的输入和输出分别放在input和output两个子目录中。

所有的评测用例输入按照input1.txt、input2.txt、input3.txt依次命名,所有的评测用例输出按照output1.txt、output2.txt、output3.txt依次命名。

Arguments:

Arguments Type Default Description
testcase_dir String Required 评测用例目录路径
testcase_num Integer Required 评测用例的数目

Returns:

Type Description Example
TestCases PyGrading创建的评测用例实例类型 -

3. gg.create_testcase(total_score: int = 100)

创建一个空的TestCases实例。

详细信息(点击以展开...)

在无法使用推荐的方式构建评测用例的情况下,可以创建一个空的TestCases实例并手动添加评测用例。添加方法请参考gg.TestCases类的介绍。

Arguments:

Arguments Type Default Description
total_score Integer 100 评测用例总分

Returns:

Type Description Example
TestCases PyGrading创建的评测用例实例类型

4. gg.TestCases 类

该类用于存储和传递关于评测用例的全部信息,通过迭代的方式将每个测试用例传入到评测用例执行函数中。
请使用gg.create_std_testcase()gg.create_testcase()获取TestCases实例。

详细信息(点击以展开...)

gg.TestCases类包含有1个子类__SingleTestCase,该子类的实例用于存储单个评测用例的信息,且会作为一个必要参数传入评测用例执行函数中。

子类__SingleTestCase包含的属性如下:

Attributes Type Default Description
name String Required 评测用例的名称
score Integer Required 评测用例的满分分值
input_src Object Required 评测用例的输入,可以为任何类型
output_src Object Required 评测用例的输出,可以为任何类型
extension Object None 评测用例的额外信息,可以为任何类型

gg.TestCases类包含有如下属性:

Attributes Type Default Description
__count Integer 0 保存该实例中评测用例的数量
__cases List [ ] 以列表的形式保存实例中所有的评测用例
__total_score Integer 100 保存评测用例总分,默认为百分制

gg.TestCases类包含有如下方法:

Function Return Description
TestCases.append(self, name: str, score: float, input_src: object, output_src: object, extension: object = None) None 向一个TestCases实例添加评测用例,传入参数的属性和__SingleTestCase中对应
TestCases.get_count(self) Integer 获取评测用例数目值
TestCases.get_total_score(self) Integer 获取评测用例总分
TestCases.get_testcases(self) List[__SingleTestCase] 获取评测用例列表
TestCases.set_total_score(self, total_score: int) None 设定评测用例总分
TestCases.isempty(self) Boolean 判断评测用例是否为空

5. gg.job(prework=None, run=None, postwork=None)

该方法用于创建评测任务实例,传入评测任务预处理、评测用例执行、评测结果处理的相关函数,返回一个Job实例。

详细信息(点击以展开...)

Arguments:

Arguments Type Default Description
prework Function None 评测任务预处理函数
run Function None 评测用例执行函数
postwork Function None 评测结果处理函数

Returns:

Type Description Example
Job PyGrading创建的评测任务实例类型

6. gg.Job 类

该类用于创建评测任务实例,提供任务初始化、任务执行、输出结果的功能。

详细信息(点击以展开...)

gg.Job类包含有如下属性:

Attributes Type Default Description
__prework Function None 评测任务预处理函数
__run Function None 评测用例执行函数
__postwork Function None 评测结果处理函数
__testcases TestCases TestCases() 评测用例
__config Dict None 配置信息
__terminate Boolean False 程序结束标记
__result Dict {
"verdict": "Unknown Error",
"score": "0",
"rank": {"rank": "-1"},
"HTML": "enable"
}
评测任务执行结果的内部存储字典
__summary List [ ] 每个测试用例的执行结果汇总列表

gg.Job类包含有如下方法:

Function Return Description
Job.verdict(self, src: str) None 修改返回结果中的verdict字段
Job.score(self, src: int) None 修改返回结果中的score字段
Job.rank(self, src: Dict) None 修改返回结果中的rank字段
Job.images(self, src: str) None 修改返回结果中的images字段
Job.comment(self, src: str) None 修改返回结果中的comment字段
Job.detail(self, src: str) None 修改返回结果中的detail字段
Job.detail(self, src: str) None 修改返回结果中的detail字段
Job.HTML(self, src: str) None 修改返回结果中的HTML字段
Job.custom(self, key: str, value: str) None 在返回结果中创建自定义字段并赋值
Job.get_summary(self) List 获取评测用例汇总结果列表
Job.get_config(self) Dict 获取评测任务配置信息
Job.get_total_score(self) Integer 获取评测任务总分
Job.get_total_time(self) Integer 获取评测任务执行总时间
Job.set_testcases(self, testcases: TestCases) None 设定评测任务的评测用例
Job.set_config(self, config: Dict) None 设定评测用例的配置信息
Job.terminate(self) None 将终止标记置于True,执行完当前函数后评测终止
Job.start(self) List 开始任务,返回评测用例汇总结果列表
Job.print(self) None 将评测结果转化为JSON格式并打印到标准输出

7. gg.version 和 gg.version()

用于查看PyGrading的版本信息

详细信息(点击以展开...)
import pygrading.general_test as gg

# 获取版本信息字符串
print(gg.__version__)

# 直接打印版本信息
gg.version()

pygrading.general_test.utils

该包封装了在编写评测内核的过程中可能会使用到的基本操作,可分为如下几类:

1. 执行操作

执行Shell命令的方法,返回值包括执行状态、执行后输出的内容、执行时间。

详细信息(点击以展开...)
Function Return Description
gg.utils.bash(cmd: str) Tuple 执行Shell命令,返回值包括执行状态(status)、执行后输出的内容(output)、执行时间(time)

2. 文件操作

有关文件读写增删的相关操作。

详细信息(点击以展开...)
Function Return Description
gg.utils.copyfile(src: str, dst: str) None 将src路径所指示的文件复制到dst所指示的位置
gg.utils.copytree(src: str, dst: str) None 将src路径所指示的目录递归地复制到dst所指示的位置
gg.utils.move(src: str, dst: str) None 将src路径所指示的文件移动到dst所指示的位置
gg.utils.rmtree(src: str) None 递归地删除src路径所指示的目录
gg.utils.rmfile(src: str) None 删除src路径所指示的文件
gg.utils.rename(old: str, new: str) None 将old路径所指示的文件重命名为new
gg.utils.readfile(src: str) String 读取路径为src的文件,并以字符串的形式返回文件内容,文件中的换行以'\n'表示
gg.utils.writefile(src: str, lines: str, option: str = "w") None 将lines中的内容写入src,通过option选项选择写入模式,默认为“w”
gg.utils.readfile_list(src: str) List[str] 读取路径为src的文件,并以列表的形式返回文件内容,文件中每行为列表中的一个元素
gg.utils.writefile_list(src: str, lines: List, option: str = "w") None 将lines中的内容写入src,通过option选项选择写入模式,默认为"w"
gg.utils.str2list(src: str) List[str] 将普通字符串转化为列表,根据"\n"进行分隔,并会自动去掉字符串末尾的空行

3. 比较操作

关于评测用例执行结果比较打分的相关操作。

详细信息(点击以展开...)
compiler
Function Return Description
gg.utils.compare_str(str1: str, str2: str) Boolean 返回输入的两个字符串是否相同,并自动忽略字符串尾的换行符
gg.utils.compare_list(list1: List, list2: List) Boolean 返回输入的两个列表是否相同,并自动忽略列表最后的换行符和空白字符串
gg.utils.edit_distance(obj1, obj2) Integer 返回两个可迭代类型的参数是否相同,在比较字符串和列表时不会自动处理空行,建议在进行字符串比较时,使用str2list()函数预处理传入的数据

pygrading.general_test.compiler

该包封装了部分编程语言的编译方法,目前支持如下编程语言:

c, cpp

包含的方法如下:

详细信息(点击以展开...)
Function Return Description
gg.compiler.compile_c(source: str, target: str, compiler_type: str = "gcc", flag: str = "-O2 -Wall -std=c99", option: str = "") Tuple 针对c语言编译的方法,通过source传入源文件路径,target指定编译后文件路径,compiler_type选择编译器类型,通过flag设定版本标签和一些前置选项,通过option添加编译选项。返回执行状态和执行过程中的输出
gg.compiler.compile_cpp(source: str, target: str, compiler_type: str = "g++", flag: str = "-O2 -Wall -std=c++11", option: str = "") Tuple 针对c++语言编译的方法,通过source传入源文件路径,target指定编译后文件路径,compiler_type选择编译器类型,通过flag设定版本标签和一些前置选项,通过option添加编译选项。返回执行状态和执行过程中的输出

pygrading.html

该包提供了创建HTML标签文本的相关功能,支持成对标签和不成对标签的创建,支持标签之间的嵌套创建,推荐导入方式如下:

from pygrading.html import *

导入包之后目前可以通过如下方式创建并打印HTML标签实例,详细的使用方法请参考评测结果的展示优化

from pygrading.html import *

a = table(
    tr(
        td(font(color="red").set_text("Hello World"))
    )
)

print(a)

生成的HTML文本如下:

<table><tr><td><font color='red'>Hello World</font></td></tr></table>
详细信息(点击以展开...)

目前已经支持的内置标签有如下几种:

  1. 成组标签:
<a> <body> <div> <font> <form> <h1> <h2> <h3> <h4> <h5> <h6>
<head> <html> <p> <table> <th> <title> <tr> <td>
  1. 不成组标签
<br> <img> <input> 

由于input()为Python内置方法,故创建<input>标签的方法为input_tag()

这些标签均可通过tag()的方式创建,并可以通过print(tag())的方式打印,或通过str(tag())的方式转化为字符串。

除了内置标签外,PyGrading还支持创建自定义标签:

custom_single_tag(tag_name) 传入自定义的标签名称,创建一个不成组的标签
custom(tag_name) 传入自定义的标签名称,创建一个成组的标签

成组标签继承于pygrading.html.Tag类,包含有如下方法:

Function Return Description
Tag.append(self, obj) None 向当前标签实例中添加一个子标签
Tag.pop(self, index=-1) None 从当前标签实例中移除一个子标签,默认移除最后的一个
Tag.insert(self, index, obj) None 向当前标签实例中插入一个子标签
Tag.extend(self, seq) None 向当前标签实例中追加一个子标签列表
Tag.set_text(self, src: str) None 设定当前标签中的文本
Tag.print(self) None 将标签实例转化为HTML文本并打印到标准输出
Tag.__str__(self) String 重载方法,将标签实例转化为HTML文本字符串
Tag.__lshift__(self, other) Tag 重载操作符<<,用法同append,向当前标签实例中添加一个子标签

不成组标签继承于pygrading.html.SingleTag类,包含有如下方法:

Function Return Description
SingleTag.print(self) None 将标签实例转化为HTML文本并打印到标准输出
SingleTag.__str__(self) String 重载方法,将标签实例转化为HTML文本字符串

此外,该包还提供了一个方法用于将普通字符串中的换行符转换为HTML的换行符:

Function Return Description
str2html(src: str) String 将src字符串中的"\n"替换为"
"并返回新的字符串

Tutorials

▴ Back to top

在本章中,将会通过例子,详细解析不同模块的用法。本章的所有例子均可在example目录下找到。

目录 (点击以展开...)

构建并读取配置文件

在第一个例子中,我们将要演示如何构建并读取你的配置文件。由于在通用评测内核的实际使用中,教师账号没有权限操作实验环境的管理,因此对于不同的题目需要使用测试数据编辑器功能进行配置,配置文件就成为了用一个评测内核支持多个题目的关键。

配置文件是一个JSON格式的文本文件,用于向评测内核传递当前题目所需的配置项。一般来说,配置文件中的配置项至少应当包含:评测用例的数目(testcase_num)、评测用例的挂载路径(testcase_dir)和学生提交文件的挂载路径(submit_path)。如下面例子所示:

{
    "testcase_num": "5",
    "testcase_dir": "./example/GettingStart/testdata",
    "submit_path": "./example/GettingStart/submit/main.py"
}

当然,配置文件中的配置项可以根据实际情况自行添加,因为如何处理这些配置项也是由开发者决定的。

PyGrading提供了load_config(source)方法来读取配置文件,该方法传入配置文件的路径,返回一个由配置文件中的JSON串转换成的字典。我们推荐在评测任务预处理函数(prework)中完成配置文件的读取,并将配置信息传递给当前任务(job)的config属性,以便在其他函数中可以使用这些配置信息。

下面一段代码展示了如何读取并使用我们配置文件,由于还没配置评测用例,所以我们创建一个只包含prework和postwork的评测任务:

import pygrading.general_test as gg

def prework(job):
    config = gg.load_config("./example/构建并读取配置文件/config.json")
    job.set_config(config)

def postwork(job):
    config = job.get_config()
    testcase_num = config["testcase_num"]
    testcase_dir = config["testcase_dir"]
    submit_path = config["submit_path"]

    print("testcase_num:", testcase_num)
    print("testcase_dir:", testcase_dir)
    print("submit_path:", submit_path)

myjob = gg.job(prework=prework, run=None, postwork=postwork)

myjob.start()

执行结果如下:

testcase_num: 5
testcase_dir: ./example/GettingStart/testdata
submit_path: ./example/GettingStart/submit/main.py

以上就是构建并读取配置文件的方法。

构建并读取评测用例

接下来,我们尝试构建几组评测用例。评测用例一般来说是一组包含输入和输出的文件,每一组评测用例都会作为参数传递给评测用例执行函数(run())迭代执行并返回结果。

PyGrading推荐使用gg.create_std_testcase()方法进行评测用例实例的创建,该方法要求遵照标准形式配置评测用例目录,具体要求如下:

testdata
├── input
│   ├── input1.txt
│   ├── input2.txt
│   ├── input3.txt
│   ├── input4.txt
│   └── input5.txt
└── output
    ├── output1.txt
    ├── output2.txt
    ├── output3.txt
    ├── output4.txt
    └── output5.txt

testdata为评测用例目录的根目录,建议将这个路径写入配置文件。根目录下分别设有inputoutput目录,分别用于存放输入和输出文件。

输入和输出文件的命名从input1.txtoutput1.txt开始并一一对应。

在以此方式创建的评测用例实例中,总分默认为100分,传递给评测用例执行函数(run())的输入输出参数为每组inputoutput文件的路径,下面通过一段代码展示具体的使用方法:

import pygrading.general_test as gg


def prework(job):
    config = gg.load_config("./example/构建并读取评测用例/config.json")
    testcases = gg.create_std_testcase(config["testcase_dir"], config["testcase_num"])

    job.set_testcases(testcases)


def run(job, testcase):
    print("######################")
    print("Name:", testcase.name)
    print("score:", testcase.score)
    print("input_src:", testcase.input_src)
    print("output_src:", testcase.output_src)
    print("extension:", testcase.extension)


myjob = gg.job(prework=prework, run=run, postwork=None)

myjob.start()

执行结果如下:

######################
Name: TestCase1
score: 50.0
input_src: ./example/构建并读取评测用例/testdata_std/input/input1.txt
output_src: ./example/构建并读取评测用例/testdata_std/output/output1.txt
extension: None
######################
Name: TestCase2
score: 50.0
input_src: ./example/构建并读取评测用例/testdata_std/input/input2.txt
output_src: ./example/构建并读取评测用例/testdata_std/output/output2.txt
extension: None

上面的例子展示了如何使用推荐的方式构建、读取并使用评测用例,接下来我们将展示如何使用gg.testcase()方法来自定义评测用例。

假设根据要求,不必从文件中读取评测用例,而是直接由程序创建。参见下面代码实例:

import pygrading.general_test as gg


def prework(job):
    # 自定义评测用例总分
    testcases = gg.create_testcase(100)

    for i in range(1, 5):
        input_src = i
        output_src = pow(2, i)

        # 使用append()方法向testcases追加评测用例
        testcases.append("TestCase{}".format(i), 25, input_src, output_src)

    job.set_testcases(testcases)


def run(job, testcase):
    print("######################")
    print("Name:", testcase.name)
    print("score:", testcase.score)
    print("input_src:", testcase.input_src)
    print("output_src:", testcase.output_src)
    print("extension:", testcase.extension)


myjob = gg.job(prework=prework, run=run, postwork=None)

myjob.start()

执行结果如下:

######################
Name: TestCase1
score: 25.0
input_src: 1
output_src: 2
extension: None
######################
Name: TestCase2
score: 25.0
input_src: 2
output_src: 4
extension: None
######################
Name: TestCase3
score: 25.0
input_src: 3
output_src: 8
extension: None
######################
Name: TestCase4
score: 25.0
input_src: 4
output_src: 16
extension: None

以上就是构建并读取评测用例的方法。

评测结果的收集反馈

我们在评测任务预处理阶段完成了读取配置文件和评测用例的任务,接下来评测用例将会传递给评测用例执行函数执行并收集结果。

如果在评测过程中需要执行Shell命令,可以使用gg.utils.bash()方法,该方法接收一个Shell命令字符串,返回值依次为:执行状态、执行过程输出、执行时间。

对于获取到的执行结果,我们提供了字符串比较和编辑距离比较两种方式:

  1. 字符串比较支持单个字符串和字符串列表两种形式,返回结果为两个字符串是否相同的布尔值;
  2. 编辑距离比较在接收到两个字符串列表时会返回两个列表之间的编辑距离,常用与按行比较的情况,返回结果为两个列表之间的编辑距离数值。

推荐以字典的形式收集并返回每个评测用例的执行结果,这些结果会保存在评测任务实例jobsummary变量中,可以使用job.get_summary()来获取。

下面以一个简单的例子演示如何获取并收集评测结果:

import pygrading.general_test as gg


def prework(job):
    testcases = gg.create_testcase(100)

    for i in range(1, 5):
        input_src = i
        output_src = pow(2, i)
        testcases.append("TestCase{}".format(i), 25, input_src, output_src)

    job.set_testcases(testcases)


def run(job, testcase):
    # 使用Shell命令计算2^n
    cmd = ["echo", "$((", "2", "**", str(testcase.input_src), "))"]

    # 获取执行情况
    status, output, time = gg.utils.bash(" ".join(cmd))

    # 初始化返回结果的字典
    result = {"name": testcase.name, "time": time}

    # 获取评测用例给出的答案
    answer = testcase.output_src

    result["output"] = output
    result["answer"] = answer

    # 根据字符串比较结果返回单个测试用例的评判情况
    if gg.utils.compare_str(str(output), str(answer)):
        result["verdict"] = "Accept"
        result["score"] = testcase.score
    else:
        result["verdict"] = "Wrong Answer"
        result["score"] = 0

    return result


def postwork(job):
    # 打印收集到的每个评测用例的结果
    print(job.get_summary())


myjob = gg.job(prework=prework, run=run, postwork=postwork)

myjob.start()

根据上面的例子,结合特定的评测用例情况,应当可以编写出满足需求的评测用例执行函数。

在实际应用中,应当在各个阶段的函数中对可能发生的异常进行捕获,以保证评测程序顺利执行,让学生能查看到正确的反馈信息并预防可能的作弊行为。

评测结果的打印输出

在评测任务执行完毕后,我们在评测结果处理函数中对收集到的评测结果进行最后处理并生成CG平台支持的JSON串格式,关于所支持的格式详情请查看通用评测题开发指南

在评测任务实例Job中,有如下几个内置的方法,用于设定待输出的JSON串字段:

Function Description
job.verdict() 基本判定,一般为简要的评测结果描述或者简写,例如OJ系统的AC、PE、CE等
job.rank() 选择排行榜模式时,必须有该项,浮点数(正常值 ≥0),该值决定了本次提交在排行榜上的位置,排行榜从小到大排序。如果提交的材料有误或者其它异常,将rank值置为负数,不参与排行!
job.score() 选择直接评测得分时,必须有该项,按照百分制给分,必须为大于等于0的整数,例如90
job.images() 可选,如果评测结果有图表,需要转换为base64或者SVG(启用HTML)格式
job.comment() 可选,评测结果的简要描述。
job.detail() 可选,评测结果的详细描述,可以包含协助查错的信息。布置作业的时候,可以选择是否显示这项信息。
job.secret() 可选,该信息只有教师评阅时才能看到。
job.HTML() 可选,如果置为enable,开发者可以使用HTML标签对verdict、comment、detail的输出内容进行渲染。
job.custom() 可选,自定义字段。

设定好JSON字段之后可以使用job.print()打印JSON字段到标准输出。

下面通过一个简单的示例展示如何配置评测任务的输出结果:

import pygrading.general_test as gg


def postwork(job):
    job.verdict("Accept")
    job.score(100)
    job.detail("Detail Message!")
    job.custom("custom_key", "custom_value")


myjob = gg.job(prework=None, run=None, postwork=postwork)

myjob.start()

myjob.print()

输出结果如下:

{"verdict": "Accept", "score": "100", "rank": {"rank": "-1"}, "HTML": "enable", "detail": "Detail Message!", "custom_key": "custom_value"}

实际应用中,请使用job.get_summary()获取评测结果列表,再根据评测结果决定要输出的内容。

评测结果的展示优化

为了更好地展示评测结果,CG平台支持在返回JSON结果中的verdictcommentdetailsecret字段中添加HTML标签,经过渲染后显示为最终展示结果。

PyGrading中提供了强大的HTML文档构建工具pygrading.html,详细的API请参考pygrading.html API,接下来通过几个实例讲解各种场景的使用方法。

1. 修改文本颜色

verdict字段中我们通常希望展示如AcceptWrong Answer这样的内容,为了让这些信息显示的更加直观,则需要给他们添加颜色。

下面创建一段HTML文本,对Accept显示为绿色,对Wrong Answer显示为红色:

from pygrading.html import *

# 在括号中可以输入任意组键值对,他们将作为标签的属性显示在最终的HTML文本中
accept = font(color="green").set_text("Accept")
wrong_answer = font(color="red").set_text("Wrong Answer")

accept.print()
wrong_answer.print()

输出结果如下:

<font color='green'>Accept</font>
<font color='red'>Wrong Answer</font>

显示效果如下:

html01

2. 创建表格

commentdetailsecret字段中,通常需要表格进行内容的展示,接下来通过一个实例说明如何创建表格。

假设我们需要创建一个表格来比较每个评测用例中,学生输出的内容和标准答案的区别,解决方案如下:

from pygrading.html import *

# 可以看到字符串中含有换行符,推荐使用str2html()函数进行处理,将换行符转化为<br>
outputs = ["1", "1\n2", "1\n2\n3", "1\n2\n3\n4", "5\n4\n3\n2\n1"]
answers = ["1", "1\n2", "1\n2\n3", "1\n2\n3\n4", "1\n2\n3\n4\n5"]

# 标签之间可以相互嵌套,任意数量的子标签可以作为参数传递给父标签
result = table(
    tr(
        th().set_text("Output"),
        th().set_text("Answer")
    )
)

for out, ans in zip(outputs, answers):
    tmp = tr(
        td().set_text(str2html(out)),
        td().set_text(str2html(ans)),
    )
    # 可以使用“<<”操作符将一个标签作为子标签传递给另一个标签
    result << tmp

result.print()

生成HTML文本如下:

<table><tr><th>Output</th><th>Answer</th></tr><tr><td>1<br></td><td>1<br></td></tr><tr><td>1<br>2<br></td><td>1<br>2<br></td></tr><tr><td>1<br>2<br>3<br></td><td>1<br>2<br>3<br></td></tr><tr><td>1<br>2<br>3<br>4<br></td><td>1<br>2<br>3<br>4<br></td></tr><tr><td>5<br>4<br>3<br>2<br>1<br></td><td>1<br>2<br>3<br>4<br>5<br></td></tr></table>

显示效果如下:

OutputAnswer
1
1
1
2
1
2
1
2
3
1
2
3
1
2
3
4
1
2
3
4
5
4
3
2
1
1
2
3
4
5

3. 创建表单

下面以创建一个用户名输入表单为例,展示如何创建不成对的HTML标签文本:

from pygrading.html import *

# 由于input()为Python内置方法,故创建<input>标签的方法为`input_tag()`
result = form(
    font().set_text("First name"),
    br(),
    input_tag(type="text", name="firstname"),
    br(),
    font().set_text("Last name"),
    br(),
    input_tag(type="text", name="lastname"),
    br(),
    input_tag(type="submit", value="Submit")
)

result.print()

生成HTML文本如下:

<form><font>First name</font><br><input type='text' name='firstname'><br><font>Last name</font><br><input type='text' name='lastname'><br><input type='submit' value='Submit'></form>

显示效果如下:

html02

FAQ

▴ Back to top

暂无提问

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

pygrading-0.2.3.tar.gz (46.4 kB view hashes)

Uploaded Source

Built Distribution

pygrading-0.2.3-py3-none-any.whl (27.3 kB view hashes)

Uploaded Python 3

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page