Mercurial > ~astiob > upreckon > hgweb
annotate problem.py @ 134:e84f33a60a5c
Moved process killing logic into platform-specific modules
| author | Oleg Oshmyan <chortos@inbox.lv> | 
|---|---|
| date | Fri, 20 May 2011 14:47:42 +0100 | 
| parents | b7fb64ce03d9 | 
| children | 523ba6907f3a | 
| rev | line source | 
|---|---|
| 77 
69eadc60f4e2
Memory limit is now applied to the RSS when os.wait4 is available
 Oleg Oshmyan <chortos@inbox.lv> parents: 
76diff
changeset | 1 # Copyright (c) 2010-2011 Chortos-2 <chortos@inbox.lv> | 
| 16 | 2 | 
| 21 | 3 from __future__ import division, with_statement | 
| 4 | |
| 91 | 5 from compat import * | 
| 6 import config, testcases | |
| 7 from __main__ import options | |
| 21 | 8 | 
| 22 | 9 import os, re, sys | 
| 21 | 10 | 
| 16 | 11 try: | 
| 75 
007f7eb6fb2b
The test context stack is now a deque.
 Oleg Oshmyan <chortos@inbox.lv> parents: 
74diff
changeset | 12 from collections import deque | 
| 
007f7eb6fb2b
The test context stack is now a deque.
 Oleg Oshmyan <chortos@inbox.lv> parents: 
74diff
changeset | 13 except ImportError: | 
| 
007f7eb6fb2b
The test context stack is now a deque.
 Oleg Oshmyan <chortos@inbox.lv> parents: 
74diff
changeset | 14 deque = list | 
| 
007f7eb6fb2b
The test context stack is now a deque.
 Oleg Oshmyan <chortos@inbox.lv> parents: 
74diff
changeset | 15 | 
| 
007f7eb6fb2b
The test context stack is now a deque.
 Oleg Oshmyan <chortos@inbox.lv> parents: 
74diff
changeset | 16 try: | 
| 21 | 17 import signal | 
| 18 except ImportError: | |
| 19 signalnames = () | |
| 20 else: | |
| 21 # Construct a cache of all signal names available on the current | |
| 22 # platform. Prefer names from the UNIX standards over other versions. | |
| 23 unixnames = frozenset(('HUP', 'INT', 'QUIT', 'ILL', 'ABRT', 'FPE', 'KILL', 'SEGV', 'PIPE', 'ALRM', 'TERM', 'USR1', 'USR2', 'CHLD', 'CONT', 'STOP', 'TSTP', 'TTIN', 'TTOU', 'BUS', 'POLL', 'PROF', 'SYS', 'TRAP', 'URG', 'VTALRM', 'XCPU', 'XFSZ')) | |
| 24 signalnames = {} | |
| 25 for name in dir(signal): | |
| 26 if re.match('SIG[A-Z]+$', name): | |
| 27 value = signal.__dict__[name] | |
| 22 | 28 if isinstance(value, int) and (value not in signalnames or name[3:] in unixnames): | 
| 21 | 29 signalnames[value] = name | 
| 30 del unixnames | |
| 16 | 31 | 
| 39 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 32 __all__ = 'Problem', 'TestContext', 'test_context_end', 'TestGroup' | 
| 21 | 33 | 
| 26 | 34 def strerror(e): | 
| 94 | 35 s = getattr(e, 'strerror', e) | 
| 26 | 36 if not s: s = str(e) | 
| 37 return ' (%s%s)' % (s[0].lower(), s[1:]) if s else '' | |
| 21 | 38 | 
| 39 class Cache(object): | |
| 40 def __init__(self, mydict): | |
| 41 self.__dict__ = mydict | |
| 16 | 42 | 
| 39 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 43 class TestContext(object): | 
| 90 
1fb319ec33af
Skimming mode added (-k/--skim option)
 Oleg Oshmyan <chortos@inbox.lv> parents: 
88diff
changeset | 44 __slots__ = () | 
| 39 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 45 | 
| 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 46 test_context_end = object() | 
| 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 47 | 
| 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 48 class TestGroup(TestContext): | 
| 76 
0e5ae28e0b2b
Points are now weighted on a test context basis
 Oleg Oshmyan <chortos@inbox.lv> parents: 
