Description:
a cleaner, testable way to log out user after contest changed
Commit status:
[Not Reviewed]
References:
Comments:
0 Commit comments 0 Inline Comments
Unresolved TODOs:
There are no unresolved TODOs
Add another comment

r295:4f81b9ab5d77 - - 11 files changed: 170 inserted, 18 deleted

@@ -0,0 +1,9
1 + class AddForcedLogoutToUserContestStat < ActiveRecord::Migration
2 + def self.up
3 + add_column :user_contest_stats, :forced_logout, :boolean
4 + end
5 +
6 + def self.down
7 + remove_column :user_contest_stats, :forced_logout, :boolean
8 + end
9 + end
@@ -0,0 +1,73
1 + require 'delorean'
2 +
3 + require File.dirname(__FILE__) + '/../spec_helper'
4 + require File.dirname(__FILE__) + '/../config_spec_helper'
5 +
6 + describe UserAdminController, "when manage contest" do
7 +
8 + include ConfigSpecHelperMethods
9 +
10 + fixtures :users
11 + fixtures :problems
12 + fixtures :contests
13 + fixtures :roles
14 +
15 + def change_users_contest_to(user_login_list, contest, reset_timer=false)
16 + post_data = {
17 + :contest => {:id => contest.id},
18 + :operation => 'assign',
19 + :login_list => user_login_list
20 + }
21 + post_data[:reset_timer] = true if reset_timer
22 + post 'manage_contest', post_data, {:user_id => @admin_user.id}
23 + end
24 +
25 + before(:each) do
26 + @admin_user = users(:mary)
27 + @contest_b = contests(:contest_b)
28 + @james = users(:james)
29 + @jack = users(:jack)
30 +
31 + set_contest_time_limit('3:00')
32 + set_indv_contest_mode
33 + end
34 +
35 + it "should allow admin to see contest management page" do
36 + get 'contest_management', {}, {:user_id => @admin_user.id}
37 +
38 + response.should render_template 'user_admin/contest_management'
39 + end
40 +
41 + it "should change users' contest" do
42 + change_users_contest_to("james\njack", @contest_b)
43 + response.should redirect_to :action => 'contest_management'
44 +
45 + @james.contests(true).should include @contest_b
46 + @jack.contests(true).should_not include @contest_a
47 + end
48 +
49 + it "should reset users' timer when their contests change" do
50 + @james.update_start_time
51 +
52 + Delorean.time_travel_to(190.minutes.since) do
53 + @james.contest_finished?.should be_true
54 +
55 + change_users_contest_to("james", @contest_b, true)
56 +
57 + @james.contest_finished?.should be_false
58 + end
59 + end
60 +
61 + it "should set forced_logout flag for users when their contests change" do
62 + @james.update_start_time
63 +
64 + Delorean.time_travel_to(190.minutes.since) do
65 + @james.contest_finished?.should be_true
66 +
67 + change_users_contest_to("james", @contest_b, true)
68 +
69 + @james.contest_stat(true).forced_logout.should be_true
70 + end
71 + end
72 +
73 + end
@@ -1,74 +1,84
1 # Filters added to this controller apply to all controllers in the application.
1 # Filters added to this controller apply to all controllers in the application.
2 # Likewise, all the methods added will be available for all controllers.
2 # Likewise, all the methods added will be available for all controllers.
3
3
4 class ApplicationController < ActionController::Base
4 class ApplicationController < ActionController::Base
5
5
6 SINGLE_USER_MODE_CONF_KEY = 'system.single_user_mode'
6 SINGLE_USER_MODE_CONF_KEY = 'system.single_user_mode'
7
7
8 def admin_authorization
8 def admin_authorization
9 return false unless authenticate
9 return false unless authenticate
10 user = User.find(session[:user_id], :include => ['roles'])
10 user = User.find(session[:user_id], :include => ['roles'])
11 redirect_to :controller => 'main', :action => 'login' unless user.admin?
11 redirect_to :controller => 'main', :action => 'login' unless user.admin?
12 end
12 end
13
13
14 def authorization_by_roles(allowed_roles)
14 def authorization_by_roles(allowed_roles)
15 return false unless authenticate
15 return false unless authenticate
16 user = User.find(session[:user_id])
16 user = User.find(session[:user_id])
17 unless user.roles.detect { |role| allowed_roles.member?(role.name) }
17 unless user.roles.detect { |role| allowed_roles.member?(role.name) }
18 flash[:notice] = 'You are not authorized to view the page you requested'
18 flash[:notice] = 'You are not authorized to view the page you requested'
19 redirect_to :controller => 'main', :action => 'login'
19 redirect_to :controller => 'main', :action => 'login'
20 return false
20 return false
21 end
21 end
22 end
22 end
23
23
24 protected
24 protected
25
25
26 def authenticate
26 def authenticate
27 unless session[:user_id]
27 unless session[:user_id]
28 redirect_to :controller => 'main', :action => 'login'
28 redirect_to :controller => 'main', :action => 'login'
29 return false
29 return false
30 end
30 end
31
31
32 - #Configuration.reload
33 # check if run in single user mode
32 # check if run in single user mode
34 - if (Configuration[SINGLE_USER_MODE_CONF_KEY])
33 + if Configuration[SINGLE_USER_MODE_CONF_KEY]
35 user = User.find(session[:user_id])
34 user = User.find(session[:user_id])
36 if user==nil or (not user.admin?)
35 if user==nil or (not user.admin?)
37 redirect_to :controller => 'main', :action => 'login'
36 redirect_to :controller => 'main', :action => 'login'
38 return false
37 return false
39 end
38 end
39 + return true
40 end
40 end
41
41
42 + if Configuration.multicontests?
43 + user = User.find(session[:user_id])
44 + begin
45 + if user.contest_stat(true).forced_logout
46 + flash[:notice] = 'You have been automatically logged out.'
47 + redirect_to :controller => 'main', :action => 'index'
48 + end
49 + rescue
50 + end
51 + end
42 return true
52 return true
43 end
53 end
44
54
45 def authorization
55 def authorization
46 return false unless authenticate
56 return false unless authenticate
47 user = User.find(session[:user_id])
57 user = User.find(session[:user_id])
48 unless user.roles.detect { |role|
58 unless user.roles.detect { |role|
49 role.rights.detect{ |right|
59 role.rights.detect{ |right|
50 right.controller == self.class.controller_name and
60 right.controller == self.class.controller_name and
51 (right.action == 'all' or right.action == action_name)
61 (right.action == 'all' or right.action == action_name)
52 }
62 }
53 }
63 }
54 flash[:notice] = 'You are not authorized to view the page you requested'
64 flash[:notice] = 'You are not authorized to view the page you requested'
55 #request.env['HTTP_REFERER'] ? (redirect_to :back) : (redirect_to :controller => 'login')
65 #request.env['HTTP_REFERER'] ? (redirect_to :back) : (redirect_to :controller => 'login')
56 redirect_to :controller => 'main', :action => 'login'
66 redirect_to :controller => 'main', :action => 'login'
57 return false
67 return false
58 end
68 end
59 end
69 end
60
70
61 def verify_time_limit
71 def verify_time_limit
62 return true if session[:user_id]==nil
72 return true if session[:user_id]==nil
63 user = User.find(session[:user_id], :include => :site)
73 user = User.find(session[:user_id], :include => :site)
64 return true if user==nil or user.site == nil
74 return true if user==nil or user.site == nil
65 if user.contest_finished?
75 if user.contest_finished?
66 flash[:notice] = 'Error: the contest you are participating is over.'
76 flash[:notice] = 'Error: the contest you are participating is over.'
67 redirect_to :back
77 redirect_to :back
68 return false
78 return false
69 end
79 end
70 return true
80 return true
71 end
81 end
72
82
73 end
83 end
74
84
@@ -1,39 +1,51
1 class LoginController < ApplicationController
1 class LoginController < ApplicationController
2
2
3 def index
3 def index
4 # show login screen
4 # show login screen
5 reset_session
5 reset_session
6 redirect_to :controller => 'main', :action => 'login'
6 redirect_to :controller => 'main', :action => 'login'
7 end
7 end
8
8
9 def login
9 def login
10 if user = User.authenticate(params[:login], params[:password])
10 if user = User.authenticate(params[:login], params[:password])
11 session[:user_id] = user.id
11 session[:user_id] = user.id
12 session[:admin] = user.admin?
12 session[:admin] = user.admin?
13 +
14 + # clear forced logout flag for multicontests contest change
15 + if Configuration.multicontests?
16 + contest_stat = user.contest_stat
17 + if contest_stat.respond_to? :forced_logout
18 + if contest_stat.forced_logout
19 + contest_stat.forced_logout = false
20 + contest_stat.save
21 + end
22 + end
23 + end
24 +
13 redirect_to :controller => 'main', :action => 'list'
25 redirect_to :controller => 'main', :action => 'list'
14 else
26 else
15 flash[:notice] = 'Wrong password'
27 flash[:notice] = 'Wrong password'
16 redirect_to :controller => 'main', :action => 'login'
28 redirect_to :controller => 'main', :action => 'login'
17 end
29 end
18 end
30 end
19
31
20 def site_login
32 def site_login
21 begin
33 begin
22 site = Site.find(params[:login][:site_id])
34 site = Site.find(params[:login][:site_id])
23 rescue ActiveRecord::RecordNotFound
35 rescue ActiveRecord::RecordNotFound
24 site = nil
36 site = nil
25 end
37 end
26 if site==nil
38 if site==nil
27 flash[:notice] = 'Wrong site'
39 flash[:notice] = 'Wrong site'
28 redirect_to :controller => 'main', :action => 'login' and return
40 redirect_to :controller => 'main', :action => 'login' and return
29 end
41 end
30 if (site.password) and (site.password == params[:login][:password])
42 if (site.password) and (site.password == params[:login][:password])
31 session[:site_id] = site.id
43 session[:site_id] = site.id
32 redirect_to :controller => 'site', :action => 'index'
44 redirect_to :controller => 'site', :action => 'index'
33 else
45 else
34 flash[:notice] = 'Wrong site password'
46 flash[:notice] = 'Wrong site password'
35 redirect_to :controller => 'site', :action => 'login'
47 redirect_to :controller => 'site', :action => 'login'
36 end
48 end
37 end
49 end
38
50
39 end
51 end
@@ -298,52 +298,59
298
298
299 # copied from grader/script/lib/test_request_helper.rb
299 # copied from grader/script/lib/test_request_helper.rb
300 def extract_running_stat(results)
300 def extract_running_stat(results)
301 running_stat_line = results[-1]
301 running_stat_line = results[-1]
302
302
303 # extract exit status line
303 # extract exit status line
304 run_stat = ""
304 run_stat = ""
305 if !(/[Cc]orrect/.match(results[0]))
305 if !(/[Cc]orrect/.match(results[0]))
306 run_stat = results[0].chomp
306 run_stat = results[0].chomp
307 else
307 else
308 run_stat = 'Program exited normally'
308 run_stat = 'Program exited normally'
309 end
309 end
310
310
311 logger.info "Stat line: #{running_stat_line}"
311 logger.info "Stat line: #{running_stat_line}"
312
312
313 # extract running time
313 # extract running time
314 if res = /r(.*)u(.*)s/.match(running_stat_line)
314 if res = /r(.*)u(.*)s/.match(running_stat_line)
315 seconds = (res[1].to_f + res[2].to_f)
315 seconds = (res[1].to_f + res[2].to_f)
316 time_stat = "Time used: #{seconds} sec."
316 time_stat = "Time used: #{seconds} sec."
317 else
317 else
318 seconds = nil
318 seconds = nil
319 time_stat = "Time used: n/a sec."
319 time_stat = "Time used: n/a sec."
320 end
320 end
321
321
322 # extract memory usage
322 # extract memory usage
323 if res = /s(.*)m/.match(running_stat_line)
323 if res = /s(.*)m/.match(running_stat_line)
324 memory_used = res[1].to_i
324 memory_used = res[1].to_i
325 else
325 else
326 memory_used = -1
326 memory_used = -1
327 end
327 end
328
328
329 return {
329 return {
330 :msg => "#{run_stat}\n#{time_stat}",
330 :msg => "#{run_stat}\n#{time_stat}",
331 :running_time => seconds,
331 :running_time => seconds,
332 :exit_status => run_stat,
332 :exit_status => run_stat,
333 :memory_usage => memory_used
333 :memory_usage => memory_used
334 }
334 }
335 end
335 end
336
336
337 def update_user_start_time
337 def update_user_start_time
338 user = User.find(session[:user_id])
338 user = User.find(session[:user_id])
339 user.update_start_time
339 user.update_start_time
340 end
340 end
341
341
342 def reject_announcement_refresh_when_logged_out
342 def reject_announcement_refresh_when_logged_out
343 if not session[:user_id]
343 if not session[:user_id]
344 render :text => 'Access forbidden', :status => 403
344 render :text => 'Access forbidden', :status => 403
345 end
345 end
346 +
347 + if Configuration.multicontests?
348 + user = User.find(session[:user_id])
349 + if user.contest_stat.forced_logout
350 + render :text => 'Access forbidden', :status => 403
351 + end
352 + end
346 end
353 end
347
354
348 end
355 end
349
356
@@ -151,119 +151,122
151 @changed = true
151 @changed = true
152 end
152 end
153 end
153 end
154
154
155 # contest management
155 # contest management
156
156
157 def add_to_contest
157 def add_to_contest
158 user = User.find(params[:id])
158 user = User.find(params[:id])
159 contest = Contest.find(params[:contest_id])
159 contest = Contest.find(params[:contest_id])
160 if user and contest
160 if user and contest
161 user.contests << contest
161 user.contests << contest
162 end
162 end
163 redirect_to :action => 'list'
163 redirect_to :action => 'list'
164 end
164 end
165
165
166 def remove_from_contest
166 def remove_from_contest
167 user = User.find(params[:id])
167 user = User.find(params[:id])
168 contest = Contest.find(params[:contest_id])
168 contest = Contest.find(params[:contest_id])
169 if user and contest
169 if user and contest
170 user.contests.delete(contest)
170 user.contests.delete(contest)
171 end
171 end
172 redirect_to :action => 'list'
172 redirect_to :action => 'list'
173 end
173 end
174
174
175 def contest_management
175 def contest_management
176 end
176 end
177
177
178 def manage_contest
178 def manage_contest
179 contest = Contest.find(params[:contest][:id])
179 contest = Contest.find(params[:contest][:id])
180 if !contest
180 if !contest
181 flash[:notice] = 'You did not choose the contest.'
181 flash[:notice] = 'You did not choose the contest.'
182 redirect_to :action => 'contest_management' and return
182 redirect_to :action => 'contest_management' and return
183 end
183 end
184
184
185 operation = params[:operation]
185 operation = params[:operation]
186
186
187 if not ['add','remove','assign'].include? operation
187 if not ['add','remove','assign'].include? operation
188 flash[:notice] = 'You did not choose the operation to perform.'
188 flash[:notice] = 'You did not choose the operation to perform.'
189 redirect_to :action => 'contest_management' and return
189 redirect_to :action => 'contest_management' and return
190 end
190 end
191
191
192 lines = params[:login_list]
192 lines = params[:login_list]
193 if !lines or lines.blank?
193 if !lines or lines.blank?
194 flash[:notice] = 'You entered an empty list.'
194 flash[:notice] = 'You entered an empty list.'
195 redirect_to :action => 'contest_management' and return
195 redirect_to :action => 'contest_management' and return
196 end
196 end
197
197
198 note = []
198 note = []
199 - user_ids = {}
199 + users = []
200 lines.split("\n").each do |line|
200 lines.split("\n").each do |line|
201 user = User.find_by_login(line.chomp)
201 user = User.find_by_login(line.chomp)
202 if user
202 if user
203 if operation=='add'
203 if operation=='add'
204 if ! user.contests.include? contest
204 if ! user.contests.include? contest
205 user.contests << contest
205 user.contests << contest
206 end
206 end
207 elsif operation=='remove'
207 elsif operation=='remove'
208 user.contests.delete(contest)
208 user.contests.delete(contest)
209 else
209 else
210 user.contests = [contest]
210 user.contests = [contest]
211 end
211 end
212
212
213 - user.contest_stat.destroy if params[:reset_timer]
213 + if params[:reset_timer]
214 + user.contest_stat.forced_logout = true
215 + user.contest_stat.reset_timer_and_save
216 + end
214
217
215 note << user.login
218 note << user.login
216 - user_ids[user.id] = true
219 + users << user
217 end
220 end
218 end
221 end
219
222
220 if params[:reset_timer]
223 if params[:reset_timer]
221 - logout_users(user_ids)
224 + logout_users(users)
222 end
225 end
223
226
224 flash[:notice] = 'User(s) ' + note.join(', ') +
227 flash[:notice] = 'User(s) ' + note.join(', ') +
225 ' were successfully modified. '
228 ' were successfully modified. '
226 redirect_to :action => 'contest_management'
229 redirect_to :action => 'contest_management'
227 end
230 end
228
231
229 # admin management
232 # admin management
230
233
231 def admin
234 def admin
232 @admins = User.find(:all).find_all {|user| user.admin? }
235 @admins = User.find(:all).find_all {|user| user.admin? }
233 end
236 end
234
237
235 def grant_admin
238 def grant_admin
236 login = params[:login]
239 login = params[:login]
237 user = User.find_by_login(login)
240 user = User.find_by_login(login)
238 if user!=nil
241 if user!=nil
239 admin_role = Role.find_by_name('admin')
242 admin_role = Role.find_by_name('admin')
240 user.roles << admin_role
243 user.roles << admin_role
241 else
244 else
242 flash[:notice] = 'Unknown user'
245 flash[:notice] = 'Unknown user'
243 end
246 end
244 flash[:notice] = 'User added as admins'
247 flash[:notice] = 'User added as admins'
245 redirect_to :action => 'admin'
248 redirect_to :action => 'admin'
246 end
249 end
247
250
248 def revoke_admin
251 def revoke_admin
249 user = User.find(params[:id])
252 user = User.find(params[:id])
250 if user==nil
253 if user==nil
251 flash[:notice] = 'Unknown user'
254 flash[:notice] = 'Unknown user'
252 redirect_to :action => 'admin' and return
255 redirect_to :action => 'admin' and return
253 elsif user.login == 'root'
256 elsif user.login == 'root'
254 flash[:notice] = 'You cannot revoke admisnistrator permission from root.'
257 flash[:notice] = 'You cannot revoke admisnistrator permission from root.'
255 redirect_to :action => 'admin' and return
258 redirect_to :action => 'admin' and return
256 end
259 end
257
260
258 admin_role = Role.find_by_name('admin')
261 admin_role = Role.find_by_name('admin')
259 user.roles.delete(admin_role)
262 user.roles.delete(admin_role)
260 flash[:notice] = 'User permission revoked'
263 flash[:notice] = 'User permission revoked'
261 redirect_to :action => 'admin'
264 redirect_to :action => 'admin'
262 end
265 end
263
266
264 protected
267 protected
265
268
266 def random_password(length=5)
269 def random_password(length=5)
267 chars = 'abcdefghijkmnopqrstuvwxyz23456789'
270 chars = 'abcdefghijkmnopqrstuvwxyz23456789'
268 newpass = ""
271 newpass = ""
269 length.times { newpass << chars[rand(chars.size-1)] }
272 length.times { newpass << chars[rand(chars.size-1)] }
@@ -286,58 +289,59
286 countries[id] = c
289 countries[id] = c
287 @import_log << "Found #{country[:name]}\n"
290 @import_log << "Found #{country[:name]}\n"
288 else
291 else
289 countries[id] = Country.new(:name => country[:name])
292 countries[id] = Country.new(:name => country[:name])
290 countries[id].save
293 countries[id].save
291 @import_log << "Created #{country[:name]}\n"
294 @import_log << "Created #{country[:name]}\n"
292 end
295 end
293 end
296 end
294
297
295 # import sites
298 # import sites
296 sites = {}
299 sites = {}
297 site_data.each_pair do |id,site|
300 site_data.each_pair do |id,site|
298 s = Site.find_by_name(site[:name])
301 s = Site.find_by_name(site[:name])
299 if s!=nil
302 if s!=nil
300 @import_log << "Found #{site[:name]}\n"
303 @import_log << "Found #{site[:name]}\n"
301 else
304 else
302 s = Site.new(:name => site[:name])
305 s = Site.new(:name => site[:name])
303 @import_log << "Created #{site[:name]}\n"
306 @import_log << "Created #{site[:name]}\n"
304 end
307 end
305 s.password = site[:password]
308 s.password = site[:password]
306 s.country = countries[site[:country_id]]
309 s.country = countries[site[:country_id]]
307 s.save
310 s.save
308 sites[id] = s
311 sites[id] = s
309 end
312 end
310
313
311 # import users
314 # import users
312 user_data.each_pair do |id,user|
315 user_data.each_pair do |id,user|
313 u = User.find_by_login(user[:login])
316 u = User.find_by_login(user[:login])
314 if u!=nil
317 if u!=nil
315 @import_log << "Found #{user[:login]}\n"
318 @import_log << "Found #{user[:login]}\n"
316 else
319 else
317 u = User.new(:login => user[:login])
320 u = User.new(:login => user[:login])
318 @import_log << "Created #{user[:login]}\n"
321 @import_log << "Created #{user[:login]}\n"
319 end
322 end
320 u.full_name = user[:name]
323 u.full_name = user[:name]
321 u.password = user[:password]
324 u.password = user[:password]
322 u.country = countries[user[:country_id]]
325 u.country = countries[user[:country_id]]
323 u.site = sites[user[:site_id]]
326 u.site = sites[user[:site_id]]
324 u.activated = true
327 u.activated = true
325 u.email = "empty-#{u.login}@none.com"
328 u.email = "empty-#{u.login}@none.com"
326 if not u.save
329 if not u.save
327 @import_log << "Errors\n"
330 @import_log << "Errors\n"
328 u.errors.each { |attr,msg| @import_log << "#{attr} - #{msg}\n" }
331 u.errors.each { |attr,msg| @import_log << "#{attr} - #{msg}\n" }
329 end
332 end
330 end
333 end
331
334
332 end
335 end
333
336
334 - def logout_users(user_ids)
337 + def logout_users(users)
335 - sessions = ActiveRecord::SessionStore::Session.find(:all, :conditions => ["updated_at >= ?", 60.minutes.ago])
338 + users.each do |user|
336 - sessions.each do |session|
339 + contest_stat = user.contest_stat(true)
337 - if user_ids.has_key? session.data[:user_id]
340 + if contest_stat and !contest_stat.forced_logout
338 - session.destroy
341 + contest_stat.forced_logout = true
342 + contest_stat.save
339 end
343 end
340 end
344 end
341 end
345 end
342
346
343 end
347 end
@@ -88,138 +88,138
88 def alias_for_editing
88 def alias_for_editing
89 if self.alias==nil
89 if self.alias==nil
90 "(unknown)"
90 "(unknown)"
91 elsif self.alias==''
91 elsif self.alias==''
92 "(blank)"
92 "(blank)"
93 else
93 else
94 self.alias
94 self.alias
95 end
95 end
96 end
96 end
97
97
98 def alias_for_editing=(e)
98 def alias_for_editing=(e)
99 self.alias=e
99 self.alias=e
100 end
100 end
101
101
102 def activation_key
102 def activation_key
103 if self.hashed_password==nil
103 if self.hashed_password==nil
104 encrypt_new_password
104 encrypt_new_password
105 end
105 end
106 Digest::SHA1.hexdigest(self.hashed_password)[0..7]
106 Digest::SHA1.hexdigest(self.hashed_password)[0..7]
107 end
107 end
108
108
109 def verify_activation_key(key)
109 def verify_activation_key(key)
110 key == activation_key
110 key == activation_key
111 end
111 end
112
112
113 def self.random_password(length=5)
113 def self.random_password(length=5)
114 chars = 'abcdefghjkmnopqrstuvwxyz'
114 chars = 'abcdefghjkmnopqrstuvwxyz'
115 password = ''
115 password = ''
116 length.times { password << chars[rand(chars.length - 1)] }
116 length.times { password << chars[rand(chars.length - 1)] }
117 password
117 password
118 end
118 end
119
119
120 def self.find_non_admin_with_prefix(prefix='')
120 def self.find_non_admin_with_prefix(prefix='')
121 users = User.find(:all)
121 users = User.find(:all)
122 return users.find_all { |u| !(u.admin?) and u.login.index(prefix)==0 }
122 return users.find_all { |u| !(u.admin?) and u.login.index(prefix)==0 }
123 end
123 end
124
124
125 # Contest information
125 # Contest information
126
126
127 def contest_time_left
127 def contest_time_left
128 if Configuration.contest_mode?
128 if Configuration.contest_mode?
129 return nil if site==nil
129 return nil if site==nil
130 return site.time_left
130 return site.time_left
131 elsif Configuration.indv_contest_mode?
131 elsif Configuration.indv_contest_mode?
132 time_limit = Configuration.contest_time_limit
132 time_limit = Configuration.contest_time_limit
133 if time_limit == nil
133 if time_limit == nil
134 return nil
134 return nil
135 end
135 end
136 - if contest_stat==nil
136 + if contest_stat==nil or contest_stat.started_at==nil
137 return (Time.now.gmtime + time_limit) - Time.now.gmtime
137 return (Time.now.gmtime + time_limit) - Time.now.gmtime
138 else
138 else
139 finish_time = contest_stat.started_at + time_limit
139 finish_time = contest_stat.started_at + time_limit
140 current_time = Time.now.gmtime
140 current_time = Time.now.gmtime
141 if current_time > finish_time
141 if current_time > finish_time
142 return 0
142 return 0
143 else
143 else
144 return finish_time - current_time
144 return finish_time - current_time
145 end
145 end
146 end
146 end
147 else
147 else
148 return nil
148 return nil
149 end
149 end
150 end
150 end
151
151
152 def contest_finished?
152 def contest_finished?
153 if Configuration.contest_mode?
153 if Configuration.contest_mode?
154 return false if site==nil
154 return false if site==nil
155 return site.finished?
155 return site.finished?
156 elsif Configuration.indv_contest_mode?
156 elsif Configuration.indv_contest_mode?
157 return false if self.contest_stat(true)==nil
157 return false if self.contest_stat(true)==nil
158 return contest_time_left == 0
158 return contest_time_left == 0
159 else
159 else
160 return false
160 return false
161 end
161 end
162 end
162 end
163
163
164 def contest_started?
164 def contest_started?
165 if Configuration.contest_mode?
165 if Configuration.contest_mode?
166 return true if site==nil
166 return true if site==nil
167 return site.started
167 return site.started
168 else
168 else
169 return true
169 return true
170 end
170 end
171 end
171 end
172
172
173 def update_start_time
173 def update_start_time
174 stat = self.contest_stat
174 stat = self.contest_stat
175 - if stat == nil
175 + if stat == nil or stat.started_at == nil
176 - stat = UserContestStat.new(:user => self,
176 + stat ||= UserContestStat.new(:user => self)
177 - :started_at => Time.now.gmtime)
177 + stat.started_at = Time.now.gmtime
178 stat.save
178 stat.save
179 end
179 end
180 end
180 end
181
181
182 def problem_in_user_contests?(problem)
182 def problem_in_user_contests?(problem)
183 problem_contests = problem.contests.all
183 problem_contests = problem.contests.all
184
184
185 if problem_contests.length == 0 # this is public contest
185 if problem_contests.length == 0 # this is public contest
186 return true
186 return true
187 end
187 end
188
188
189 contests.each do |contest|
189 contests.each do |contest|
190 if problem_contests.find {|c| c.id == contest.id }
190 if problem_contests.find {|c| c.id == contest.id }
191 return true
191 return true
192 end
192 end
193 end
193 end
194 return false
194 return false
195 end
195 end
196
196
197 def available_problems_group_by_contests
197 def available_problems_group_by_contests
198 contest_problems = []
198 contest_problems = []
199 pin = {}
199 pin = {}
200 contests.enabled.each do |contest|
200 contests.enabled.each do |contest|
201 available_problems = contest.problems.available
201 available_problems = contest.problems.available
202 contest_problems << {
202 contest_problems << {
203 :contest => contest,
203 :contest => contest,
204 :problems => available_problems
204 :problems => available_problems
205 }
205 }
206 available_problems.each {|p| pin[p.id] = true}
206 available_problems.each {|p| pin[p.id] = true}
207 end
207 end
208 other_avaiable_problems = Problem.available.find_all {|p| pin[p.id]==nil and p.contests.length==0}
208 other_avaiable_problems = Problem.available.find_all {|p| pin[p.id]==nil and p.contests.length==0}
209 contest_problems << {
209 contest_problems << {
210 :contest => nil,
210 :contest => nil,
211 :problems => other_avaiable_problems
211 :problems => other_avaiable_problems
212 }
212 }
213 return contest_problems
213 return contest_problems
214 end
214 end
215
215
216 def available_problems
216 def available_problems
217 if not Configuration.multicontests?
217 if not Configuration.multicontests?
218 return Problem.find_available_problems
218 return Problem.find_available_problems
219 else
219 else
220 contest_problems = []
220 contest_problems = []
221 pin = {}
221 pin = {}
222 contests.enabled.each do |contest|
222 contests.enabled.each do |contest|
223 contest.problems.available.each do |problem|
223 contest.problems.available.each do |problem|
224 if not pin.has_key? problem.id
224 if not pin.has_key? problem.id
225 contest_problems << problem
225 contest_problems << problem
@@ -1,5 +1,10
1 class UserContestStat < ActiveRecord::Base
1 class UserContestStat < ActiveRecord::Base
2
2
3 belongs_to :user
3 belongs_to :user
4
4
5 + def reset_timer_and_save
6 + self.started_at = nil
7 + save
5 end
8 end
9 +
10 + end
@@ -1,60 +1,60
1 # This file is auto-generated from the current state of the database. Instead of editing this file,
1 # This file is auto-generated from the current state of the database. Instead of editing this file,
2 # please use the migrations feature of Active Record to incrementally modify your database, and
2 # please use the migrations feature of Active Record to incrementally modify your database, and
3 # then regenerate this schema definition.
3 # then regenerate this schema definition.
4 #
4 #
5 # Note that this schema.rb definition is the authoritative source for your database schema. If you need
5 # Note that this schema.rb definition is the authoritative source for your database schema. If you need
6 # to create the application database on another system, you should be using db:schema:load, not running
6 # to create the application database on another system, you should be using db:schema:load, not running
7 # all the migrations from scratch. The latter is a flawed and unsustainable approach (the more migrations
7 # all the migrations from scratch. The latter is a flawed and unsustainable approach (the more migrations
8 # you'll amass, the slower it'll run and the greater likelihood for issues).
8 # you'll amass, the slower it'll run and the greater likelihood for issues).
9 #
9 #
10 # It's strongly recommended to check this file into your version control system.
10 # It's strongly recommended to check this file into your version control system.
11
11
12 - ActiveRecord::Schema.define(:version => 20100303095700) do
12 + ActiveRecord::Schema.define(:version => 20100328123325) do
13
13
14 create_table "announcements", :force => true do |t|
14 create_table "announcements", :force => true do |t|
15 t.string "author"
15 t.string "author"
16 t.text "body"
16 t.text "body"
17 t.boolean "published"
17 t.boolean "published"
18 t.datetime "created_at"
18 t.datetime "created_at"
19 t.datetime "updated_at"
19 t.datetime "updated_at"
20 t.boolean "frontpage", :default => false
20 t.boolean "frontpage", :default => false
21 t.boolean "contest_only", :default => false
21 t.boolean "contest_only", :default => false
22 t.string "title"
22 t.string "title"
23 end
23 end
24
24
25 create_table "configurations", :force => true do |t|
25 create_table "configurations", :force => true do |t|
26 t.string "key"
26 t.string "key"
27 t.string "value_type"
27 t.string "value_type"
28 t.string "value"
28 t.string "value"
29 t.datetime "created_at"
29 t.datetime "created_at"
30 t.datetime "updated_at"
30 t.datetime "updated_at"
31 t.text "description"
31 t.text "description"
32 end
32 end
33
33
34 create_table "contests", :force => true do |t|
34 create_table "contests", :force => true do |t|
35 t.string "title"
35 t.string "title"
36 t.boolean "enabled"
36 t.boolean "enabled"
37 t.datetime "created_at"
37 t.datetime "created_at"
38 t.datetime "updated_at"
38 t.datetime "updated_at"
39 t.string "name"
39 t.string "name"
40 end
40 end
41
41
42 create_table "contests_problems", :id => false, :force => true do |t|
42 create_table "contests_problems", :id => false, :force => true do |t|
43 t.integer "contest_id"
43 t.integer "contest_id"
44 t.integer "problem_id"
44 t.integer "problem_id"
45 end
45 end
46
46
47 create_table "contests_users", :id => false, :force => true do |t|
47 create_table "contests_users", :id => false, :force => true do |t|
48 t.integer "contest_id"
48 t.integer "contest_id"
49 t.integer "user_id"
49 t.integer "user_id"
50 end
50 end
51
51
52 create_table "countries", :force => true do |t|
52 create_table "countries", :force => true do |t|
53 t.string "name"
53 t.string "name"
54 t.datetime "created_at"
54 t.datetime "created_at"
55 t.datetime "updated_at"
55 t.datetime "updated_at"
56 end
56 end
57
57
58 create_table "descriptions", :force => true do |t|
58 create_table "descriptions", :force => true do |t|
59 t.text "body"
59 t.text "body"
60 t.boolean "markdowned"
60 t.boolean "markdowned"
@@ -165,67 +165,68
165 t.string "source_filename"
165 t.string "source_filename"
166 end
166 end
167
167
168 add_index "submissions", ["user_id", "problem_id", "number"], :name => "index_submissions_on_user_id_and_problem_id_and_number", :unique => true
168 add_index "submissions", ["user_id", "problem_id", "number"], :name => "index_submissions_on_user_id_and_problem_id_and_number", :unique => true
169 add_index "submissions", ["user_id", "problem_id"], :name => "index_submissions_on_user_id_and_problem_id"
169 add_index "submissions", ["user_id", "problem_id"], :name => "index_submissions_on_user_id_and_problem_id"
170
170
171 create_table "tasks", :force => true do |t|
171 create_table "tasks", :force => true do |t|
172 t.integer "submission_id"
172 t.integer "submission_id"
173 t.datetime "created_at"
173 t.datetime "created_at"
174 t.integer "status"
174 t.integer "status"
175 t.datetime "updated_at"
175 t.datetime "updated_at"
176 end
176 end
177
177
178 create_table "test_pairs", :force => true do |t|
178 create_table "test_pairs", :force => true do |t|
179 t.integer "problem_id"
179 t.integer "problem_id"
180 t.text "input", :limit => 16777215
180 t.text "input", :limit => 16777215
181 t.text "solution", :limit => 16777215
181 t.text "solution", :limit => 16777215
182 t.datetime "created_at"
182 t.datetime "created_at"
183 t.datetime "updated_at"
183 t.datetime "updated_at"
184 end
184 end
185
185
186 create_table "test_requests", :force => true do |t|
186 create_table "test_requests", :force => true do |t|
187 t.integer "user_id"
187 t.integer "user_id"
188 t.integer "problem_id"
188 t.integer "problem_id"
189 t.integer "submission_id"
189 t.integer "submission_id"
190 t.string "input_file_name"
190 t.string "input_file_name"
191 t.string "output_file_name"
191 t.string "output_file_name"
192 t.string "running_stat"
192 t.string "running_stat"
193 t.integer "status"
193 t.integer "status"
194 t.datetime "updated_at"
194 t.datetime "updated_at"
195 t.datetime "submitted_at"
195 t.datetime "submitted_at"
196 t.datetime "compiled_at"
196 t.datetime "compiled_at"
197 t.text "compiler_message"
197 t.text "compiler_message"
198 t.datetime "graded_at"
198 t.datetime "graded_at"
199 t.string "grader_comment"
199 t.string "grader_comment"
200 t.datetime "created_at"
200 t.datetime "created_at"
201 t.float "running_time"
201 t.float "running_time"
202 t.string "exit_status"
202 t.string "exit_status"
203 t.integer "memory_usage"
203 t.integer "memory_usage"
204 end
204 end
205
205
206 add_index "test_requests", ["user_id", "problem_id"], :name => "index_test_requests_on_user_id_and_problem_id"
206 add_index "test_requests", ["user_id", "problem_id"], :name => "index_test_requests_on_user_id_and_problem_id"
207
207
208 create_table "user_contest_stats", :force => true do |t|
208 create_table "user_contest_stats", :force => true do |t|
209 t.integer "user_id"
209 t.integer "user_id"
210 t.datetime "started_at"
210 t.datetime "started_at"
211 t.datetime "created_at"
211 t.datetime "created_at"
212 t.datetime "updated_at"
212 t.datetime "updated_at"
213 + t.boolean "forced_logout"
213 end
214 end
214
215
215 create_table "users", :force => true do |t|
216 create_table "users", :force => true do |t|
216 t.string "login", :limit => 50
217 t.string "login", :limit => 50
217 t.string "full_name"
218 t.string "full_name"
218 t.string "hashed_password"
219 t.string "hashed_password"
219 t.string "salt", :limit => 5
220 t.string "salt", :limit => 5
220 t.string "alias"
221 t.string "alias"
221 t.string "email"
222 t.string "email"
222 t.integer "site_id"
223 t.integer "site_id"
223 t.integer "country_id"
224 t.integer "country_id"
224 t.boolean "activated", :default => false
225 t.boolean "activated", :default => false
225 t.datetime "created_at"
226 t.datetime "created_at"
226 t.datetime "updated_at"
227 t.datetime "updated_at"
227 end
228 end
228
229
229 add_index "users", ["login"], :name => "index_users_on_login", :unique => true
230 add_index "users", ["login"], :name => "index_users_on_login", :unique => true
230
231
231 end
232 end
@@ -1,40 +1,41
1
1
2 var Announcement = {
2 var Announcement = {
3
3
4 mostRecentId: 0,
4 mostRecentId: 0,
5
5
6 refreshUrl: '/main/announcements',
6 refreshUrl: '/main/announcements',
7
7
8 setMostRecentId: function(id) {
8 setMostRecentId: function(id) {
9 Announcement.mostRecentId = id;
9 Announcement.mostRecentId = id;
10 },
10 },
11
11
12 updateRecentId: function(id) {
12 updateRecentId: function(id) {
13 if(Announcement.mostRecentId < id)
13 if(Announcement.mostRecentId < id)
14 Announcement.mostRecentId = id;
14 Announcement.mostRecentId = id;
15 },
15 },
16
16
17 refreshAnnouncement: function() {
17 refreshAnnouncement: function() {
18 var url = Announcement.refreshUrl;
18 var url = Announcement.refreshUrl;
19 new Ajax.Request(url, {
19 new Ajax.Request(url, {
20 method: 'get',
20 method: 'get',
21 parameters: { recent: Announcement.mostRecentId },
21 parameters: { recent: Announcement.mostRecentId },
22 onSuccess: function(transport) {
22 onSuccess: function(transport) {
23 - if(transport.responseText.match(/\S/)!=null) {
23 + if((transport.status == 200) &&
24 + (transport.responseText.match(/\S/)!=null)) {
24 var announcementBody = $("announcementbox-body");
25 var announcementBody = $("announcementbox-body");
25 announcementBody.insert({ top: transport.responseText });
26 announcementBody.insert({ top: transport.responseText });
26 var announcementBoxes = $$(".announcementbox");
27 var announcementBoxes = $$(".announcementbox");
27 if(announcementBoxes.length!=0)
28 if(announcementBoxes.length!=0)
28 announcementBoxes[0].show();
29 announcementBoxes[0].show();
30 + Announcement.registerRefreshEventTimer();
29 }
31 }
30 }
32 }
31 });
33 });
32 - Announcement.registerRefreshEventTimer();
33 },
34 },
34
35
35 registerRefreshEventTimer: function() {
36 registerRefreshEventTimer: function() {
36 setTimeout(function () {
37 setTimeout(function () {
37 Announcement.refreshAnnouncement();
38 Announcement.refreshAnnouncement();
38 }, 30000);
39 }, 30000);
39 }
40 }
40 };
41 };
@@ -1,87 +1,117
1 require 'spec_helper'
1 require 'spec_helper'
2 require 'config_spec_helper'
2 require 'config_spec_helper'
3 require 'delorean'
3 require 'delorean'
4
4
5 describe "ContestManagements" do
5 describe "ContestManagements" do
6 include ConfigSpecHelperMethods
6 include ConfigSpecHelperMethods
7
7
8 fixtures :users
8 fixtures :users
9 fixtures :problems
9 fixtures :problems
10 fixtures :contests
10 fixtures :contests
11 fixtures :roles
11 fixtures :roles
12
12
13 before(:each) do
13 before(:each) do
14 @admin_user = users(:mary)
14 @admin_user = users(:mary)
15 @contest_b = contests(:contest_b)
15 @contest_b = contests(:contest_b)
16 @james = users(:james)
16 @james = users(:james)
17 @jack = users(:jack)
17 @jack = users(:jack)
18
18
19 set_contest_time_limit('3:00')
19 set_contest_time_limit('3:00')
20 set_indv_contest_mode
20 set_indv_contest_mode
21 + enable_multicontest
21 end
22 end
22
23
23 it "should reset users' timer when their contests change" do
24 it "should reset users' timer when their contests change" do
24 james_session = open_session
25 james_session = open_session
25 james_session.extend(MainSessionMethods)
26 james_session.extend(MainSessionMethods)
26
27
27 james_login_and_get_main_list(james_session)
28 james_login_and_get_main_list(james_session)
28 james_session.response.should_not have_text(/OVER/)
29 james_session.response.should_not have_text(/OVER/)
29
30
30 Delorean.time_travel_to(190.minutes.since) do
31 Delorean.time_travel_to(190.minutes.since) do
31 james_session.get_main_list
32 james_session.get_main_list
32 james_session.response.should have_text(/OVER/)
33 james_session.response.should have_text(/OVER/)
33
34
34 james_session.get '/' # logout
35 james_session.get '/' # logout
35 james_session.get '/main/list' # clearly log out
36 james_session.get '/main/list' # clearly log out
36 james_session.response.should_not render_template 'main/list'
37 james_session.response.should_not render_template 'main/list'
37
38
38 admin_change_users_contest_to("james", @contest_b, true)
39 admin_change_users_contest_to("james", @contest_b, true)
39
40
40 james_login_and_get_main_list(james_session)
41 james_login_and_get_main_list(james_session)
41 james_session.response.should_not have_text(/OVER/)
42 james_session.response.should_not have_text(/OVER/)
42 end
43 end
43 end
44 end
44
45
46 + it "should force users to log out when their contests change" do
47 + james_session = open_session
48 + james_session.extend(MainSessionMethods)
49 +
50 + james_login_and_get_main_list(james_session)
51 + james_session.response.should_not have_text(/OVER/)
52 +
53 + Delorean.time_travel_to(190.minutes.since) do
54 + james_session.get_main_list
55 + james_session.response.should have_text(/OVER/)
56 +
57 + admin_change_users_contest_to("james", @contest_b, true)
58 +
59 + james_session.get '/main/list'
60 + james_session.response.should_not render_template 'main/list'
61 + james_session.should be_redirect
62 +
63 + Delorean.time_travel_to(200.minutes.since) do
64 + james_login_and_get_main_list(james_session)
65 + james_session.response.should_not have_text(/OVER/)
66 + end
67 + end
68 + end
69 +
45 private
70 private
46
71
47 module MainSessionMethods
72 module MainSessionMethods
48 def login(login_name, password)
73 def login(login_name, password)
49 post '/login/login', :login => login_name, :password => password
74 post '/login/login', :login => login_name, :password => password
50 assert_redirected_to '/main/list'
75 assert_redirected_to '/main/list'
51 end
76 end
52
77
53 def get_main_list
78 def get_main_list
54 get '/main/list'
79 get '/main/list'
55 assert_template 'main/list'
80 assert_template 'main/list'
56 end
81 end
82 +
83 + def get_main_list_and_assert_logout
84 + get '/main/list'
85 + assert_redirected_to '/main'
86 + end
57 end
87 end
58
88
59 module ContestManagementSessionMethods
89 module ContestManagementSessionMethods
60 def change_users_contest_to(user_login_list, contest, reset_timer=false)
90 def change_users_contest_to(user_login_list, contest, reset_timer=false)
61 post_data = {
91 post_data = {
62 :contest => {:id => contest.id},
92 :contest => {:id => contest.id},
63 :operation => 'assign',
93 :operation => 'assign',
64 :login_list => user_login_list
94 :login_list => user_login_list
65 }
95 }
66 post_data[:reset_timer] = true if reset_timer
96 post_data[:reset_timer] = true if reset_timer
67 post '/user_admin/manage_contest', post_data
97 post '/user_admin/manage_contest', post_data
68 end
98 end
69 end
99 end
70
100
71 def admin_change_users_contest_to(user_list, contest, reset_timer)
101 def admin_change_users_contest_to(user_list, contest, reset_timer)
72 admin_session = open_session
102 admin_session = open_session
73 admin_session.extend(MainSessionMethods)
103 admin_session.extend(MainSessionMethods)
74 admin_session.extend(ContestManagementSessionMethods)
104 admin_session.extend(ContestManagementSessionMethods)
75
105
76 admin_session.login('mary','goodbye')
106 admin_session.login('mary','goodbye')
77 admin_session.get '/main/list'
107 admin_session.get '/main/list'
78 admin_session.change_users_contest_to(user_list, contest, reset_timer)
108 admin_session.change_users_contest_to(user_list, contest, reset_timer)
79 end
109 end
80
110
81 def james_login_and_get_main_list(session)
111 def james_login_and_get_main_list(session)
82 session.login('james', 'morning')
112 session.login('james', 'morning')
83 session.get_main_list
113 session.get_main_list
84 end
114 end
85
115
86 end
116 end
87
117
You need to be logged in to leave comments. Login now