- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
假设我有一个提交文件,fileFromStudent.py,其中唯一的内容是:
print("hello world")
我想测试标准输出,看看学生是否正确写出了打印语句。根据我所读到的内容,我已经能够创建以下代码:
from io import StringIO
from unittest.mock import patch
import unittest, importlib, sys
class TestStringMethods(unittest.TestCase):
def setUp(self):
studentSubmission = 'fileFromStudent'
## Stores output from print() in fakeOutput
with patch('sys.stdout', new=StringIO()) as self.fakeOutput:
## Loads submission on first test, reloads on subsequent tests
if studentSubmission in sys.modules:
importlib.reload(sys.modules[ studentSubmission ] )
else:
importlib.import_module( studentSubmission )
## Test Cases
def test_print_passes(self):
test_case = "Checking Output Statement - Will Pass"
self.output = self.fakeOutput.getvalue().strip()
self.assertEqual(self.output, 'hello world', msg=test_case)
def test_print_fails(self):
test_case = "Checking Output Statement - Will Fail"
self.output = self.fakeOutput.getvalue().strip()
self.assertEqual(self.output, 'hell world', msg=test_case)
if __name__ == '__main__':
suite = unittest.TestLoader().loadTestsFromTestCase(TestStringMethods)
testResult = unittest.TextTestRunner(verbosity=2).run(suite)
上面的方法有效,但是我的处理方式正确吗?我添加的内容之一是调用 import.reload()
来重新加载学生的程序。这是因为在最初几周,我会让学生使用 print() 作为最终输出(直到我们进入函数)。
我知道它看起来模棱两可,或者为什么我应该打扰它,因为它有效,但是上面的代码是构建此代码的正确方法还是我完全错过了使这一切变得简单的东西?
最佳答案
我花了几个星期的时间来解决这个问题,取得了一定的成功,但也遇到了一些麻烦,还需要 Google。我没有走 Popen 路线的原因之一是我想捕获学生是否提交了立即崩溃的错误代码。不管你信不信,入门类(class)的前几周就是这样。由于我发现的所有内容都是 2011 年至 2012 年的,因此我想我应该发布此内容,以便 future 的 Google 员工可以找到它。
扩展我上面写的内容,让我们假设下一个任务是获取输入并说“嗨”
name = input("What's your name? ")
print("Hi " + name)
现在,我想自动化测试,看看是否可以输入 "Adam"
并返回 "Hi Adam"
。为此,我选择使用 StringIO 作为我的标准输入 (sys.stdin = StringIO("Adam"))。这使我能够控制文本流的来源和去向。此外,我不想看到学生可能发生的所有错误 (sys.stderr = StringIO()
)。
正如我所提到的,我选择使用 importlib
而不是 Popen
。我想确保,如果学生提交了虚假代码,而不是破坏所有内容,而是让我正在运行的任何测试失败。我尝试了 subprocess
和 py.test
,虽然它们可能更好、更干净,但我找不到任何对我来说有意义的关于如何让它移动的东西正确地。
下面是我最新版本测试的副本:
from io import StringIO
from unittest.mock import patch
import unittest, importlib, sys, os
from time import sleep
# setup the environment
backup = sys.stderr
class TestTypingExercise(unittest.TestCase):
def __init__(self, test_name, filename, inputs):
super(TestTypingExercise, self).__init__(test_name)
self.library = filename.split('.')[0]
self.inputs = inputs
def setUp(self):
sys.stdin = StringIO(self.inputs[0])
try:
## Stores output from print() in fakeOutput
with patch('sys.stdout', new=StringIO()) as self.fakeOutput:
## Loads submission on first test, reloads on subsequent tests
if self.library in sys.modules:
importlib.reload(sys.modules[ self.library ] )
else:
importlib.import_module( self.library )
except Exception as e:
self.fail("Failed to Load - {0}".format(str(e)))
## Test Cases
def test_code_runs(self):
test_case = "Checking to See if code can run"
self.assertTrue(True, msg=test_case)
def test_says_hello(self):
test_case = "Checking to See if code said 'Hi Adam'"
# Regex might be cleaner, but this typically solves most cases
self.output = self.fakeOutput.getvalue().strip().lower()
self.assertTrue('hi adam' in self.output, msg=test_case)
if __name__ == '__main__':
ignore_list = ["grader.py"]
# Run Through Each Submitted File
directory = os.listdir('.')
for filename in sorted(directory):
if (filename.split('.')[-1] != 'py') or (filename in ignore_list):
continue
#print("*"*15, filename, "*"*15)
# 'Disables' stderr, so I don't have to see all their errors
sys.stderr = StringIO() # capture output
# Run Tests Across Student's Submission
suite = unittest.TestSuite()
suite.addTest(TestTypingExercise('test_code_runs', filename, 'Adam'))
suite.addTest(TestTypingExercise('test_says_hello', filename, 'Adam'))
results = unittest.TextTestRunner().run(suite)
# Reset stderr
out = sys.stderr.getvalue() # release output
sys.stderr.close() # close the stream
sys.stderr = backup # restore original stderr
# Display Test Results
print(filename,"Test Results - ", end='')
if not results.wasSuccessful():
print("Failed (test cases that failed):")
for error in results.failures:
print('\t',error[1].split('\n')[-2])
else:
print("Pass!")
sleep(0.05)
最终结果如下:
StudentSubmission01.py Test Results - Failed (test cases that failed):
AssertionError: Failed to Load - EOL while scanning string literal (StudentSubmission01.py, line 23)
AssertionError: Failed to Load - EOL while scanning string literal (StudentSubmission01.py, line 23)
StudentSubmission02.py Test Results - Pass!
StudentSubmission03.py Test Results - Pass!
StudentSubmission04.py Test Results - Pass!
StudentSubmission05.py Test Results - Pass!
StudentSubmission06.py Test Results - Pass!
StudentSubmission07.py Test Results - Pass!
StudentSubmission08.py Test Results - Pass!
StudentSubmission09.py Test Results - Pass!
StudentSubmission10.py Test Results - Pass!
StudentSubmission11.py Test Results - Pass!
StudentSubmission12.py Test Results - Pass!
StudentSubmission13.py Test Results - Pass!
[Finished in 0.9s]
如果我想测试多个不同的输入,我可能需要移动一些东西,但目前这是可行的。
关于python - 这是使用 Python 3 unittest 测试 stdout 的正确方法吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34689852/
我有以下代码来捕获和处理运行命令输出。 如何修改它以便运行命令窗口显示输出并同时记录输出?更换 @SW_HIDE与 @SW_SHOW (或等效的)只显示一个空白的命令窗口。 类似于linux的东西te
[已编辑] 用于处理 subprocess.Popen 标准输出的 2 个选项是 stdout="a_file_name" 和 stdout=subprocess.PIPE。 stderr 可以通过
c99 的 7.19.3/7 节指出: At program start-up, three text streams are predefined and need not be opened ex
我正在运行以下 proc_open 函数。加载页面时,出现错误: Use of undefined constant STDOUT - assumed 'STDOUT'` 如何正确设置STDOUT和S
我有一个运行多个进程的开发堆栈:网络服务器、自动测试、后台编译等。所有这些都是基本的命令行命令,例如 node app.js 或 lein midje :autotest. 是否可以使用一个脚本在“后
我正在使用 SLURM 在 super 计算机上调度作业。我已设置 --output=log.out 选项,将作业标准输出中的内容放入文件 (log.out)。我发现该文件每 30-60 分钟更新一次
ansible/ansible-playbook 版本:2.1.2.0/2.2.0.0 我正在尝试使用 yum/apt 安装软件包,但由于安装软件包的存储库位于 packagecloud.io 中,有
bala@hp:~$ echo "Hello World" > stdout bala@hp:~$ cat stdout Hello World bala@hp:~$ echo "Hello Worl
在从 Fortran 编写的外部库中调用嘈杂的函数之前,我正在使用如下代码重定向标准输出: // copy standard output out = dup(STDOUT_FILENO); // c
这个问题在这里已经有了答案: How can I pipe stderr, and not stdout? (11 个答案) 关闭 6 年前。 我有一个程序,我想检查其 STDERR 输出并在其上运
我正在从 perl 运行一个 java 应用程序 这是脚本 #!/usr/bin/perl use strict; use warnings; $| = 1; my $args = join (" "
我正在尝试将 python 脚本的 STDOUT 重定向到一个文件。 如果 STDOUT 是从 sys 导入的,脚本的输出不会被重定向到一个文件: from sys import stdout std
我正在尝试使用 PHP 和 Apache(在 Docker 的前台运行)写入 stdout(或 stderr)。 这些作品: file_put_contents( "php://stderr","wo
我正在尝试重定向标准输出,以便 Windows 应用程序中的 printf 将转到我选择的文件。 我这样做: outFile = fopen("log.txt", "w"); *stdout = *o
在 Ruby 中,$stdout(前面有一个美元符号)和 STDOUT(全部大写)有什么区别?在进行输出重定向时,应该使用哪个,为什么? $stderr 和 STDERR 也是如此。 编辑: 刚找到一
让我们看看这个Hello World程序 #include int main(int argc, char ** argv) { printf("Hello, World!"); c
我在 64 位机器上运行 Ubuntu 20.04。 我想将 stdout 重定向到从 memfd_create 获得的描述符。似乎使用 memfd_create 创建的匿名文件只有在 stdout
我想在 Python 脚本末尾捕获控制台输出。也就是说,我既想正常打印到控制台,又想在执行结束时将控制台输出保存到文件中。 我看过各种相关的 SO 问题 1 , 2 , 3尽管他们要么简单地重定向输出
我知道可以将两者都重定向到特定文件: ./command 1> out.log 2> err.log 或 ./command 1>test.log 2>&1 将两者写入文件。但是我不知道在只打印其中一
我知道可以将两者都重定向到特定文件: ./command 1> out.log 2> err.log 或 ./command 1>test.log 2>&1 将两者写入文件。但是我不知道在只打印其中一
我是一名优秀的程序员,十分优秀!