75diff
changeset | 49 __slots__ = 'points', 'case', 'log', 'correct', 'allcorrect', 'real', 'max', 'ntotal', 'nvalued', 'ncorrect', 'ncorrectvalued' | 
| 39 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 50 | 
| 76 
0e5ae28e0b2b
Points are now weighted on a test context basis
 Oleg Oshmyan <chortos@inbox.lv> parents: 
75diff
changeset | 51 def __init__(self, points=None): | 
| 
0e5ae28e0b2b
Points are now weighted on a test context basis
 Oleg Oshmyan <chortos@inbox.lv> parents: 
75diff
changeset | 52 self.points = points | 
| 39 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 53 self.real = self.max = self.ntotal = self.nvalued = self.ncorrect = self.ncorrectvalued = 0 | 
| 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 54 self.allcorrect = True | 
| 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 55 self.log = [] | 
| 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 56 | 
| 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 57 def case_start(self, case): | 
| 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 58 self.case = case | 
| 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 59 self.correct = False | 
| 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 60 self.ntotal += 1 | 
| 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 61 if case.points: | 
| 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 62 self.nvalued += 1 | 
| 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 63 | 
| 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 64 def case_correct(self): | 
| 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 65 self.correct = True | 
| 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 66 self.ncorrect += 1 | 
| 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 67 if self.case.points: | 
| 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 68 self.ncorrectvalued += 1 | 
| 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 69 | 
| 76 
0e5ae28e0b2b
Points are now weighted on a test context basis
 Oleg Oshmyan <chortos@inbox.lv> parents: 
75diff
changeset | 70 def case_end(self): | 
| 
0e5ae28e0b2b
Points are now weighted on a test context basis
 Oleg Oshmyan <chortos@inbox.lv> parents: 
75diff
changeset | 71 self.log.append((self.case, self.correct)) | 
| 39 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 72 del self.case | 
| 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 73 if not self.correct: | 
| 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 74 self.allcorrect = False | 
| 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 75 | 
| 76 
0e5ae28e0b2b
Points are now weighted on a test context basis
 Oleg Oshmyan <chortos@inbox.lv> parents: 
75diff
changeset | 76 def score(self, real, max): | 
| 
0e5ae28e0b2b
Points are now weighted on a test context basis
 Oleg Oshmyan <chortos@inbox.lv> parents: 
75diff
changeset | 77 self.real += real | 
| 
0e5ae28e0b2b
Points are now weighted on a test context basis
 Oleg Oshmyan <chortos@inbox.lv> parents: 
75diff
changeset | 78 self.max += max | 
| 
0e5ae28e0b2b
Points are now weighted on a test context basis
 Oleg Oshmyan <chortos@inbox.lv> parents: 
75diff
changeset | 79 | 
| 39 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 80 def end(self): | 
| 76 
0e5ae28e0b2b
Points are now weighted on a test context basis
 Oleg Oshmyan <chortos@inbox.lv> parents: 
75diff
changeset | 81 if not self.allcorrect: | 
| 
0e5ae28e0b2b
Points are now weighted on a test context basis
 Oleg Oshmyan <chortos@inbox.lv> parents: 
75diff
changeset | 82 self.real = 0 | 
| 
0e5ae28e0b2b
Points are now weighted on a test context basis
 Oleg Oshmyan <chortos@inbox.lv> parents: 
75diff
changeset | 83 if self.points is not None and self.points != self.max: | 
| 
0e5ae28e0b2b
Points are now weighted on a test context basis
 Oleg Oshmyan <chortos@inbox.lv> parents: 
75diff
changeset | 84 max, weighted = self.points, self.real * self.points / self.max if self.max else 0 | 
| 
0e5ae28e0b2b
Points are now weighted on a test context basis
 Oleg Oshmyan <chortos@inbox.lv> parents: 
75diff
changeset | 85 before_weighting = ' (%g/%g before weighting)' % (self.real, self.max) | 
| 
0e5ae28e0b2b
Points are now weighted on a test context basis
 Oleg Oshmyan <chortos@inbox.lv> parents: 
75diff
changeset | 86 else: | 
| 
0e5ae28e0b2b
Points are now weighted on a test context basis
 Oleg Oshmyan <chortos@inbox.lv> parents: 
