Description:
shows submission timeouts
Commit status:
[Not Reviewed]
References:
Comments:
0 Commit comments 0 Inline Comments
Unresolved TODOs:
There are no unresolved TODOs
Add another comment

r382:c141e39c9118 - - 4 files changed: 57 inserted, 22 deleted

@@ -0,0 +1,37
1 + var TOIContest = {
2 + NO_TIMEOUT: -1,
3 +
4 + timeOuts: {},
5 +
6 + problemSelectClick: function() {
7 + $$(".submission-submit-divs").each(function(item) {
8 + item.hide();
9 + });
10 + var problem_id = $('submission_problem_id').value;
11 + if ( problem_id < 0 ) {
12 + return;
13 + }
14 + $("submission_submit_div_" + problem_id + "_id").show();
15 + },
16 +
17 + confirmDownload: function() {
18 + return confirm("แน่ใจ?");
19 + },
20 +
21 + refreshTimeOutMessages: function() {
22 + for ( var pid in TOIContest.timeOuts ) {
23 + var timeOut = TOIContest.timeOuts[ pid ];
24 + if ( timeOut != TOIContest.NO_TIMEOUT ) {
25 + if ( timeOut > 0 ) {
26 + var minLeft = parseInt(timeOut / 60);
27 + var secLeft = parseInt(timeOut % 60);
28 + $('submission_time_left_' + pid + '_id').innerHTML = '| <b>เหลือเวลาอีก ' + minLeft + ':' + secLeft + ' นาที</b>';
29 + } else {
30 + $('submission_time_left_' + pid + '_id').innerHTML = '| <b>หมดเวลาส่ง</a>';
31 + $('submission_form_'+ pid + '_id').hide();
32 + }
33 + }
34 + }
35 + }
36 + };
37 +
@@ -1,478 +1,482
1 1 class MainController < ApplicationController
2 2
3 3 before_filter :authenticate, :except => [:index, :login]
4 4 before_filter :check_viewability, :except => [:index, :login]
5 5
6 6 append_before_filter :confirm_and_update_start_time,
7 7 :except => [:index,
8 8 :login,
9 9 :confirm_contest_start]
10 10
11 11 # to prevent log in box to be shown when user logged out of the
12 12 # system only in some tab
13 13 prepend_before_filter :reject_announcement_refresh_when_logged_out,
14 14 :only => [:announcements]
15 15
16 16 # COMMENTED OUT: filter in each action instead
17 17 # before_filter :verify_time_limit, :only => [:submit]
18 18
19 19 verify :method => :post, :only => [:submit],
20 20 :redirect_to => { :action => :index }
21 21
22 22 # COMMENT OUT: only need when having high load
23 23 # caches_action :index, :login
24 24
25 25 # NOTE: This method is not actually needed, 'config/routes.rb' has
26 26 # assigned action login as a default action.
27 27 def index
28 28 redirect_to :action => 'login'
29 29 end
30 30
31 31 def login
32 32 saved_notice = flash[:notice]
33 33 reset_session
34 34 flash.now[:notice] = saved_notice
35 35
36 36 # EXPERIMENT:
37 37 # Hide login if in single user mode and the url does not
38 38 # explicitly specify /login
39 39 #
40 40 # logger.info "PATH: #{request.path}"
41 41 # if GraderConfiguration['system.single_user_mode'] and
42 42 # request.path!='/main/login'
43 43 # @hidelogin = true
44 44 # end
45 45
46 46 @announcements = Announcement.find_for_frontpage
47 47 render :action => 'login', :layout => 'empty'
48 48 end
49 49
50 50 def list
51 51 prepare_list_information
52 52 end
53 53
54 54 def help
55 55 @user = User.find(session[:user_id])
56 56 end
57 57
58 58 def submit
59 59 user = User.find(session[:user_id])
60 60
61 61 @submission = Submission.new
62 62 @submission.problem_id = params[:submission][:problem_id]
63 63 @submission.user = user
64 64 @submission.language_id = 0
65 65 if (params['file']) and (params['file']!='')
66 66 @submission.source = params['file'].read
67 67 @submission.source_filename = params['file'].original_filename
68 68 end
69 69 @submission.submitted_at = Time.new.gmtime
70 70
71 71 if GraderConfiguration.time_limit_mode? and user.contest_finished?
72 72 @submission.errors.add(:base,"The contest is over.")
73 73 prepare_list_information
74 74 render :action => 'list' and return
75 75 end
76 76
77 77 if @submission.valid?
78 78 if @submission.save == false
79 79 flash[:notice] = 'Error saving your submission'
80 80 elsif Task.create(:submission_id => @submission.id,
81 81 :status => Task::STATUS_INQUEUE) == false
82 82 flash[:notice] = 'Error adding your submission to task queue'
83 83 end
84 84 else
85 85 prepare_list_information
86 86 render :action => 'list' and return
87 87 end
88 88 redirect_to :action => 'list'
89 89 end
90 90
91 91 def source
92 92 submission = Submission.find(params[:id])
93 93 if ((submission.user_id == session[:user_id]) and
94 94 (submission.problem != nil) and
95 95 (submission.problem.available))
96 96 send_data(submission.source,
97 97 {:filename => submission.download_filename,
98 98 :type => 'text/plain'})
99 99 else
100 100 flash[:notice] = 'Error viewing source'
101 101 redirect_to :action => 'list'
102 102 end
103 103 end
104 104
105 105 def compiler_msg
106 106 @submission = Submission.find(params[:id])
107 107 if @submission.user_id == session[:user_id]
108 108 render :action => 'compiler_msg', :layout => 'empty'
109 109 else
110 110 flash[:notice] = 'Error viewing source'
111 111 redirect_to :action => 'list'
112 112 end
113 113 end
114 114
115 115 def submission
116 116 @user = User.find(session[:user_id])
117 117 @problems = @user.available_problems
118 118 if params[:id]==nil
119 119 @problem = nil
120 120 @submissions = nil
121 121 else
122 122 @problem = Problem.find_by_name(params[:id])
123 123 if not @problem.available
124 124 redirect_to :action => 'list'
125 125 flash[:notice] = 'Error: submissions for that problem are not viewable.'
126 126 return
127 127 end
128 128 @submissions = Submission.find_all_by_user_problem(@user.id, @problem.id)
129 129 end
130 130 end
131 131
132 132 def result
133 133 if !GraderConfiguration.show_grading_result
134 134 redirect_to :action => 'list' and return
135 135 end
136 136 @user = User.find(session[:user_id])
137 137 @submission = Submission.find(params[:id])
138 138 if @submission.user!=@user
139 139 flash[:notice] = 'You are not allowed to view result of other users.'
140 140 redirect_to :action => 'list' and return
141 141 end
142 142 prepare_grading_result(@submission)
143 143 end
144 144
145 145 def load_output
146 146 if !GraderConfiguration.show_grading_result or params[:num]==nil
147 147 redirect_to :action => 'list' and return
148 148 end
149 149 @user = User.find(session[:user_id])
150 150 @submission = Submission.find(params[:id])
151 151 if @submission.user!=@user
152 152 flash[:notice] = 'You are not allowed to view result of other users.'
153 153 redirect_to :action => 'list' and return
154 154 end
155 155 case_num = params[:num].to_i
156 156 out_filename = output_filename(@user.login,
157 157 @submission.problem.name,
158 158 @submission.id,
159 159 case_num)
160 160 if !FileTest.exists?(out_filename)
161 161 flash[:notice] = 'Output not found.'
162 162 redirect_to :action => 'list' and return
163 163 end
164 164
165 165 if defined?(USE_APACHE_XSENDFILE) and USE_APACHE_XSENDFILE
166 166 response.headers['Content-Type'] = "application/force-download"
167 167 response.headers['Content-Disposition'] = "attachment; filename=\"output-#{case_num}.txt\""
168 168 response.headers["X-Sendfile"] = out_filename
169 169 response.headers['Content-length'] = File.size(out_filename)
170 170 render :nothing => true
171 171 else
172 172 send_file out_filename, :stream => false, :filename => "output-#{case_num}.txt", :type => "text/plain"
173 173 end
174 174 end
175 175
176 176 def error
177 177 @user = User.find(session[:user_id])
178 178 end
179 179
180 180 # announcement refreshing and hiding methods
181 181
182 182 def announcements
183 183 if params.has_key? 'recent'
184 184 prepare_announcements(params[:recent])
185 185 else
186 186 prepare_announcements
187 187 end
188 188 render(:partial => 'announcement',
189 189 :collection => @announcements,
190 190 :locals => {:announcement_effect => true})
191 191 end
192 192
193 193 def confirm_contest_start
194 194 user = User.find(session[:user_id])
195 195 if request.method == 'POST'
196 196 user.update_start_time
197 197 redirect_to :action => 'list'
198 198 else
199 199 @contests = user.contests
200 200 @user = user
201 201 end
202 202 end
203 203
204 204 # thailandoi contests
205 205
206 206 def verifying_testcase
207 207 problem = Problem.find(params[:id])
208 208 if !problem.available
209 209 flash[:notice] = 'Error: problem is not available'
210 210 redirect_to :action => 'list'
211 211 else
212 212 test_pair = TestPair.get_for(problem, false)
213 213 send_data(test_pair.input,
214 214 {:filename => problem.name + '-verifying-input.txt',
215 215 :type => 'text/plain'})
216 216 end
217 217 end
218 218
219 219 def testcase
220 220 problem = Problem.find(params[:id])
221 221 if !problem.available
222 222 flash[:notice] = 'Error: problem is not available'
223 223 redirect_to :action => 'list'
224 224 else
225 225 test_pair = TestPair.get_for(problem, true)
226 226
227 227 user = User.find(session[:user_id])
228 228 assignent = user.get_test_pair_assignment_for(problem)
229 229
230 230 if !assignent
231 231 assignent = TestPairAssignment.new
232 232 assignent.user = user
233 233 assignent.problem = problem
234 234 assignent.test_pair = test_pair
235 235 assignent.submitted = false
236 236 assignent.save
237 237 end
238 238
239 239 send_data(test_pair.input,
240 240 {:filename => problem.name + '-input.txt',
241 241 :type => 'text/plain'})
242 242 end
243 243 end
244 244
245 245 def verifying_submit
246 246 user = User.find(session[:user_id])
247 247 problem_id = params[:id]
248 248 problem = Problem.find(problem_id)
249 249
250 250 if !problem or !problem.available
251 251 flash[:notice] = 'Error: problem is not available'
252 252 redirect_to :action => 'list' and return
253 253 end
254 254
255 255 test_pair = TestPair.get_for(problem, false)
256 256 if (params['output_file']) and (params['output_file']!='')
257 257 output = params['output_file'].read
258 258
259 259 @current_problem = problem
260 260 @grading_result = grade(output, test_pair.solution)
261 261 prepare_list_information
262 262 render :action => 'list' and return
263 263 else
264 264 flash[:notice] = 'Error: output file errors'
265 265 redirect_to :action => 'list'
266 266 end
267 267 end
268 268
269 269 protected
270 270
271 271 def grade(output, solution)
272 272 out_items = output.split
273 273 sol_items = solution.split
274 274 res = ''
275 + f = 0
276 + s = 0
275 277 sol_items.length.times do |i|
278 + f += 1
276 279 if out_items[i] == sol_items[i]
277 280 res = res + 'P'
281 + s += 1
278 282 else
279 283 res = res + '-'
280 284 end
281 285 end
282 - return res
286 + return { :score => s, :full_score => f, :msg => res }
283 287 end
284 288
285 289 def prepare_announcements(recent=nil)
286 290 if GraderConfiguration.show_tasks_to?(@user)
287 291 @announcements = Announcement.find_published(true)
288 292 else
289 293 @announcements = Announcement.find_published
290 294 end
291 295 if recent!=nil
292 296 recent_id = recent.to_i
293 297 @announcements = @announcements.find_all { |a| a.id > recent_id }
294 298 end
295 299 end
296 300
297 301 def prepare_timeout_information(problems)
298 302 @submission_timeouts = {}
299 303 problems.each do |problem|
300 304 assignment = @user.get_test_pair_assignment_for(problem)
301 305 if assignment == nil
302 306 timeout = nil
303 307 else
304 308 if (assignment.expired?) or (assignment.submitted)
305 309 timeout = 0
306 310 else
307 311 timeout = assignment.created_at + TEST_ASSIGNMENT_EXPIRATION_DURATION - Time.new.gmtime
308 312 end
309 313 end
310 314 @submission_timeouts[problem.id] = timeout
311 315 end
312 316 end
313 317
314 318 def prepare_list_information
315 319 @user = User.find(session[:user_id])
316 320 if not GraderConfiguration.multicontests?
317 321 @problems = @user.available_problems
318 322 else
319 323 @contest_problems = @user.available_problems_group_by_contests
320 324 @problems = @user.available_problems
321 325 end
322 326 @prob_submissions = {}
323 327 @problems.each do |p|
324 328 sub = Submission.find_last_by_user_and_problem(@user.id,p.id)
325 329 if sub!=nil
326 330 @prob_submissions[p.id] = { :count => sub.number, :submission => sub }
327 331 else
328 332 @prob_submissions[p.id] = { :count => 0, :submission => nil }
329 333 end
330 334 end
331 335 prepare_announcements
332 336 prepare_timeout_information(@problems)
333 337 end
334 338
335 339 def check_viewability
336 340 @user = User.find(session[:user_id])
337 341 if (!GraderConfiguration.show_tasks_to?(@user)) and
338 342 ((action_name=='submission') or (action_name=='submit'))
339 343 redirect_to :action => 'list' and return
340 344 end
341 345 end
342 346
343 347 def prepare_grading_result(submission)
344 348 if GraderConfiguration.task_grading_info.has_key? submission.problem.name
345 349 grading_info = GraderConfiguration.task_grading_info[submission.problem.name]
346 350 else
347 351 # guess task info from problem.full_score
348 352 cases = submission.problem.full_score / 10
349 353 grading_info = {
350 354 'testruns' => cases,
351 355 'testcases' => cases
352 356 }
353 357 end
354 358 @test_runs = []
355 359 if grading_info['testruns'].is_a? Integer
356 360 trun_count = grading_info['testruns']
357 361 trun_count.times do |i|
358 362 @test_runs << [ read_grading_result(@user.login,
359 363 submission.problem.name,
360 364 submission.id,
361 365 i+1) ]
362 366 end
363 367 else
364 368 grading_info['testruns'].keys.sort.each do |num|
365 369 run = []
366 370 testrun = grading_info['testruns'][num]
367 371 testrun.each do |c|
368 372 run << read_grading_result(@user.login,
369 373 submission.problem.name,
370 374 submission.id,
371 375 c)
372 376 end
373 377 @test_runs << run
374 378 end
375 379 end
376 380 end
377 381
378 382 def grading_result_dir(user_name, problem_name, submission_id, case_num)
379 383 return "#{GRADING_RESULT_DIR}/#{user_name}/#{problem_name}/#{submission_id}/test-result/#{case_num}"
380 384 end
381 385
382 386 def output_filename(user_name, problem_name, submission_id, case_num)
383 387 dir = grading_result_dir(user_name,problem_name, submission_id, case_num)
384 388 return "#{dir}/output.txt"
385 389 end
386 390
387 391 def read_grading_result(user_name, problem_name, submission_id, case_num)
388 392 dir = grading_result_dir(user_name,problem_name, submission_id, case_num)
389 393 result_file_name = "#{dir}/result"
390 394 if !FileTest.exists?(result_file_name)
391 395 return {:num => case_num, :msg => 'program did not run'}
392 396 else
393 397 results = File.open(result_file_name).readlines
394 398 run_stat = extract_running_stat(results)
395 399 output_filename = "#{dir}/output.txt"
396 400 if FileTest.exists?(output_filename)
397 401 output_file = true
398 402 output_size = File.size(output_filename)
399 403 else
400 404 output_file = false
401 405 output_size = 0
402 406 end
403 407
404 408 return {
405 409 :num => case_num,
406 410 :msg => results[0],
407 411 :run_stat => run_stat,
408 412 :output => output_file,
409 413 :output_size => output_size
410 414 }
411 415 end
412 416 end
413 417
414 418 # copied from grader/script/lib/test_request_helper.rb
415 419 def extract_running_stat(results)
416 420 running_stat_line = results[-1]
417 421
418 422 # extract exit status line
419 423 run_stat = ""
420 424 if !(/[Cc]orrect/.match(results[0]))
421 425 run_stat = results[0].chomp
422 426 else
423 427 run_stat = 'Program exited normally'
424 428 end
425 429
426 430 logger.info "Stat line: #{running_stat_line}"
427 431
428 432 # extract running time
429 433 if res = /r(.*)u(.*)s/.match(running_stat_line)
430 434 seconds = (res[1].to_f + res[2].to_f)
431 435 time_stat = "Time used: #{seconds} sec."
432 436 else
433 437 seconds = nil
434 438 time_stat = "Time used: n/a sec."
435 439 end
436 440
437 441 # extract memory usage
438 442 if res = /s(.*)m/.match(running_stat_line)
439 443 memory_used = res[1].to_i
440 444 else
441 445 memory_used = -1
442 446 end
443 447
444 448 return {
445 449 :msg => "#{run_stat}\n#{time_stat}",
446 450 :running_time => seconds,
447 451 :exit_status => run_stat,
448 452 :memory_usage => memory_used
449 453 }
450 454 end
451 455
452 456 def confirm_and_update_start_time
453 457 user = User.find(session[:user_id])
454 458 if (GraderConfiguration.indv_contest_mode? and
455 459 GraderConfiguration['contest.confirm_indv_contest_start'] and
456 460 !user.contest_started?)
457 461 redirect_to :action => 'confirm_contest_start' and return
458 462 end
459 463 if not GraderConfiguration.analysis_mode?
460 464 user.update_start_time
461 465 end
462 466 end
463 467
464 468 def reject_announcement_refresh_when_logged_out
465 469 if not session[:user_id]
466 470 render :text => 'Access forbidden', :status => 403
467 471 end
468 472
469 473 if GraderConfiguration.multicontests?
470 474 user = User.find(session[:user_id])
471 475 if user.contest_stat.forced_logout
472 476 render :text => 'Access forbidden', :status => 403
473 477 end
474 478 end
475 479 end
476 480
477 481 end
478 482
@@ -1,51 +1,36
1 1 <div class="submitbox">
2 2 <b>Problem:</b> <%= select 'submission', 'problem_id',
3 3 [['กรุณาเลือกข้อที่ต้องการส่งหรือทดสอบ','-1']] +
4 4 @problems.collect {|p| [p.full_name, p.id]},
5 5 { :selected => '-1' },
6 - { :onchange => 'select_click()' } %>
6 + { :onchange => 'TOIContest.problemSelectClick()' } %>
7 7 </div>
8 8
9 9 <% @problems.each do |problem| %>
10 10 <div class="submission-submit-divs" id="submission_submit_div_<%= problem.id %>_id" style="displanny: none;">
11 11 <div style="border: 1px solid #c0c0c0; padding: 5px; margin: 5px 0 5px 0;">
12 - <b><%= problem.full_name %>: ข้อมูลสำหรับตรวจสอบ</b> (สามารถดาวน์โหลดและส่งกี่ครั้งก็ได้):
12 + <b><%= problem.full_name %>: ข้อมูลสำหรับตรวจสอบ</b> (สามารถดาวน์โหลดและส่งกี่ครั้งก็ได้,ไม่มีคะแนน):
13 13 <%= link_to 'ดาวน์โหลด input', :action => 'verifying_testcase', :id => problem.id %>
14 - <% if @current_problem.id == problem.id %>
14 + <% if @current_problem and @current_problem.id == problem.id %>
15 15 <% if @grading_result %>
16 - | <b>ผลการตรวจ:</b> <%= @grading_result %>
16 + | <b>ผลการตรวจ:</b> <%= "#{@grading_result[:score]}/#{@grading_result[:full_score]} [#{@grading_result[:msg]}]" %>
17 17 <% end %>
18 18 <% end %>
19 19 <%= form_tag({:controller => 'main', :action => 'verifying_submit', :id => problem.id}, {:method => 'post', :multipart => true}) do %>
20 20 ส่งคำตอบของข้อมูลสำหรับตรวจสอบ:
21 21 <%= file_field_tag 'output_file' %>
22 22 <%= submit_tag 'Submit' %>
23 23 <% end %>
24 24 </div>
25 25 <div style="border: 1px solid #c0c0c0; padding: 5px; margin: 5px 0 5px 0;">
26 26 <b><%= problem.full_name %>: ข้อมูลทดสอบจริง</b> (ส่งกี่ครั้งก็ได้ภายในเวลา 5 นาทีหลังดาวน์โหลด):
27 - <%= link_to 'ดาวน์โหลด input และเริ่มจับเวลา', { :action => 'testcase', :id => problem.id}, { :onclick => 'return confirm_download()' } %>
27 + <%= link_to 'ดาวน์โหลด input และเริ่มจับเวลา', { :action => 'testcase', :id => problem.id}, { :onclick => 'return TOIContest.confirmDownload()' } %>
28 28 <span id="submission_time_left_<%= problem.id %>_id"></span>
29 - <%= form_tag do %>
29 + <%= form_tag({:controller => 'main', :action => 'submit', :id => problem.id}, {:method => 'post', :multipart => true, :id => "submission_form_#{problem.id}_id" }) do %>
30 30 ข้อมูลส่งออก: <%= file_field_tag 'output_file' %>
31 31 โปรแกรมคำตอบ: <%= file_field_tag 'source_file' %>
32 32 <%= submit_tag 'Submit' %>
33 33 <% end %>
34 34 </div>
35 35 </div>
36 36 <% end %>
37 - <script>
38 - function select_click() {
39 - $$(".submission-submit-divs").each(function(item) {
40 - item.hide();
41 - });
42 - var problem_id = $('submission_problem_id').value;
43 - if ( problem_id < 0 ) {
44 - return;
45 - }
46 - $("submission_submit_div_" + problem_id + "_id").show();
47 - }
48 - function confirm_download() {
49 - return confirm("แน่ใจ?");
50 - }
51 - </script>
@@ -1,50 +1,59
1 1 - content_for :head do
2 2 = javascript_include_tag "announcement_refresh"
3 + = javascript_include_tag "toicontest"
3 4
4 5 = user_title_bar(@user)
5 6
6 7 .announcementbox{:style => (@announcements.length==0 ? "display:none" : "")}
7 8 %span{:class => 'title'}
8 9 Announcements
9 10 #announcementbox-body
10 11 = render :partial => 'announcement', :collection => @announcements
11 12
12 13 - if GraderConfiguration.show_submitbox_to?(@user)
13 14 = error_messages_for 'submission'
14 15 = render :partial => 'submission_box'
15 16
16 17
17 18 %hr/
18 19
19 20 - if (GraderConfiguration.contest_mode?) and (@user.site!=nil) and (@user.site.started!=true)
20 21 %p=t 'main.start_soon'
21 22
22 23 - if GraderConfiguration.show_tasks_to?(@user)
23 24 - if not GraderConfiguration.multicontests?
24 25 %table.info
25 26 %tr.info-head
26 27 %th
27 28 %th Tasks
28 29 %th # of sub(s)
29 30 %th Results
30 31 = render :partial => 'problem', :collection => @problems
31 32 - else
32 33 - @contest_problems.each do |cp|
33 34 - if cp[:problems].length > 0
34 35 %h2{:class =>'contest-title'}
35 36 = "#{cp[:contest] ? cp[:contest].title : 'Public problems'}"
36 37 %table.info
37 38 %tr.info-head
38 39 %th
39 40 %th Tasks
40 41 %th # of sub(s)
41 42 %th Results
42 43 = render :partial => 'problem', :collection => cp[:problems]
43 44
44 45
45 46 %hr/
46 47
47 48 %script{:type => 'text/javascript'}
48 49 = "Announcement.refreshUrl = '#{url_for :controller => 'main', :action => 'announcements'}';"
49 50 Announcement.registerRefreshEventTimer();
50 51
52 + TOIContest.timeOuts = {};
53 + - @problems.each do |p|
54 + - if (@submission_timeouts.has_key? p.id) and (@submission_timeouts[p.id] != nil)
55 + = "TOIContest.timeOuts[#{p.id}] = #{@submission_timeouts[p.id]};"
56 + - else
57 + = "TOIContest.timeOuts[#{p.id}] = TOIContest.NO_TIMEOUT;"
58 +
59 + TOIContest.refreshTimeOutMessages();
You need to be logged in to leave comments. Login now