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
@@ -8,58 +8,68
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?
@@ -1,36 +1,48
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
@@ -322,28 +322,35
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
@@ -175,71 +175,74
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'
@@ -310,34 +313,35
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
@@ -112,90 +112,90
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
@@ -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
8 + end
9 +
5 end
10 end
@@ -1,36 +1,36
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"
@@ -189,43 +189,44
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,80 +1,110
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
You need to be logged in to leave comments. Login now