75diff
changeset | 87 max, weighted = self.max, self.real | 
| 
0e5ae28e0b2b
Points are now weighted on a test context basis
 Oleg Oshmyan <chortos@inbox.lv> parents: 
75diff
changeset | 88 before_weighting = '' | 
| 
0e5ae28e0b2b
Points are now weighted on a test context basis
 Oleg Oshmyan <chortos@inbox.lv> parents: 
75diff
changeset | 89 say('Group total: %d/%d tests, %g/%g points%s' % (self.ncorrect, self.ntotal, weighted, max, before_weighting)) | 
| 39 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 90 # No real need to flush stdout, as it will anyway be flushed in a moment, | 
| 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 91 # when either the problem total or the next test case's ID is printed | 
| 76 
0e5ae28e0b2b
Points are now weighted on a test context basis
 Oleg Oshmyan <chortos@inbox.lv> parents: 
75diff
changeset | 92 return weighted, max, self.log | 
| 39 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 93 | 
| 16 | 94 class Problem(object): | 
| 95 __slots__ = 'name', 'config', 'cache', 'testcases' | |
| 96 | |
| 97 def __init__(prob, name): | |
| 98 if not isinstance(name, basestring): | |
| 99 # This shouldn't happen, of course | |
| 21 | 100 raise TypeError('Problem() argument 1 must be string, not ' + type(name).__name__) | 
| 16 | 101 prob.name = name | 
| 21 | 102 prob.config = config.load_problem(name) | 
| 39 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 103 prob.cache = Cache({'padoutput': 0}) | 
| 21 | 104 prob.testcases = testcases.load_problem(prob) | 
| 105 | |
| 106 # TODO | |
| 107 def build(prob): | |
| 108 raise NotImplementedError | |
| 16 | 109 | 
| 110 def test(prob): | |
| 23 | 111 case = None | 
| 22 | 112 try: | 
| 75 
007f7eb6fb2b
The test context stack is now a deque.
 Oleg Oshmyan <chortos@inbox.lv> parents: 
74diff
changeset | 113 contexts = deque((TestGroup(),)) | 
| 22 | 114 for case in prob.testcases: | 
| 39 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 115 if case is test_context_end: | 
| 76 
0e5ae28e0b2b
Points are now weighted on a test context basis
 Oleg Oshmyan <chortos@inbox.lv> parents: 
75diff
changeset | 116 real, max, log = contexts.pop().end() | 
| 
0e5ae28e0b2b
Points are now weighted on a test context basis
 Oleg Oshmyan <chortos@inbox.lv> parents: 
75diff
changeset | 117 for case, correct in log: | 
| 39 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 118 contexts[-1].case_start(case) | 
| 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 119 if correct: | 
| 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 120 contexts[-1].case_correct() | 
| 76 
0e5ae28e0b2b
Points are now weighted on a test context basis
 Oleg Oshmyan <chortos@inbox.lv> parents: 
75diff
changeset | 121 contexts[-1].case_end() | 
| 
0e5ae28e0b2b
Points are now weighted on a test context basis
 Oleg Oshmyan <chortos@inbox.lv> parents: 
75diff
changeset | 122 contexts[-1].score(real, max) | 
| 39 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 123 continue | 
| 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 124 elif isinstance(case, TestContext): | 
| 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 125 contexts.append(case) | 
| 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 126 continue | 
| 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 127 contexts[-1].case_start(case) | 
| 22 | 128 granted = 0 | 
| 129 id = str(case.id) | |
| 130 if case.isdummy: | |
| 131 id = 'sample ' + id | |
| 132 say('%*s: ' % (prob.cache.padoutput, id), end='') | |
| 133 sys.stdout.flush() | |
| 134 try: | |
| 135 granted = case(lambda: (say('%7.3f%s s, ' % (case.time_stopped - case.time_started, case.time_limit_string), end=''), sys.stdout.flush())) | |
| 90 
1fb319ec33af
Skimming mode added (-k/--skim option)
 Oleg Oshmyan <chortos@inbox.lv> parents: 
88diff
changeset | 136 except testcases.TestCaseSkipped: | 
| 
1fb319ec33af
Skimming mode added (-k/--skim option)
 Oleg Oshmyan <chortos@inbox.lv> parents: 
