Description:
add more problem stat
Commit status:
[Not Reviewed]
References:
Comments:
0 Commit comments 0 Inline Comments
Unresolved TODOs:
There are no unresolved TODOs
Add another comment

r457:c81651eecbbd - - 7 files changed: 222 inserted, 121 deleted

@@ -0,0 +1,44
1 + - param = {} unless param
2 + - graph_height = param[:graph_height] || 100
3 + - bar_width = param[:bar_width] || 14
4 + - graph_width = (bar_width * histogram[:data].count) + 20
5 + :css
6 + .hist_bar {
7 + width: #{bar_width-1}px;
8 + position: absolute;
9 + background-color: lightblue;
10 + }
11 + .hist_fill {
12 + width: #{bar_width-1}px;
13 + position: absolute;
14 + background-color: #eee;
15 + }
16 + .hist_text {
17 + position: absolute;
18 + font-size:5px;
19 + }
20 +
21 + %div{style: "position: relative; width: #{graph_width}px; height: 150px; background-color:#fff;" }
22 + //draw background
23 + - histogram[:data].each_index do |i|
24 + - height = histogram[:data][i] * graph_height / histogram[:summary][:max]
25 + - top = graph_height - height
26 + - left = graph_width - (i+1)*bar_width
27 + %div.hist_fill{style: "top: 0px; height: #{graph_height - height}px; left: #{left}px;" }
28 + // draw horizontal line
29 + - line = 3
30 + - line.times do |i|
31 + - top = graph_height - graph_height * (i+0.5)/ line
32 + %div{style: "position:absolute;width: #{graph_width-21}px;height: 1px;left: 20px;top:#{top}px;background-color: #333;"}
33 + %div.hist_text{style: "position:absolute;left: 0px;top:#{top-6}px"}
34 + =((i+0.5) * histogram[:summary][:max] / line).to_i
35 + // draw the actual bar and text
36 + - @histogram[:data].each_index do |i|
37 + - height = histogram[:data][i] * graph_height / histogram[:summary][:max]
38 + - top = graph_height - height
39 + - left = graph_width - (i+1)*bar_width
40 + %div.hist_bar{style: "top: #{top}px; height: #{height}px; left: #{left}px; dae: #{histogram[:data][i]}" }
41 + - if i % 7 == 1
42 + %div.hist_text{style: "top:#{graph_height + 5}px;left: #{left}px;"} #{(Time.zone.today - i.day).strftime('%-d')}
43 + - if (Time.now.in_time_zone - i.day).day == 15
44 + %div.hist_text{style: "top:#{graph_height + 15}px;left: #{left}px;"} #{(Time.zone.today - i.day).strftime('%b')}
@@ -105,101 +105,116
105 out_filename = "#{out_dirname}/#{@problem.name}.pdf"
105 out_filename = "#{out_dirname}/#{@problem.name}.pdf"
106 if FileTest.exists? out_filename
106 if FileTest.exists? out_filename
107 File.delete out_filename
107 File.delete out_filename
108 end
108 end
109
109
110 File.open(out_filename,"wb") do |file|
110 File.open(out_filename,"wb") do |file|
111 file.write(params[:file].read)
111 file.write(params[:file].read)
112 end
112 end
113 @problem.description_filename = "#{@problem.name}.pdf"
113 @problem.description_filename = "#{@problem.name}.pdf"
114 @problem.save
114 @problem.save
115 end
115 end
116 redirect_to :action => 'show', :id => @problem
116 redirect_to :action => 'show', :id => @problem
117 else
117 else
118 render :action => 'edit'
118 render :action => 'edit'
119 end
119 end
120 end
120 end
121
121
122 def destroy
122 def destroy
123 Problem.find(params[:id]).destroy
123 Problem.find(params[:id]).destroy
124 redirect_to :action => 'list'
124 redirect_to :action => 'list'
125 end
125 end
126
126
127 def toggle
127 def toggle
128 @problem = Problem.find(params[:id])
128 @problem = Problem.find(params[:id])
129 @problem.available = !(@problem.available)
129 @problem.available = !(@problem.available)
130 @problem.save
130 @problem.save
131 end
131 end
132
132
133 def turn_all_off
133 def turn_all_off
134 Problem.find(:all,
134 Problem.find(:all,
135 :conditions => "available = 1").each do |problem|
135 :conditions => "available = 1").each do |problem|
136 problem.available = false
136 problem.available = false
137 problem.save
137 problem.save
138 end
138 end
139 redirect_to :action => 'list'
139 redirect_to :action => 'list'
140 end
140 end
141
141
142 def turn_all_on
142 def turn_all_on
143 Problem.find(:all,
143 Problem.find(:all,
144 :conditions => "available = 0").each do |problem|
144 :conditions => "available = 0").each do |problem|
145 problem.available = true
145 problem.available = true
146 problem.save
146 problem.save
147 end
147 end
148 redirect_to :action => 'list'
148 redirect_to :action => 'list'
149 end
149 end
150
150
151 def stat
151 def stat
152 @problem = Problem.find(params[:id])
152 @problem = Problem.find(params[:id])
153 - if !@problem.available
153 + unless @problem.available or session[:admin]
154 redirect_to :controller => 'main', :action => 'list'
154 redirect_to :controller => 'main', :action => 'list'
155 - else
155 + return
156 - @submissions = Submission.includes(:user).where(problem_id: params[:id]).order(:user_id,:id)
157 end
156 end
157 + @submissions = Submission.includes(:user).where(problem_id: params[:id]).order(:user_id,:id)
158 +
159 + #stat summary
160 + range =65
161 + @histogram = { data: Array.new(range,0), summary: {} }
162 + @histogram[:data] = Array.new(range,0)
163 + user = Hash.new(0)
164 + @submissions.find_each do |sub|
165 + d = (DateTime.now.in_time_zone - sub.submitted_at) / 24 / 60 / 60
166 + @histogram[:data][d.to_i] += 1 if d < range
167 + user[sub.user_id] = [user[sub.user_id], (sub.points >= @problem.full_score) ? 1 : 0].max
168 + end
169 + @histogram[:summary][:max] = [@histogram[:data].max,1].max
170 +
171 + @summary = { attempt: user.count, solve: 0 }
172 + user.each_value { |v| @summary[:solve] += 1 if v == 1 }
158 end
173 end
159
174
160 def manage
175 def manage
161 @problems = Problem.find(:all, :order => 'date_added DESC')
176 @problems = Problem.find(:all, :order => 'date_added DESC')
162 end
177 end
163
178
164 def do_manage
179 def do_manage
165 if params.has_key? 'change_date_added'
180 if params.has_key? 'change_date_added'
166 change_date_added
181 change_date_added
167 elsif params.has_key? 'add_to_contest'
182 elsif params.has_key? 'add_to_contest'
168 add_to_contest
183 add_to_contest
169 elsif params.has_key? 'enable_problem'
184 elsif params.has_key? 'enable_problem'
170 set_available(true)
185 set_available(true)
171 elsif params.has_key? 'disable_problem'
186 elsif params.has_key? 'disable_problem'
172 set_available(false)
187 set_available(false)
173 end
188 end
174 redirect_to :action => 'manage'
189 redirect_to :action => 'manage'
175 end
190 end
176
191
177 def import
192 def import
178 @allow_test_pair_import = allow_test_pair_import?
193 @allow_test_pair_import = allow_test_pair_import?
179 end
194 end
180
195
181 def do_import
196 def do_import
182 old_problem = Problem.find_by_name(params[:name])
197 old_problem = Problem.find_by_name(params[:name])
183 if !allow_test_pair_import? and params.has_key? :import_to_db
198 if !allow_test_pair_import? and params.has_key? :import_to_db
184 params.delete :import_to_db
199 params.delete :import_to_db
185 end
200 end
186 @problem, import_log = Problem.create_from_import_form_params(params,
201 @problem, import_log = Problem.create_from_import_form_params(params,
187 old_problem)
202 old_problem)
188
203
189 if !@problem.errors.empty?
204 if !@problem.errors.empty?
190 render :action => 'import' and return
205 render :action => 'import' and return
191 end
206 end
192
207
193 if old_problem!=nil
208 if old_problem!=nil
194 flash[:notice] = "The test data has been replaced for problem #{@problem.name}"
209 flash[:notice] = "The test data has been replaced for problem #{@problem.name}"
195 end
210 end
196 @log = import_log
211 @log = import_log
197 end
212 end
198
213
199 def remove_contest
214 def remove_contest
200 problem = Problem.find(params[:id])
215 problem = Problem.find(params[:id])
201 contest = Contest.find(params[:contest_id])
216 contest = Contest.find(params[:contest_id])
202 if problem!=nil and contest!=nil
217 if problem!=nil and contest!=nil
203 problem.contests.delete(contest)
218 problem.contests.delete(contest)
204 end
219 end
205 redirect_to :action => 'manage'
220 redirect_to :action => 'manage'
@@ -212,49 +227,52
212 if defined? ALLOW_TEST_PAIR_IMPORT
227 if defined? ALLOW_TEST_PAIR_IMPORT
213 return ALLOW_TEST_PAIR_IMPORT
228 return ALLOW_TEST_PAIR_IMPORT
214 else
229 else
215 return false
230 return false
216 end
231 end
217 end
232 end
218
233
219 def change_date_added
234 def change_date_added
220 problems = get_problems_from_params
235 problems = get_problems_from_params
221 year = params[:date_added][:year].to_i
236 year = params[:date_added][:year].to_i
222 month = params[:date_added][:month].to_i
237 month = params[:date_added][:month].to_i
223 day = params[:date_added][:day].to_i
238 day = params[:date_added][:day].to_i
224 date = Date.new(year,month,day)
239 date = Date.new(year,month,day)
225 problems.each do |p|
240 problems.each do |p|
226 p.date_added = date
241 p.date_added = date
227 p.save
242 p.save
228 end
243 end
229 end
244 end
230
245
231 def add_to_contest
246 def add_to_contest
232 problems = get_problems_from_params
247 problems = get_problems_from_params
233 contest = Contest.find(params[:contest][:id])
248 contest = Contest.find(params[:contest][:id])
234 if contest!=nil and contest.enabled
249 if contest!=nil and contest.enabled
235 problems.each do |p|
250 problems.each do |p|
236 p.contests << contest
251 p.contests << contest
237 end
252 end
238 end
253 end
239 end
254 end
240
255
241 def set_available(avail)
256 def set_available(avail)
242 problems = get_problems_from_params
257 problems = get_problems_from_params
243 problems.each do |p|
258 problems.each do |p|
244 p.available = avail
259 p.available = avail
245 p.save
260 p.save
246 end
261 end
247 end
262 end
248
263
249 def get_problems_from_params
264 def get_problems_from_params
250 problems = []
265 problems = []
251 params.keys.each do |k|
266 params.keys.each do |k|
252 if k.index('prob-')==0
267 if k.index('prob-')==0
253 name, id, order = k.split('-')
268 name, id, order = k.split('-')
254 problems << Problem.find(id)
269 problems << Problem.find(id)
255 end
270 end
256 end
271 end
257 problems
272 problems
258 end
273 end
259
274
275 + def get_problems_stat
276 + end
277 +
260 end
278 end
@@ -60,136 +60,133
60 begin
60 begin
61 @until_time = DateTime.strptime(params[:until_datetime],date_and_time)
61 @until_time = DateTime.strptime(params[:until_datetime],date_and_time)
62 rescue
62 rescue
63 @until_time = DateTime.new(3000,1,1)
63 @until_time = DateTime.new(3000,1,1)
64 end
64 end
65
65
66 @submissions = {}
66 @submissions = {}
67
67
68 User.find_each do |user|
68 User.find_each do |user|
69 @submissions[user.id] = { login: user.login, full_name: user.full_name, count: 0, sub: { } }
69 @submissions[user.id] = { login: user.login, full_name: user.full_name, count: 0, sub: { } }
70 end
70 end
71
71
72 Submission.where("submitted_at >= ? AND submitted_at <= ?",@since_time,@until_time).find_each do |s|
72 Submission.where("submitted_at >= ? AND submitted_at <= ?",@since_time,@until_time).find_each do |s|
73 if @submissions[s.user_id]
73 if @submissions[s.user_id]
74 if not @submissions[s.user_id][:sub].has_key?(s.problem_id)
74 if not @submissions[s.user_id][:sub].has_key?(s.problem_id)
75 a = nil
75 a = nil
76 begin
76 begin
77 a = Problem.find(s.problem_id)
77 a = Problem.find(s.problem_id)
78 rescue
78 rescue
79 a = nil
79 a = nil
80 end
80 end
81 @submissions[s.user_id][:sub][s.problem_id] =
81 @submissions[s.user_id][:sub][s.problem_id] =
82 { prob_name: (a ? a.full_name : '(NULL)'),
82 { prob_name: (a ? a.full_name : '(NULL)'),
83 sub_ids: [s.id] }
83 sub_ids: [s.id] }
84 else
84 else
85 @submissions[s.user_id][:sub][s.problem_id][:sub_ids] << s.id
85 @submissions[s.user_id][:sub][s.problem_id][:sub_ids] << s.id
86 end
86 end
87 @submissions[s.user_id][:count] += 1
87 @submissions[s.user_id][:count] += 1
88 end
88 end
89 end
89 end
90 end
90 end
91
91
92 def problem_hof
92 def problem_hof
93 # gen problem list
93 # gen problem list
94 @user = User.find(session[:user_id])
94 @user = User.find(session[:user_id])
95 @problems = @user.available_problems
95 @problems = @user.available_problems
96
96
97 # get selected problems or the default
97 # get selected problems or the default
98 if params[:id]
98 if params[:id]
99 begin
99 begin
100 @problem = Problem.available.find(params[:id])
100 @problem = Problem.available.find(params[:id])
101 rescue
101 rescue
102 redirect_to action: :problem_hof
102 redirect_to action: :problem_hof
103 flash[:notice] = 'Error: submissions for that problem are not viewable.'
103 flash[:notice] = 'Error: submissions for that problem are not viewable.'
104 return
104 return
105 end
105 end
106 end
106 end
107
107
108 - if @problem
108 + return unless @problem
109 - #aggregrate by language
109 +
110 - @by_lang = {}
110 + @by_lang = {} #aggregrate by language
111 - Submission.where(problem_id: @problem.id).find_each do |sub|
112 - lang = Language.find_by_id(sub.language_id)
113 - next unless lang
114 - next unless sub.points >= @problem.full_score
115
111
116 - #initialize
112 + range =65
117 - unless @by_lang.has_key?(lang.pretty_name)
113 + @histogram = { data: Array.new(range,0), summary: {} }
118 - @by_lang[lang.pretty_name] = {
114 + @histogram[:data] = Array.new(range,0)
119 - runtime: { avail: false, value: 2**30-1 },
115 + @summary = {count: 0, solve: 0, attempt: 0}
120 - memory: { avail: false, value: 2**30-1 },
116 + user = Hash.new(0)
121 - length: { avail: false, value: 2**30-1 },
117 + Submission.where(problem_id: @problem.id).find_each do |sub|
122 - first: { avail: false, value: DateTime.new(3000,1,1) }
118 + #histogram
123 - }
119 + d = (DateTime.now.in_time_zone - sub.submitted_at) / 24 / 60 / 60
124 - end
120 + @histogram[:data][d.to_i] += 1 if d < range
125
121
126 - if sub.max_runtime and sub.max_runtime < @by_lang[lang.pretty_name][:runtime][:value]
122 + @summary[:count] += 1
127 - @by_lang[lang.pretty_name][:runtime] = {
123 + user[sub.user_id] = [user[sub.user_id], (sub.points >= @problem.full_score) ? 1 : 0].max
128 - avail: true,
129 - user_id: sub.user_id,
130 - value: sub.max_runtime,
131 - sub_id: sub.id
132 - }
133 - end
134
124
135 - if sub.peak_memory and sub.peak_memory < @by_lang[lang.pretty_name][:memory][:value]
125 + lang = Language.find_by_id(sub.language_id)
136 - @by_lang[lang.pretty_name][:memory] = {
126 + next unless lang
137 - avail: true,
127 + next unless sub.points >= @problem.full_score
138 - user_id: sub.user_id,
139 - value: sub.peak_memory,
140 - sub_id: sub.id
141 - }
142 - end
143
128
144 - if sub.submitted_at and sub.submitted_at < @by_lang[lang.pretty_name][:first][:value] and
129 + #initialize
145 - !sub.user.admin?
130 + unless @by_lang.has_key?(lang.pretty_name)
146 - @by_lang[lang.pretty_name][:first] = {
131 + @by_lang[lang.pretty_name] = {
147 - avail: true,
132 + runtime: { avail: false, value: 2**30-1 },
148 - user_id: sub.user_id,
133 + memory: { avail: false, value: 2**30-1 },
149 - value: sub.submitted_at,
134 + length: { avail: false, value: 2**30-1 },
150 - sub_id: sub.id
135 + first: { avail: false, value: DateTime.new(3000,1,1) }
151 - }
136 + }
152 - end
137 + end
153
138
154 - if @by_lang[lang.pretty_name][:length][:value] > sub.effective_code_length
139 + if sub.max_runtime and sub.max_runtime < @by_lang[lang.pretty_name][:runtime][:value]
155 - @by_lang[lang.pretty_name][:length] = {
140 + @by_lang[lang.pretty_name][:runtime] = { avail: true, user_id: sub.user_id, value: sub.max_runtime, sub_id: sub.id }
156 - avail: true,
157 - user_id: sub.user_id,
158 - value: sub.effective_code_length,
159 - sub_id: sub.id
160 - }
161 - end
162 - end
163 -
164 - #process user_id
165 - @by_lang.each do |lang,prop|
166 - prop.each do |k,v|
167 - v[:user] = User.exists?(v[:user_id]) ? User.find(v[:user_id]).full_name : "(NULL)"
168 - end
169 end
141 end
170
142
171 - #sum into best
143 + if sub.peak_memory and sub.peak_memory < @by_lang[lang.pretty_name][:memory][:value]
172 - if @by_lang and @by_lang.first
144 + @by_lang[lang.pretty_name][:memory] = { avail: true, user_id: sub.user_id, value: sub.peak_memory, sub_id: sub.id }
173 - @best = @by_lang.first[1].clone
145 + end
174 - @by_lang.each do |lang,prop|
146 +
175 - if @best[:runtime][:value] >= prop[:runtime][:value]
147 + if sub.submitted_at and sub.submitted_at < @by_lang[lang.pretty_name][:first][:value] and
176 - @best[:runtime] = prop[:runtime]
148 + !sub.user.admin?
177 - @best[:runtime][:lang] = lang
149 + @by_lang[lang.pretty_name][:first] = { avail: true, user_id: sub.user_id, value: sub.submitted_at, sub_id: sub.id }
178 - end
150 + end
179 - if @best[:memory][:value] >= prop[:memory][:value]
151 +
180 - @best[:memory] = prop[:memory]
152 + if @by_lang[lang.pretty_name][:length][:value] > sub.effective_code_length
181 - @best[:memory][:lang] = lang
153 + @by_lang[lang.pretty_name][:length] = { avail: true, user_id: sub.user_id, value: sub.effective_code_length, sub_id: sub.id }
182 - end
154 + end
183 - if @best[:length][:value] >= prop[:length][:value]
155 + end
184 - @best[:length] = prop[:length]
156 +
185 - @best[:length][:lang] = lang
157 + #process user_id
186 - end
158 + @by_lang.each do |lang,prop|
187 - if @best[:first][:value] >= prop[:first][:value]
159 + prop.each do |k,v|
188 - @best[:first] = prop[:first]
160 + v[:user] = User.exists?(v[:user_id]) ? User.find(v[:user_id]).full_name : "(NULL)"
189 - @best[:first][:lang] = lang
161 + end
190 - end
162 + end
163 +
164 + #sum into best
165 + if @by_lang and @by_lang.first
166 + @best = @by_lang.first[1].clone
167 + @by_lang.each do |lang,prop|
168 + if @best[:runtime][:value] >= prop[:runtime][:value]
169 + @best[:runtime] = prop[:runtime]
170 + @best[:runtime][:lang] = lang
171 + end
172 + if @best[:memory][:value] >= prop[:memory][:value]
173 + @best[:memory] = prop[:memory]
174 + @best[:memory][:lang] = lang
175 + end
176 + if @best[:length][:value] >= prop[:length][:value]
177 + @best[:length] = prop[:length]
178 + @best[:length][:lang] = lang
179 + end
180 + if @best[:first][:value] >= prop[:first][:value]
181 + @best[:first] = prop[:first]
182 + @best[:first][:lang] = lang
191 end
183 end
192 end
184 end
193 end
185 end
186 +
187 + @histogram[:summary][:max] = [@histogram[:data].max,1].max
188 + @summary[:attempt] = user.count
189 + user.each_value { |v| @summary[:solve] += 1 if v == 1 }
194 end
190 end
191 +
195 end
192 end
@@ -9,96 +9,103
9 validates_presence_of :full_name
9 validates_presence_of :full_name
10
10
11 scope :available, :conditions => {:available => true}
11 scope :available, :conditions => {:available => true}
12
12
13 DEFAULT_TIME_LIMIT = 1
13 DEFAULT_TIME_LIMIT = 1
14 DEFAULT_MEMORY_LIMIT = 32
14 DEFAULT_MEMORY_LIMIT = 32
15
15
16 def self.find_available_problems
16 def self.find_available_problems
17 Problem.available.all(:order => "date_added DESC, name ASC")
17 Problem.available.all(:order => "date_added DESC, name ASC")
18 end
18 end
19
19
20 def self.create_from_import_form_params(params, old_problem=nil)
20 def self.create_from_import_form_params(params, old_problem=nil)
21 org_problem = old_problem || Problem.new
21 org_problem = old_problem || Problem.new
22 import_params, problem = Problem.extract_params_and_check(params,
22 import_params, problem = Problem.extract_params_and_check(params,
23 org_problem)
23 org_problem)
24
24
25 if !problem.errors.empty?
25 if !problem.errors.empty?
26 return problem, 'Error importing'
26 return problem, 'Error importing'
27 end
27 end
28
28
29 problem.full_score = 100
29 problem.full_score = 100
30 problem.date_added = Time.new
30 problem.date_added = Time.new
31 problem.test_allowed = true
31 problem.test_allowed = true
32 problem.output_only = false
32 problem.output_only = false
33 problem.available = false
33 problem.available = false
34
34
35 if not problem.save
35 if not problem.save
36 return problem, 'Error importing'
36 return problem, 'Error importing'
37 end
37 end
38
38
39 import_to_db = params.has_key? :import_to_db
39 import_to_db = params.has_key? :import_to_db
40
40
41 importer = TestdataImporter.new(problem)
41 importer = TestdataImporter.new(problem)
42
42
43 if not importer.import_from_file(import_params[:file],
43 if not importer.import_from_file(import_params[:file],
44 import_params[:time_limit],
44 import_params[:time_limit],
45 import_params[:memory_limit],
45 import_params[:memory_limit],
46 import_params[:checker_name],
46 import_params[:checker_name],
47 import_to_db)
47 import_to_db)
48 problem.errors.add_to_base('Import error.')
48 problem.errors.add_to_base('Import error.')
49 end
49 end
50
50
51 return problem, importer.log_msg
51 return problem, importer.log_msg
52 end
52 end
53
53
54 def self.download_file_basedir
54 def self.download_file_basedir
55 return "#{Rails.root}/data/tasks"
55 return "#{Rails.root}/data/tasks"
56 end
56 end
57 +
58 + def get_submission_stat
59 + result = Hash.new
60 + #total number of submission
61 + result[:total_sub] = Submission.where(problem_id: self.id).count
62 + result[:attempted_user] = Submission.where(problem_id: self.id).group_by(:user_id)
63 + end
57
64
58 protected
65 protected
59
66
60 def self.to_i_or_default(st, default)
67 def self.to_i_or_default(st, default)
61 if st!=''
68 if st!=''
62 result = st.to_i
69 result = st.to_i
63 end
70 end
64 result ||= default
71 result ||= default
65 end
72 end
66
73
67 def self.to_f_or_default(st, default)
74 def self.to_f_or_default(st, default)
68 if st!=''
75 if st!=''
69 result = st.to_f
76 result = st.to_f
70 end
77 end
71 result ||= default
78 result ||= default
72 end
79 end
73
80
74 def self.extract_params_and_check(params, problem)
81 def self.extract_params_and_check(params, problem)
75 time_limit = Problem.to_f_or_default(params[:time_limit],
82 time_limit = Problem.to_f_or_default(params[:time_limit],
76 DEFAULT_TIME_LIMIT)
83 DEFAULT_TIME_LIMIT)
77 memory_limit = Problem.to_i_or_default(params[:memory_limit],
84 memory_limit = Problem.to_i_or_default(params[:memory_limit],
78 DEFAULT_MEMORY_LIMIT)
85 DEFAULT_MEMORY_LIMIT)
79
86
80 if time_limit<=0 or time_limit >60
87 if time_limit<=0 or time_limit >60
81 problem.errors.add_to_base('Time limit out of range.')
88 problem.errors.add_to_base('Time limit out of range.')
82 end
89 end
83
90
84 if memory_limit==0 and params[:memory_limit]!='0'
91 if memory_limit==0 and params[:memory_limit]!='0'
85 problem.errors.add_to_base('Memory limit format errors.')
92 problem.errors.add_to_base('Memory limit format errors.')
86 elsif memory_limit<=0 or memory_limit >512
93 elsif memory_limit<=0 or memory_limit >512
87 problem.errors.add_to_base('Memory limit out of range.')
94 problem.errors.add_to_base('Memory limit out of range.')
88 end
95 end
89
96
90 if params[:file]==nil or params[:file]==''
97 if params[:file]==nil or params[:file]==''
91 problem.errors.add_to_base('No testdata file.')
98 problem.errors.add_to_base('No testdata file.')
92 end
99 end
93
100
94 checker_name = 'text'
101 checker_name = 'text'
95 if ['text','float'].include? params[:checker]
102 if ['text','float'].include? params[:checker]
96 checker_name = params[:checker]
103 checker_name = params[:checker]
97 end
104 end
98
105
99 file = params[:file]
106 file = params[:file]
100
107
101 if !problem.errors.empty?
108 if !problem.errors.empty?
102 return nil, problem
109 return nil, problem
103 end
110 end
104
111
@@ -1,34 +1,51
1 :css
1 :css
2 .fix-width {
2 .fix-width {
3 font-family: "Consolas, Monaco, Droid Sans Mono,Mono, Monospace,Courier"
3 font-family: "Consolas, Monaco, Droid Sans Mono,Mono, Monospace,Courier"
4 }
4 }
5
5
6 %h1 Problem stat: #{@problem.name}
6 %h1 Problem stat: #{@problem.name}
7 %h2 Overview
7 %h2 Overview
8
8
9 +
10 + %table.info
11 + %thead
12 + %tr.info-head
13 + %th Stat
14 + %th Value
15 + %tbody
16 + %tr{class: cycle('info-even','info-odd')}
17 + %td Submissions
18 + %td= @submissions.count
19 + %tr{class: cycle('info-even','info-odd')}
20 + %td Solved/Attempted User
21 + %td #{@summary[:solve]}/#{@summary[:attempt]} (#{(@summary[:solve]*100.0/@summary[:attempt]).round(1)}%)
22 +
23 + %h2 Submissions Count
24 + = render partial: 'application/bar_graph', locals: { histogram: @histogram }
25 +
9 %h2 Submissions
26 %h2 Submissions
10 - if @submissions and @submissions.count > 0
27 - if @submissions and @submissions.count > 0
11 %table.info#main_table
28 %table.info#main_table
12 %thead
29 %thead
13 %tr.info-head
30 %tr.info-head
14 %th ID
31 %th ID
15 %th Login
32 %th Login
16 %th Name
33 %th Name
17 %th Submitted_at
34 %th Submitted_at
18 %th Points
35 %th Points
19 %th comment
36 %th comment
20 %tbody
37 %tbody
21 - row_odd,curr = true,''
38 - row_odd,curr = true,''
22 - @submissions.each do |sub|
39 - @submissions.each do |sub|
23 - next unless sub.user
40 - next unless sub.user
24 - row_odd,curr = !row_odd, sub.user if curr != sub.user
41 - row_odd,curr = !row_odd, sub.user if curr != sub.user
25 %tr{class: row_odd ? "info-odd" : "info-even"}
42 %tr{class: row_odd ? "info-odd" : "info-even"}
26 %td= link_to sub.id, controller: 'graders', action: 'submission', id: sub.id
43 %td= link_to sub.id, controller: 'graders', action: 'submission', id: sub.id
27 %td= link_to sub.user.login, controller: :users, action: :profile, id: sub.user.id
44 %td= link_to sub.user.login, controller: :users, action: :profile, id: sub.user.id
28 %td= sub.user.full_name
45 %td= sub.user.full_name
29 %td= time_ago_in_words(sub.submitted_at) + " ago"
46 %td= time_ago_in_words(sub.submitted_at) + " ago"
30 %td= sub.points
47 %td= sub.points
31 %td.fix-width= sub.grader_comment
48 %td.fix-width= sub.grader_comment
32 - else
49 - else
33 No submission
50 No submission
34
51
@@ -1,79 +1,96
1 :css
1 :css
2 .hof_user { color: orangered; font-style: italic; }
2 .hof_user { color: orangered; font-style: italic; }
3 .hof_language { color: green; font-style: italic; }
3 .hof_language { color: green; font-style: italic; }
4 .hof_value { color: deeppink;font-style: italic; }
4 .hof_value { color: deeppink;font-style: italic; }
5 + .info_param { font-weight: bold;text-align: right; }
5
6
6 - %h2 Overall of #{Problem.find(params[:id]).full_name}
7 + %h1 (#{Problem.find(params[:id]).name}) #{Problem.find(params[:id]).full_name}
8 +
9 + %h2 Problem Stat
10 + %table.info
11 + %thead
12 + %tr.info-head
13 + %th Stat
14 + %th Value
15 + %tbody
16 + %tr{class: cycle('info-even','info-odd')}
17 + %td.info_param Submissions
18 + %td= @summary[:count]
19 + %tr{class: cycle('info-even','info-odd')}
20 + %td.info_param Solved/Attempted User
21 + %td #{@summary[:solve]}/#{@summary[:attempt]} (#{(@summary[:solve]*100.0/@summary[:attempt]).round(1)}%)
22 + - if @best
23 + %tr{class: cycle('info-even','info-odd')}
24 + %td.info_param Best Runtime
25 + %td
26 + by #{link_to @best[:runtime][:user], controller:'users', action:'profile', id:@best[:memory][:user_id]}
27 + using <span class="hof_language">#{@best[:runtime][:lang]}</span>
28 + with <span class="hof_value">#{@best[:runtime][:value] * 1000} milliseconds</span>
29 + at submission
30 + = link_to("#" + @best[:runtime][:sub_id].to_s, controller: 'graders', action: 'submission', id:@best[:runtime][:sub_id])
31 +
32 + %tr{class: cycle('info-even','info-odd')}
33 + %td.info_param Best Memory Usage
34 + %td
35 + by #{link_to @best[:memory][:user], controller:'users', action:'profile', id:@best[:memory][:user_id]}
36 + using <span class="hof_language">#{@best[:memory][:lang]}</span>
37 + with <span class="hof_value">#{number_with_delimiter(@best[:memory][:value])} kbytes </span>
38 + at submission
39 + = link_to("#" + @best[:memory][:sub_id].to_s, controller: 'graders' , action: 'submission', id:@best[:memory][:sub_id])
40 +
41 + %tr{class: cycle('info-even','info-odd')}
42 + %td.info_param Shortest Code
43 + %td
44 + by #{link_to @best[:length][:user], controller:'users', action:'profile', id:@best[:length][:user_id]}
45 + using <span class="hof_language">#{@best[:length][:lang]}</span>
46 + with <span class="hof_value">#{@best[:length][:value]} bytes</span>
47 + at submission
48 + = link_to("#" + @best[:length][:sub_id].to_s, controller: 'graders' , action: 'submission', id: @best[:length][:sub_id])
49 +
50 + %tr{class: cycle('info-even','info-odd')}
51 + %td.info_param First solver
52 + %td
53 + #{link_to @best[:first][:user], controller:'users', action:'profile', id:@best[:first][:user_id]} is the first solver
54 + using <span class="hof_language">#{@best[:first][:lang]}</span>
55 + on <span class="hof_value">#{@best[:first][:value]}</span>
56 + at submission
57 + = link_to("#" + @best[:first][:sub_id].to_s, controller: 'graders' , action: 'submission', id: @best[:first][:sub_id])
58 +
59 +
60 + %p
61 + This counts only for submission with 100% score <br/>
62 + Right now, java is excluded from memory usage competition. (Because it always uses 2GB memory...)
7
63
8 - if @best
64 - if @best
9 - %b Best Runtime:
10 - by #{link_to @best[:runtime][:user], controller:'users', action:'profile', id:@best[:memory][:user_id]}
11 - using <span class="hof_language">#{@best[:runtime][:lang]}</span>
12 - with <span class="hof_value">#{@best[:runtime][:value] * 1000} milliseconds</span>
13 - at submission
14 - = link_to("#" + @best[:runtime][:sub_id].to_s, controller: 'graders', action: 'submission', id:@best[:runtime][:sub_id])
15 - %br/
16 -
17 - %b Best Memory Usage:
18 - by #{link_to @best[:memory][:user], controller:'users', action:'profile', id:@best[:memory][:user_id]}
19 - using <span class="hof_language">#{@best[:memory][:lang]}</span>
20 - with <span class="hof_value">#{number_with_delimiter(@best[:memory][:value])} kbytes </span>
21 - at submission
22 - = link_to("#" + @best[:memory][:sub_id].to_s, controller: 'graders' , action: 'submission', id:@best[:memory][:sub_id])
23 - %br/
24 -
25 - %b Shortest Code:
26 - by #{link_to @best[:length][:user], controller:'users', action:'profile', id:@best[:length][:user_id]}
27 - using <span class="hof_language">#{@best[:length][:lang]}</span>
28 - with <span class="hof_value">#{@best[:length][:value]} bytes</span>
29 - at submission
30 - = link_to("#" + @best[:length][:sub_id].to_s, controller: 'graders' , action: 'submission', id: @best[:length][:sub_id])
31 - %br/
32 -
33 - %b First solver:
34 - #{link_to @best[:first][:user], controller:'users', action:'profile', id:@best[:first][:user_id]} is the first solver
35 - using <span class="hof_language">#{@best[:first][:lang]}</span>
36 - on <span class="hof_value">#{@best[:first][:value]}</span>
37 - at submission
38 - = link_to("#" + @best[:first][:sub_id].to_s, controller: 'graders' , action: 'submission', id: @best[:first][:sub_id])
39 - %br/
40 -
41 -
42 - %p
43 - This counts only for submission with 100% score <br/>
44 - Right now, java is excluded from memory usage competition. (Because it always uses 2GB memory...)
45 -
46 %h2 By language
65 %h2 By language
47
66
48 %table.info
67 %table.info
49 %thead
68 %thead
50 %tr.info-head
69 %tr.info-head
51 %th Language
70 %th Language
52 %th Best runtime (ms)
71 %th Best runtime (ms)
53 %th Best memory (kbytes)
72 %th Best memory (kbytes)
54 %th Shortest Code (bytes)
73 %th Shortest Code (bytes)
55 %th First solver
74 %th First solver
56 %tbody
75 %tbody
57 - @by_lang.each do |lang,value|
76 - @by_lang.each do |lang,value|
58 %tr{class: cycle('info-even','info-odd')}
77 %tr{class: cycle('info-even','info-odd')}
59 %td= lang
78 %td= lang
60 %td
79 %td
61 = link_to value[:runtime][:user], controller: 'users', action: 'profile', id: value[:runtime][:user_id]
80 = link_to value[:runtime][:user], controller: 'users', action: 'profile', id: value[:runtime][:user_id]
62 = "(#{(value[:runtime][:value] * 1000).to_i} @"
81 = "(#{(value[:runtime][:value] * 1000).to_i} @"
63 = "#{link_to("#" + value[:runtime][:sub_id].to_s, controller: 'graders' , action: 'submission', id: value[:runtime][:sub_id])} )".html_safe
82 = "#{link_to("#" + value[:runtime][:sub_id].to_s, controller: 'graders' , action: 'submission', id: value[:runtime][:sub_id])} )".html_safe
64 %td
83 %td
65 = link_to value[:memory][:user], controller: 'users', action: 'profile', id: value[:memory][:user_id]
84 = link_to value[:memory][:user], controller: 'users', action: 'profile', id: value[:memory][:user_id]
66 = "(#{number_with_delimiter(value[:memory][:value])} @"
85 = "(#{number_with_delimiter(value[:memory][:value])} @"
67 = "#{link_to("#" + value[:memory][:sub_id].to_s, controller: 'graders' , action: 'submission', id: value[:memory][:sub_id])} )".html_safe
86 = "#{link_to("#" + value[:memory][:sub_id].to_s, controller: 'graders' , action: 'submission', id: value[:memory][:sub_id])} )".html_safe
68 %td
87 %td
69 = link_to value[:length][:user], controller: 'users', action: 'profile', id: value[:length][:user_id]
88 = link_to value[:length][:user], controller: 'users', action: 'profile', id: value[:length][:user_id]
70 = "(#{value[:length][:value]} @"
89 = "(#{value[:length][:value]} @"
71 = "#{link_to("#" + value[:length][:sub_id].to_s, controller: 'graders' , action: 'submission', id: value[:length][:sub_id])} )".html_safe
90 = "#{link_to("#" + value[:length][:sub_id].to_s, controller: 'graders' , action: 'submission', id: value[:length][:sub_id])} )".html_safe
72 %td
91 %td
73 - if value[:first][:user] != '(NULL)' #TODO: i know... this is wrong...
92 - if value[:first][:user] != '(NULL)' #TODO: i know... this is wrong...
74 = link_to value[:first][:user], controller: 'users', action: 'profile', id: value[:first][:user_id]
93 = link_to value[:first][:user], controller: 'users', action: 'profile', id: value[:first][:user_id]
75 = "(#{value[:first][:value]} @"
94 = "(#{value[:first][:value]} @"
76 = "#{link_to("#" + value[:first][:sub_id].to_s, controller: 'graders' , action: 'submission', id: value[:first][:sub_id])} )".html_safe
95 = "#{link_to("#" + value[:first][:sub_id].to_s, controller: 'graders' , action: 'submission', id: value[:first][:sub_id])} )".html_safe
77
96
78 - - else
79 - %h3 No submissions
@@ -1,22 +1,23
1
1
2 /- if params[:id]
2 /- if params[:id]
3 / %h1 Tasks Hall of Fame
3 / %h1 Tasks Hall of Fame
4 / = link_to('[back to All-Time Hall of Fame]', action: 'problem_hof', id: nil )
4 / = link_to('[back to All-Time Hall of Fame]', action: 'problem_hof', id: nil )
5 /- else
5 /- else
6 / %h1 All-Time Hall of Fame
6 / %h1 All-Time Hall of Fame
7
7
8
8
9 - %h1 Tasks Hall of Fame
9 + %h1 Hall of Fame
10 .task-menu
10 .task-menu
11 Tasks
11 Tasks
12 %br/
12 %br/
13 - @problems.each do |prob|
13 - @problems.each do |prob|
14 = link_to( "[#{prob.name}]", {id: prob.id})
14 = link_to( "[#{prob.name}]", {id: prob.id})
15
15
16 - unless params[:id]
16 - unless params[:id]
17 /=render partial: 'all_time_hof'
17 /=render partial: 'all_time_hof'
18 Please select a problem.
18 Please select a problem.
19 - else
19 - else
20 =render partial: 'task_hof'
20 =render partial: 'task_hof'
21 + %h2 Submission History
22 + =render partial: 'application/bar_graph', locals: { histogram: @histogram }
21
23
22 -
You need to be logged in to leave comments. Login now