88diff
changeset | 137 verdict = 'skipped due to skimming mode' | 
| 22 | 138 except testcases.CanceledByUser: | 
| 139 verdict = 'canceled by the user' | |
| 82 
06356af50bf9
Finished testcases reorganization and CPU time limit implementation
 Oleg Oshmyan <chortos@inbox.lv> parents: 
77diff
changeset | 140 except testcases.WallTimeLimitExceeded: | 
| 
06356af50bf9
Finished testcases reorganization and CPU time limit implementation
 Oleg Oshmyan <chortos@inbox.lv> parents: 
77diff
changeset | 141 verdict = 'wall-clock time limit exceeded' | 
| 
06356af50bf9
Finished testcases reorganization and CPU time limit implementation
 Oleg Oshmyan <chortos@inbox.lv> parents: 
77diff
changeset | 142 except testcases.CPUTimeLimitExceeded: | 
| 
06356af50bf9
Finished testcases reorganization and CPU time limit implementation
 Oleg Oshmyan <chortos@inbox.lv> parents: 
77diff
changeset | 143 verdict = 'CPU time limit exceeded' | 
| 77 
69eadc60f4e2
Memory limit is now applied to the RSS when os.wait4 is available
 Oleg Oshmyan <chortos@inbox.lv> parents: 
76diff
changeset | 144 except testcases.MemoryLimitExceeded: | 
| 
69eadc60f4e2
Memory limit is now applied to the RSS when os.wait4 is available
 Oleg Oshmyan <chortos@inbox.lv> parents: 
76diff
changeset | 145 verdict = 'memory limit exceeded' | 
| 22 | 146 except testcases.WrongAnswer: | 
| 147 e = sys.exc_info()[1] | |
| 148 if e.comment: | |
| 149 verdict = 'wrong answer (%s)' % e.comment | |
| 150 else: | |
| 151 verdict = 'wrong answer' | |
| 152 except testcases.NonZeroExitCode: | |
| 153 e = sys.exc_info()[1] | |
| 154 if e.exitcode < 0: | |
| 155 if sys.platform == 'win32': | |
| 156 verdict = 'terminated with error 0x%X' % (e.exitcode + 0x100000000) | |
| 157 elif -e.exitcode in signalnames: | |
| 158 verdict = 'terminated by signal %d (%s)' % (-e.exitcode, signalnames[-e.exitcode]) | |
| 159 else: | |
| 160 verdict = 'terminated by signal %d' % -e.exitcode | |
| 21 | 161 else: | 
| 22 | 162 verdict = 'non-zero return code %d' % e.exitcode | 
| 163 except testcases.CannotStartTestee: | |
| 26 | 164 verdict = 'cannot launch the program to test%s' % strerror(sys.exc_info()[1].upstream) | 
| 22 | 165 except testcases.CannotStartValidator: | 
| 26 | 166 verdict = 'cannot launch the validator%s' % strerror(sys.exc_info()[1].upstream) | 
| 22 | 167 except testcases.CannotReadOutputFile: | 
| 26 | 168 verdict = 'cannot read the output file%s' % strerror(sys.exc_info()[1].upstream) | 
| 22 | 169 except testcases.CannotReadInputFile: | 
| 26 | 170 verdict = 'cannot read the input file%s' % strerror(sys.exc_info()[1].upstream) | 
| 22 | 171 except testcases.CannotReadAnswerFile: | 
| 26 | 172 verdict = 'cannot read the reference output file%s' % strerror(sys.exc_info()[1].upstream) | 
| 94 | 173 except testcases.ExceptionWrapper: | 
| 174 verdict = 'unspecified reason [this may be a bug in test.py]%s' % strerror(sys.exc_info()[1].upstream) | |
| 22 | 175 except testcases.TestCaseNotPassed: | 
| 26 | 176 verdict = 'unspecified reason [this may be a bug in test.py]%s' % strerror(sys.exc_info()[1]) | 
| 22 | 177 #except Exception: | 
| 26 | 178 # verdict = 'unknown error [this may be a bug in test.py]%s' % strerror(sys.exc_info()[1]) | 
| 21 | 179 else: | 
| 39 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 180 try: | 
| 22 | 181 granted, comment = granted | 
| 39 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 182 except TypeError: | 
| 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 183 comment = '' | 
| 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 184 else: | 
| 22 | 185 if comment: | 
| 186 comment = ' (%s)' % comment | |
| 24 
c23d81f4a1a3
Score returned by TestCase.__call__() is now normalized to 0..1
 Oleg Oshmyan <chortos@inbox.lv> parents: 
23diff
changeset | 187 if granted >= 1: | 
| 39 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 188 contexts[-1].case_correct() | 
| 90 
1fb319ec33af
Skimming mode added (-k/--skim option)
 Oleg Oshmyan <chortos@inbox.lv> parents: 
88diff
changeset | 189 prob.testcases.send(True) | 
| 22 | 190 verdict = 'OK' + comment | 
| 191 elif not granted: | |
| 192 verdict = 'wrong answer' + comment | |
| 193 else: | |
| 194 verdict = 'partly correct' + comment | |
| 26 | 195 granted *= case.points | 
| 22 | 196 say('%g/%g, %s' % (granted, case.points, verdict)) | 
| 76 
0e5ae28e0b2b
Points are now weighted on a test context basis
 Oleg Oshmyan <chortos@inbox.lv> parents: 
75diff
changeset | 197 contexts[-1].case_end() | 
| 
0e5ae28e0b2b
Points are now weighted on a test context basis
 Oleg Oshmyan <chortos@inbox.lv> parents: 
75diff
changeset | 198 contexts[-1].score(granted, case.points) | 
| 39 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 199 weighted = contexts[0].real * prob.config.taskweight / contexts[0].max if contexts[0].max else 0 | 
| 76 
0e5ae28e0b2b
Points are now weighted on a test context basis
 Oleg Oshmyan <chortos@inbox.lv> parents: 
75diff
changeset | 200 before_weighting = valued = '' | 
| 
0e5ae28e0b2b
Points are now weighted on a test context basis
 Oleg Oshmyan <chortos@inbox.lv> parents: 
75diff
changeset | 201 if prob.config.taskweight != contexts[0].max: | 
| 
0e5ae28e0b2b
Points are now weighted on a test context basis
 Oleg Oshmyan <chortos@inbox.lv> parents: 
75diff
changeset | 202 before_weighting = ' (%g/%g before weighting)' % (contexts[0].real, contexts[0].max) | 
| 39 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 203 if contexts[0].nvalued != contexts[0].ntotal: | 
| 76 
0e5ae28e0b2b
Points are now weighted on a test context basis
 Oleg Oshmyan <chortos@inbox.lv> parents: 
75diff
changeset | 204 valued = ' (%d/%d valued)' % (contexts[0].ncorrectvalued, contexts[0].nvalued) | 
| 
0e5ae28e0b2b
Points are now weighted on a test context basis
 Oleg Oshmyan <chortos@inbox.lv> parents: 
75diff
changeset | 205 say('Problem total: %d/%d tests%s, %g/%g points%s' % (contexts[0].ncorrect, contexts[0].ntotal, valued, weighted, prob.config.taskweight, before_weighting)) | 
| 39 
2b459f9743b4
Test groups are now supported
 Oleg Oshmyan <chortos@inbox.lv> parents: 
26diff
changeset | 206 sys.stdout.flush() | 
| 22 | 207 return weighted, prob.config.taskweight | 
| 208 finally: | |
| 90 
1fb319ec33af
Skimming mode added (-k/--skim option)
 Oleg Oshmyan <chortos@inbox.lv> parents: 
88diff
changeset | 209 if options.erase and case and case.has_iofiles: | 
| 22 | 210 for var in 'in', 'out': | 
| 211 name = getattr(prob.config, var + 'name') | |
| 212 if name: | |
| 213 try: | |
| 214 os.remove(name) | |
| 215 except Exception: | |
| 216 pass | |
| 90 
1fb319ec33af
Skimming mode added (-k/--skim option)
 Oleg Oshmyan <chortos@inbox.lv> parents: 
88diff
changeset | 217 if case.has_ansfile: | 
| 22 | 218 if prob.config.ansname: | 
| 219 try: | |
| 220 os.remove(prob.config.ansname) | |
| 221 except Exception: | |
| 222 pass | 
